2020 第十一届蓝桥杯大赛软件赛决赛,国赛,C/C++大学B组题解

Posted 小哈里

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了2020 第十一届蓝桥杯大赛软件赛决赛,国赛,C/C++大学B组题解相关的知识,希望对你有一定的参考价值。

文章目录

试题地址:https://www.lanqiao.cn/courses/2786/learning/?id=131138
补题地址:http://lx.lanqiao.cn/problemsets.page

第1题——美丽的2 (5分)

  • 题目:

  • 直接暴力1-2020,然后判断一下有没有2即可。

  • 答案是563

    #include<bits/stdc++.h>
    using namespace std;
    
    int main()
    	int res = 0;
    	for(int i = 1; i <= 2020; i++)
    		int x = i, cc = 0;
    		while(x)
    			if(x%10==2)cc=1;
    			x/=10;
    		
    		if(cc)res++;
    	
    	cout<<res<<"\\n";
    	return 0;
    
    
    
    

第2题——扩散 (5分)

  • 题目

  • 思路:
    从四个点开始往外bfs,扩散2020轮即可。
    对于坐标负数的情况,默认加上2020,相当于离散化处理。
    注意边界处理

  • 答案是20312088

    #include<bits/stdc++.h>
    using namespace std;
    
    struct node int x, y, t = 0; ;
    int vis[2020+2020+2020][2020+2020+2020];
    node getnode(int x, int y) return nodex+2020,y+2020,0; 
    int dx[] = 0,0,-1,1;
    int dy[] = 1,-1,0,0;
    
    int main()
    	queue<node>q;
    	q.push(getnode(0,0));
    	q.push(getnode(2020,11));
    	q.push(getnode(11,14));
    	q.push(getnode(2000,2000));
    	int ans = 0;
    	while(q.size())
    		node t = q.front();  q.pop();
    		vis[t.x][t.y] = 1;
    		ans++;
    		// cout<<t.x<<' '<<t.y<<" "<<t.t<<"\\n";
    		// if(t.t==5)break;
    		for(int i = 0; i < 4; i++)
    			int nx = t.x+dx[i], ny = t.y+dy[i];
    			if(vis[nx][ny]==0 && nx>=0&&ny>=0&&t.t+1<=2020)//边界
    				q.push(nx,ny,t.t+1);
    				vis[nx][ny] = 1;
    			
    		
    	
    	cout<<ans<<"\\n";
    	return 0;
    
    
    
    

第3题——阶乘约数 (10分)

  • 题目:

  • 思路
    由质因数唯一分解定理可得,100!分解质因数后表示唯一,所以的因数个数+1乘起来就是它的因数个数。所以直接枚举1-100,累加质因数,最后统计即可。
    记得开ll不然会炸。

  • 答案:39001250856960000

    #include<bits/stdc++.h>
    using namespace std;
    int c[200];
    
    int main()
    	for(int i = 2; i <= 100; i++)
    		int x = i;
    		for(int j = 2; j*j<=i; j++)
    			while(x%j==0)
    				x /= j;
    				c[j]++;
    			
    		
    		if(x!=1)c[x]++;
    	
    	long long ans = 1;
    	for(int i = 2; i <= 100; i++)
    		ans *= (c[i]+1);
    	
    	cout<<ans<<"\\n";
    	return 0;
    
    
    
    

第4题——本质上升序列 (10分)

  • 题目

  • 思路:给出一个字符串,求有多少个不重复的单调递增子串

  • 类似于最长上升子序列,区别在于不能重复,而且求的是个数。
    记f[i]表示以s[i]结束的不重复单调递增子串个数,初始值为1即本身,转移时可以从前面所有比他小的字符转移过来,累加即可。
    对于重复的情况,可以在转移时枚举到前面有和当前字符相同字符时把转移过的值记为0。

  • 所以答案是3616159

    #include<bits/stdc++.h>
    using namespace std;
    
    int f[500];
    
    int main()
    	string s="tocyjkdzcieoiodfpbgcncsrjbhmugdnojjddhllnofawllbhfiadgdcdjstemphmnjihecoapdjjrprrqnhgccevdarufmliqijgihhfgdcmxvicfauachlifhafpdccfseflcdgjncadfclvfmadvrnaaahahndsikzssoywakgnfjjaihtniptwoulxbaeqkqhfwl";
    	for(int i = 0; i < s.size(); i++)f[i] = 1;
    	for(int i = 0; i < s.size(); i++)
    		for(int j = 0; j < i; j++)
    			if(s[i]>s[j])f[i] += f[j];
    			if(s[i]==s[j])f[i] = 0;
    		
    	
    	int ans = 0;
    	for(int i = 0; i < s.size(); i++)
    		ans += f[i];
    	
    	cout<<ans<<"\\n";
    	return 0;
    
    
    
    

第5题——玩具蛇 (15分)

  • 题目:

  • 思路:
    乍一看没啥头绪,但是考虑到只有16,方案数比较少,所以可以暴力dfs搜。
    暴力每个起点,每次往四个方向搜+1,然后对于不同的终点答案+1。

  • 所以答案是552

    #include<bits/stdc++.h>
    using namespace std;
    int n = 4, ans = 0;
    int vis[10][10];
    
    int dx[] = 0,0,-1,1;
    int dy[] = 1,-1,0,0;
    void dfs(int x, int y, int now)
    	if(now==n*n)
    		ans++; return ;
    	
    	for(int i = 0; i < 4; i++)
    		int nx = x+dx[i], ny = y+dy[i];
    		if(nx>=1&&nx<=n&&ny>=1&&ny<=n &&vis[nx][ny]==0)
    			vis[nx][ny] = 1;
    			dfs(nx,ny,now+1);
    			vis[nx][ny] = 0;
    		
    	
    
    
    int main()
    	for(int i = 1; i <= n; i++)
    		for(int j = 1; j <= n; j++)
    			memset(vis,0,sizeof(vis));
    			vis[i][j] = 1;
    			dfs(i,j,1);
    		
    	
    	cout<<ans<<"\\n";
    	return 0;
    
    
    
    

