队列:
1.求区间所有后缀最大值的位置(单调队列)
搜索:
1.天下第一(记忆化)
2.拯救oibh总部(DFS+连通性问题)
3.国王的魔镜(递归)
4.回家(BFS+三维的标记)
5.取数游戏(DFS)
6.数的划分(递归)
给定一个长度为 �n 的数列 �a,对于其中每个长度为 �k 的子区间,请你求出这个这个子区间构成的数列的所有后缀最大值的位置个数。
一个下标 �i 是是数列 �b 的后缀最大值下标当且仅当:对于所有的 �<�≤∣�∣i
第一行是两个整数,依次表示操作次数 �n 和子区间长度 �k。
第二行有 �n 个整数,第 �i 个整数表示 ��ai。
共输出 �−�+1n−k+1 行每行一个整数,按左端点从小到大的顺序依次输出每个子区间构成的数列的后缀最大值位置个数。
输入 #1复制
5 3 2 1 3 5 4
输出 #1复制
1 1 2
第一个子数列:2,1,32,1,3。其中 33 是后缀最大值。
第二个子数列:1,3,51,3,5,其中 55 是后缀最大值。
第三个子数列:3,5,43,5,4,其中 55 和 44 是后缀最大值。
对于全部的测试点,保证 1≤�≤�≤1061≤k≤n≤106,1≤��<2641≤xi<264。
思路:本质上是找单调队列的队内元素个数
#include
using namespace std;
unsigned long long a[1000005];
int main()
{
int n,k;
cin>>n>>k;
for (int i=1;i<=n;++i)cin>>a[i];
dequeq;
for (int i=1;i<=n;++i)
{
while (!q.empty() && a[q.back()]=k)q.pop_front();
q.push_back(i);
if (i>=k)cout<
天下第一的 cbw 以主席的身份在 8102 年统治全宇宙后,开始了自己休闲的生活,并邀请自己的好友每天都来和他做游戏。由于 cbw 想要显出自己平易近人,所以 zhouwc 虽然是一个蒟蒻,也有能和 cbw 玩游戏的机会。
游戏是这样的:
给定两个数 �x,�y,与一个模数 �p。
cbw 拥有数 �x,zhouwc 拥有数 �y。
第一个回合:�←(�+�) mod �x←(x+y)modp。
第二个回合:�←(�+�) mod �y←(x+y)modp。
第三个回合:�←(�+�) mod �x←(x+y)modp。
第四个回合:�←(�+�) mod �y←(x+y)modp。
以此类推....
如果 �x 先到 00,则 cbw 胜利。如果 �y 先到 00,则 zhouwc 胜利。如果 �,�x,y 都不能到 00,则为平局。
cbw 为了捍卫自己主席的尊严,想要提前知道游戏的结果,并且可以趁机动点手脚,所以他希望你来告诉他结果。
有多组数据。
第一行:�T 和 �p 表示一共有 �T 组数据且模数都为 �p。
以下 �T 行,每行两个数 �,�x,y。
共 �T 行
11 表示 cbw 获胜,22 表示 zhouwc 获胜,error
表示平局。
输入 #1复制
1 10 1 3
输出 #1复制
error
输入 #2复制
1 10 4 5
输出 #2复制
1
1≤�≤2001≤T≤200。
1≤�,�,�≤100001≤x,y,p≤10000。
思路:用记忆化搜索,不需要管它每一轮的变化,把x,y一直变。谁先到0谁就赢
#include
using namespace std;
short vis[10010][10010];
int t,mod;
int f(int x ,int y)
{
if (vis[x][y])return vis[x][y];
if (vis[x][y]==-1)return -1;
vis[x][y]=-1;
if (!x)return 1;
if (!y)return 2;
int num=(x+y)%mod;
vis[x][y]=f(num,(num+y)%mod);
return vis[x][y];
}
int main()
{
cin>>t>>mod;
while (t--)
{
int x,y;
cin>>x>>y;
int ans=f(x,y);
if (ans==-1)cout<<"error"<
oibh 总部突然被水淹没了!现在需要你的救援……
oibh 被突来的洪水淹没了,还好 oibh 总部有在某些重要的地方起一些围墙。用 *
号表示,而一个四面被围墙围住的区域洪水是进不去的。
oibh 总部内部也有许多重要区域,每个重要区域在图中用一个 0
表示。
现在给出 oibh 的围墙建设图,问有多少个没被洪水淹到的重要区域。
第一行为两个正整数 �,�x,y。
接下来 �x 行,每行 �y 个整数,由 *
和 0
组成,表示 oibh 总部的建设图。
输出没被水淹没的 oibh 总部的 0
的数量。
输入 #1复制
4 5 00000 00*00 0*0*0 00*00
输出 #1复制
1
输入 #2复制
5 5 ***** *0*0* **0** *0*0* *****
输出 #2复制
5
对于 100%100% 的数据,1≤�,�≤5001≤x,y≤500。
思路:考察搜索的连通性问题,以边界为起点,搜索一边,把0全部变成*,最后再找剩余0的个数
#include
using namespace std;
char mp[550][550];
int vis[550][550];
int n,m;
void dfs(int x ,int y)
{
if (mp[x][y]=='*')return;
mp[x][y]='*';
vis[x][y]=1;
int dir[4][2]={{0,1},{1,0},{0,-1},{-1,0}};
for (int i=0;i<4;++i)
{
int tx=x+dir[i][0],ty=y+dir[i][1];
if (tx<0|| ty<0|| tx>n || ty>m)continue;
if (mp[tx][ty]=='0' &&vis[tx][ty]==0)
{
dfs(tx,ty);
}
}
vis[x][y]=0;
}
int main()
{
cin>>n>>m;
for (int i=0;i>mp[i];
for (int i = 0; i < n; ++i)
{
dfs(i, 0);
dfs(i, m - 1);
}
for (int j = 0; j < m; ++j)
{
dfs(0, j);
dfs(n - 1, j);
}
int cnt=0;
for (int i=0;i
国王有一个魔镜,可以把任何接触镜面的东西变成原来的两倍——只是,因为是镜子嘛,增加的那部分是反的。比如一条项链,我们用AB来表示,不同的字母表示不同颜色的珍珠。如果把B端接触镜面的话,魔镜会把这条项链变为ABBA。如果再用一端接触的话,则会变成ABBAABBA(假定国王只用项链的某一端接触魔镜)。给定最终的项链,请编写程序输出国王没使用魔镜之前,最初的项链可能的最小长度。
只有一个字符串,由大写英文字母组成(字母数<=100000),表示最终的项链。
只有一个整数,表示国王没使用魔镜前,最初的项链可能的最小长度。
输入 #1复制
ABBAABBA
输出 #1复制
2
思路:递归
#include
using namespace std;
int f(string &s,int length)
{
if (length==1)return length;
int idx=length/2-1;
string qian=s.substr(0,idx+1);
string hou=s.substr(idx+1);
reverse(hou.begin(),hou.end());
if (qian==hou)
{
return f(qian,length/2);
}
else
{
return length;
}
}
int main()
{
string s;
cin>>s;
int length=s.length();
length=f(s,length);
cout<
回家https://www.luogu.com.cn/problem/P2802
小 H 在一个划分成了 �×�n×m 个方格的长方形封锁线上。 每次他能向上下左右四个方向移动一格(当然小 H 不可以静止不动), 但不能离开封锁线,否则就被打死了。 刚开始时他有满血 66 点,每移动一格他要消耗 11 点血量。一旦小 H 的血量降到 00, 他将死去。 他可以沿路通过拾取鼠标(什么鬼。。。)来补满血量。只要他走到有鼠标的格子,他不需要任何时间即可拾取。格子上的鼠标可以瞬间补满,所以每次经过这个格子都有鼠标。就算到了某个有鼠标的格子才死去, 他也不能通过拾取鼠标补满 HP。 即使在家门口死去, 他也不能算完成任务回到家中。
地图上有五种格子:
0
:障碍物。
1
:空地, 小 H 可以自由行走。
2
:小 H 出发点, 也是一片空地。
3
:小 H 的家。
4
:有鼠标在上面的空地。
小 H 能否安全回家?如果能, 最短需要多长时间呢?
第一行两个整数 �,�n,m, 表示地图的大小为 �×�n×m。
下面 �n 行, 每行 �m 个数字来描述地图。
一行, 若小 H 不能回家, 输出 -1
,否则输出他回家所需最短时间。
输入 #1复制
3 3 2 1 1 1 1 0 1 1 3
输出 #1复制
4
对于所有数据,1≤�,�≤91≤n,m≤9。
思路:BFS,但是由于还多了一个血量,所以在建立标记数组的时候应该用三维的,因为可能到达同一个点,但是血量不同
#include
using namespace std;
struct Node{
int xueliang;
int x;
int y;
int t;
};
int vis[10][10][7];
int mp[10][10];
int main()
{
int n,m;
cin>>n>>m;
int start_x,start_y;
for (int i=0;i>mp[i][j];
if (mp[i][j]==2)
{
start_x=i,start_y=j;
}
}
}
queueq;
int dir[4][2]={{0,1},{1,0},{-1,0},{0,-1}};
q.push(Node{6,start_x,start_y,0});
vis[start_x][start_y][6]=1;
while (!q.empty())
{
Node news=q.front();
q.pop();
for (int i=0;i<4;++i)
{
int tx=news.x+dir[i][0],ty=news.y+dir[i][1];
if (tx<0 || ty<0 || tx>n ||ty>m || vis[tx][ty][news.xueliang-1])continue;
if (mp[tx][ty]==1 && news.xueliang>1 )
{
Node olds=Node{news.xueliang-1,tx,ty,news.t+1};
q.push(olds);
vis[tx][ty][news.xueliang-1]=1;
}
else if (mp[tx][ty]==3&& news.xueliang>1)
{
cout<1)
{
Node olds=Node{6,tx,ty,news.t+1};
vis[tx][ty][6]=1;
q.push(olds);
}
else if (mp[tx][ty]==4 && news.xueliang<=1)
{
continue;
}
else if (news.xueliang==0)
{
continue;
}
}
}
cout<<-1;
return 0;
}
一个 �×�N×M 的由非负整数构成的数字矩阵,你需要在其中取出若干个数字,使得取出的任意两个数字不相邻(若一个数字在另外一个数字相邻 88 个格子中的一个即认为这两个数字相邻),求取出数字和最大是多少。
第一行有一个正整数 �T,表示了有 �T 组数据。
对于每一组数据,第一行有两个正整数 �N 和 �M,表示了数字矩阵为 �N 行 �M 列。
接下来 �N 行,每行 �M 个非负整数,描述了这个数字矩阵。
共 �T 行,每行一个非负整数,输出所求得的答案。
输入 #1复制
3 4 4 67 75 63 10 29 29 92 14 21 68 71 56 8 67 91 25 2 3 87 70 85 10 3 17 3 3 1 1 1 1 99 1 1 1 1
输出 #1复制
271 172 99
对于第一组数据,取数方式如下:
[67]7563102929[92]14[21]687156867[91]25[67]29[21]87529686763[92]71[91]10145625
思路:DFS加剪枝
#include
using namespace std;
int t,n,m;
int mp[8][8];
int dir[8][2]={{0,1},{1,1},{1,0},{1,-1},{0,-1},{-1,-1},{-1,0},{-1,1}};
int vis[8][8];
int maxn=0;
void dfs(int x,int y,int sum)
{
if (x==n+1)
{
maxn=max(maxn,sum);
return ;
}
if (y==m+1)
{
dfs(x+1,1,sum);
return;
}
dfs(x,y+1,sum);
if (vis[x][y]==0)
{
for (int i=0;i<8;++i)
{
int tx=x+dir[i][0],ty=y+dir[i][1];
vis[tx][ty]++;
}
dfs(x,y+1,sum+mp[x][y]);
for (int i=0;i<8;++i)
{
int tx=x+dir[i][0],ty=y+dir[i][1];
vis[tx][ty]--;
}
}
}
int main()
{
int t;
scanf("%d",&t);
while (t--)
{
memset(mp,0,sizeof(mp));
memset(vis,0,sizeof(vis));
scanf("%d%d",&n,&m);
for (int i=1;i<=n;++i)
{
for (int j=1;j<=m;++j)
{
scanf("%d",&mp[i][j]);
}
}
dfs(1,1,0);
printf("%d\n",maxn);
maxn=0;
}
}
将整数 �n 分成 �k 份,且每份不能为空,任意两个方案不相同(不考虑顺序)。
例如:�=7n=7,�=3k=3,下面三种分法被认为是相同的。
1,1,51,1,5;
1,5,11,5,1;
5,1,15,1,1.
问有多少种不同的分法。
�,�n,k (6<�≤2006 11 个整数,即不同的分法。 输入 #1复制 输出 #1复制 四种分法为: 思路:DFS,类似全排但是要剪枝输出格式
输入输出样例
7 3
4
说明/提示
1,1,51,1,5;
1,2,41,2,4;
1,3,31,3,3;
2,2,32,2,3.#include