小猫排队
题目描述
世界上最苦恼的事情莫过于排队了,特别是排在你前面的猫比你可爱的时候。----《论猫的自我修养》
小猫啾啾现在就很苦恼,它排在队伍的末尾处等着买酱油,前面还有足足 只猫咪。但幸运的是小猫啾啾会一种魔法:它可以和前面距离它最近且比它可爱(可爱值大于啾啾)的小猫交换位置(被交换的小猫会被传送到啾啾之前的位置)。
已知啾啾每一分钟开始时可以施展一次魔法,而每一分钟过后排在队伍最前面的猫咪就会离开队伍(这意味这啾啾会先交换位置然后队伍才开始移动)。 因为等会还得去买饺子所以啾啾会尽可能地与自身前方比它可爱且未出队的小猫交换位置(可以证明交换后必定更快买到酱油),现在啾啾想请你帮它计算出它需要多久才能买到酱油离开。
思路
利用stl中的list双向链表来存储比小猫可爱值大的其他小猫的位置,令另外一个变量i记录队首已经离开的小猫的下表,开始时每一次交换位置时候,记录现在小猫的位置,弹出尾部元素,然后令i++,之后判断链表是否为空并且如今的链表第一个小猫的位置是否比i小,如果小(一般应该只会小1),就代表当前第一个小猫位置该要离开了,所以弹出链表头部元素。无论是否弹出该元素,都令记录时间变量ans+1; 最终当该链表空时,将现在小猫的位置和i记录的位置相减后加1 然后加上一直记录时间的变量ans,最终得出答案
代码
#include<bits/stdc++.h>
using namespace std;
using ll = long long ;
const int N = 2e5 + 10;
int a[N],n,A;
list<int>p;
int main(){
int n;
cin >> n;
for(int i = 1; i <= n; i++){
cin >>a[i];
}
cin >> A;
for(int i = 1; i <= n; i++){
if(a[i] > A)
p.push_back(i);
}
int i = 1, ans = 0,now_pos = n + 1;
while(p.size()){
now_pos =p.back();
p.pop_back();
i++;
if(p.size() && p.front() < i)
p.pop_front();
ans++;
}
ans = ans + now_pos - i + 1;
cout <<ans <<endl;
}
造桥
题目描述
今天是愉快的周末,牛可乐和牛妹在一起玩游戏。牛妹给了牛可乐 个造桥的零件,每个零件以字符串的形式给出,每个字符串两端的字符是零件的接口,两个零件可以通过连接不同端的接口(一个零件的左端只能连接另一个零件的右端,右端则只能连接左端)得到一个更长的零件,新零件的长度是原零件的长度之和。 牛妹规定,零件不能翻转,且零件左边的接口只能由该零件左边的零件去连接(这意味着,应该按顺序选取零件),右端同理。 牛可乐想得到牛妹的赞许,因此他要想用牛妹给的零件拼接一个尽可能长的桥梁,你能帮帮他吗?
思路
不去考虑字符串左端还是右端,只考虑以该字符结尾的字符串的最大长度,建立数组len[30],因为英文字符总共26,种 然后每次输入字符串的时候,判断 1、不选当前字符串,以当前字符串最后字符结尾的长度len[s.back() -‘a’] 2、选当前字符串,当必须是前面有以该字符开头元素结尾的字符串 即 len[s.front() - ‘a’ ] + (int)s.length(); 判断上面两个选项谁大, 大的即为当前的len[s.back() -‘a’]
然后遍历len数组,找到最大值,输出。
代码
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
int len[30];
int main(){
int t;
cin >> t;
while(t--){
memset(len,0,sizeof len);
int n; cin >> n;
for(int i = 0 ; i < n; i++){
string s;cin >>s;
len[s.back() - 'a'] = max(len[s.back() - 'a'],len[s.front() - 'a'] +(int) s.length());
}
int ans = 0 ;
for(int i = 0 ; i < 26; i++)
ans = max(ans, len[i]);
cout << ans << endl;
}
}
|