第6题——皮亚诺曲线距离 (15分)

  • 题目


  • 思路:
    说实话这道题目放在第六题感觉有点不讲武德,感觉比后面的难好吧()
    求两点距离不难想到可以转化为两点到起点的距离作差。

  • 然后,考虑简化问题,k阶的图是由9个k-1阶的图组成的
    我们可以考虑它在这九个中的哪一块,然后分类讨论递归下去,直到来到1阶和2阶,输出答案。

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    
    LL calc(LL x, LL n)
    	LL ans = 1;
    	while(n)
    		if(n%2==1) ans*=x; n--;
    		x *= x;  n/= 2;
    	
    	return ans;
    
    LL dfs(LL now, LL xx, LL yy)
    	if(now==0)return 1;
    	if(now>=40)now=39;
    	//(n-1)阶时的边长,也就是n阶时,分成9个(n-1)阶时每一块的边长
    	LL k = calc(3,now-1); 
    	//判断该点在所在块内的横纵坐标(x,y), 以及所在块的横纵坐标(a,b)
    	LL x = xx%k, y = yy%k, a = xx/k, b = yy/k;
    	if(a==0&&b==0)     return 0*k*k+dfs(now-1,x,y);
    	else if(a==0&&b==1)return 1*k*k+dfs(now-1,k-1-x,y);
    	else if(a==0&&b==2)return 2*k*k+dfs(now-1,x,y);
    	else if(a==1&&b==2)return 3*k*k+dfs(now-1,x,k-1-y);
    	else if(a==1&&b==1)return 4*k*k+dfs(now-1,k-1-x,k-1-y);
    	else if(a==1&&b==0)return 5*k*k+dfs(now-1,x,k-1-y);
    	else if(a==2&&b==0)return 6*k*k+dfs(now-1,x,y);
    	else if(a==2&&b==1)return 7*k*k+dfs(now-1,k-1-x,y);
    	else if(a==2&&b==2)return 8*k*k+dfs(now-1,x,y); 
    
    
    int main()
    	LL n;  cin>>n;
    	LL x1, y1, x2, y2;  cin>>x1>>y1>>x2>>y2;
    	LL ans = dfs(n,x1,y1)-dfs(n,x2,y2);
    	if(ans<0)ans=-ans;
    	cout<<ans<<"\\n";
    	return 0;
    
    
    
    

第7题——游园安排 (20分)

  • 题目

  • 思路:
    看完就是个LIS最长上升子序列,只不过把数字换成了字符串比较。
    对于路径打印,开个数组回溯一下即可。
    暴力dp能过70%

    //70分做法
    #include<bits/stdc++.h>
    using namespace std;
    const int maxn = 1e6+10;
    
    int cnt = 0;
    string name[maxn];
    int f[maxn]; //到i为止的最大值
    
    int main()
    	string s;  cin>>s;
    	for(int i = 0; i < s.size(); i++)
    		if(isupper(s[i]))
    			name[++cnt]="";
    			name[cnt]+=s[i];
    		else
    			name[cnt]+=s[i];
    		
    	
    	int mxl = 0, pos = 1;  //记录最长序列的长度和坐标
    	for(int i = 1; i <= cnt; i++)
    		f[i] = 1;
    		for(int j = 1; j < i; j++)
    			if(name[i]>name[j])//可以从比i小的转移过来
    				f[i] = max(f[i], f[j]+1);
    			
    		
    		if(f[i]>=mxl)
    			mxl = f[i], pos = i;
    		
    	
    	string res = "";
    	for(int i = pos; i >= 1; i--)//向前回溯
    		if(f[i]==mxl)
    			res = name[i]+res;
    			mxl--;
    		
    	
    	cout<<res<<"\\n";
    	return 0;
    
    
    
    
  • 考虑贪心优化
    将所有人序列取出存储到数组中后,先将第一个序列入队,遍历剩下所有序列,如果说碰见比队尾大的序列就直接入队,否则,也就是当前序列并不小于队尾序列的话,那么就用其代替掉队列中比其大的第一个序列,至于如何找到这个第一个比他大的,因为队列是有序的,那么就可以二分,复杂度从平方优化到了log。

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn = 1e6+10;
    
    #define ios ios::sync_with_stdio(0), cin.tie(0),cout.tie(0)
    int cnt = 0;
    string name[maxn];
    vector<int>f; //到i为止的最大值
    
    int main()
    	IOS;
    	string s;  cin>>s;
    	for(int i = 0; i < s.size(); i++)
    		if(isupper(s[i]))
    			if(i!=0)cnt++;
    			name[cnt]="";
    			name[cnt]+=s[i];
    		else
    			name[cnt]+=s[i];
    		
    	
    	vector<string>vc; //当前LIS序列
    	vc.push_back(以上是关于2020 第十一届蓝桥杯大赛软件赛决赛,国赛,C/C++大学B组题解的主要内容,如果未能解决你的问题,请参考以下文章

    2021 第十二届蓝桥杯大赛软件赛决赛, 国赛,C/C++ 大学B 组

    第十一届蓝桥杯大赛软件类决赛(C/C++ 大学A组)

    2022年6月第十三届蓝桥杯大赛软件赛全国决赛C++A组题解

    2022年6月第十三届蓝桥杯大赛软件赛全国决赛C++A组题解

    2022年6月第十三届蓝桥杯大赛软件赛全国决赛C++A组题解

    第十三届蓝桥杯c++b组2022年国赛决赛题解