第七届蓝桥杯省赛A组c++

1.网友年龄

网友年龄

某君新认识一网友。
当问及年龄时,他的网友说:
“我的年龄是个2位数,我比儿子大27岁,
如果把我的年龄的两位数字交换位置,刚好就是我儿子的年龄”

请你计算:网友的年龄一共有多少种可能情况?

提示:30岁就是其中一种可能哦.

请填写表示可能情况的种数。
注意:你提交的应该是一个整数,不要填写任何多余的内容或说明性文字。

#include<iostream>
using namespace std;
int main()
{
	int res=0;
	for(int i=10;i<100;i++)
	{
		int x=i/10,y=i%10;
		if(i-(y*10+x)==27)res++;
	}
	cout<<res;
	return 0;
}

答案:7

2.生日蜡烛

生日蜡烛

某君从某年开始每年都举办一次生日party,并且每次都要吹熄与年龄相同根数的蜡烛。

现在算起来,他一共吹熄了236根蜡烛。

请问,他从多少岁开始过生日party的?

请填写他开始过生日party的年龄数。
注意:你提交的应该是一个整数,不要填写任何多余的内容或说明性文字。
236=(n*(a0+a0+n-1))/2
236×2=n×m(且n,m一奇一偶)
236×2= 59×8
答案:26

3.方格填数

如下的10个格子
±-±-±-+
| | | |
±-±-±-±-+
| | | | |
±-±-±-±-+
| | | |
±-±-±-+

(如果显示有问题,也可以参看【图1.jpg】)

填入0~9的数字。要求:连续的两个数字不能相邻。
(左右、上下、对角都算相邻)

一共有多少种可能的填数方案?

请填写表示方案数目的整数。
注意:你提交的应该是一个整数,不要填写任何多余的内容或说明性文字。
方法1:枚举

#include<iostream>
#include<algorithm> 
using namespace std;
int a[]={100,1,2,3,4,5,6,7,8,9,10,100},dx[]={-1,-1,-1,0,1,1,1,0},dy[]={-1,0,1,1,1,0,-1,-1};
bool judge()
{
	for(int i=1;i<=10;i++)
	{
		int r=i/4,c=i%4;//是i不是a[i]!!!
		for(int j=0;j<8;j++)
		{
			int x=r+dx[j],y=c+dy[j];
			if(x>=0&&x<3&&y>=0&&y<4&&abs(a[x*4+y]-a[i])==1) return false;
		}
	}
	return true;
}
int main()
{
	int res=0;
	do{
		if(judge()) res++;
	}while(next_permutation(a+1,a+11));
	cout<<res<<endl;
	return 0;
	
}

方法二:dfs(类似八皇后)

#include<iostream>
#include<algorithm> 
using namespace std;
int vis[12],a[3][4],dx[]={-1,-1,-1,0},dy[]={-1,0,1,-1},res=0;
bool judge(int r,int c,int num)
{
	for(int i=0;i<4;i++)
	{
		int x=r+dx[i],y=c+dy[i];
		if(x>=0&&x<3&&y>=0&&y<4&&abs(num-a[x][y])==1) return false;
	}
	return true;
}
void dfs(int pos)
{
	if(pos==11) res++;
	int r=pos/4,c=pos%4; 
	for(int num=1;num<=10;num++)
	{
		if(!vis[num]&&judge(r,c,num))
		{
			vis[num]=1;
			a[r][c]=num;
			dfs(pos+1);
			vis[num]=0;
			a[r][c]=0;
		}
	}
}
int main()
{
	a[0][0]=100;
	a[2][3]=100;
	dfs(1);
	cout<<res<<endl;
	return 0;
	
}

答案:1580

4.快速排序

排序在各种场合经常被用到。
快速排序是十分常用的高效率的算法。

其思想是:先选一个“标尺”,
用它把整个队列过一遍筛子,
以保证:其左边的元素都不大于它,其右边的元素都不小于它。

这样,排序问题就被分割为两个子区间。
再分别对子区间排序就可以了。

下面的代码是一种实现,请分析并填写划线部分缺少的代码。


#include <stdio.h>

void swap(int a[], int i, int j)
{
	int t = a[i];
	a[i] = a[j];
	a[j] = t;
}

int partition(int a[], int p, int r)
{
    int i = p;
    int j = r + 1;
    int x = a[p];
    while(1){
        while(i<r && a[++i]<x);
        while(a[--j]>x);
        if(i>=j) break;
        swap(a,i,j);
    }
	______________________;
    return j;
}

void quicksort(int a[], int p, int r)
{
    if(p<r){
        int q = partition(a,p,r);
        quicksort(a,p,q-1);
        quicksort(a,q+1,r);
    }
}
    
int main()
{
	int i;
	int a[] = {5,13,6,24,2,8,19,27,6,12,1,17};
	int N = 12;
	
	quicksort(a, 0, N-1);
	
	for(i=0; i<N; i++) printf("%d ", a[i]);
	printf("\n");
	
	return 0;
}

答案:swap(a,p,j)

5.消除尾一

下面的代码把一个整数的二进制表示的最右边的连续的1全部变成0
如果最后一位是0,则原数字保持不变。

如果采用代码中的测试数据,应该输出:
00000000000000000000000001100111 00000000000000000000000001100000
00000000000000000000000000001100 00000000000000000000000000001100

请仔细阅读程序,填写划线部分缺少的代码。

#include <stdio.h>

void f(int x)
{
	int i;
	for(i=0; i<32; i++) printf("%d", (x>>(31-i))&1);
	printf("   ");
	
	x = _______________________;
	
	for(i=0; i<32; i++) printf("%d", (x>>(31-i))&1);
	printf("\n");	
}

int main()
{
	f(103);
	f(12);
	return 0;
}

答案:x&(x+1)

6.寒假作业

现在小学的数学题目也不是那么好玩的。
看看这个寒假作业:

□ + □ = □
□ - □ = □
□ × □ = □
□ ÷ □ = □

(如果显示不出来,可以参见【图1.jpg】)

每个方块代表1~13中的某一个数字,但不能重复。
比如:
6 + 7 = 13
9 - 8 = 1
3 * 4 = 12
10 / 2 = 5

以及:
7 + 6 = 13
9 - 8 = 1
3 * 4 = 12
10 / 2 = 5

就算两种解法。(加法,乘法交换律后算不同的方案)
你一共找到了多少种方案?

思路:1~13中找到两组相加,两组相乘
从相乘入手:只可能是3 * 4=12,2 * 5=10;
再考虑相加,发现只能是7+6=13,8+1=9;
排列组合:2(选择加法减法)* 2(选择乘法除法)* 2^4(交换)
答案:64

7.剪邮票

如【图1.jpg】, 有12张连在一起的12生肖的邮票。
现在你要从中剪下5张来,要求必须是连着的。
(仅仅连接一个角不算相连)
比如,【图2.jpg】,【图3.jpg】中,粉红色所示部分就是合格的剪取。

请你计算,一共有多少种不同的剪取方法。

请填写表示方案数目的整数。
注意:你提交的应该是一个整数,不要填写任何多余的内容或说明性文字。
pic1在这里插入图片描述

#include<iostream>
#include<cstring>
using namespace std;
int n=12,m=5,res=0,a[3][4],vis[3][4],dx[]={-1,0,1,0},dy[]={0,1,0,-1},judge_res;
void dfs_map(int r,int c)
{
	for(int i=0;i<4;i++)
	{
		int x=r+dx[i],y=c+dy[i];
		if(x>=0&&x<3&&y>=0&&y<4&&!vis[x][y]&&a[x][y])
		{
			judge_res++;
			vis[x][y]=1;
			dfs_map(x,y);
		} 
	}	
}
void dfs(int k,int s,int state){
	if(n-k<m-s) return;
	if(s==m)
	{
		int r,c;
		memset(a,0,sizeof a);
	    memset(vis,0,sizeof vis);
	    for(int i=0;i<n;i++)
		    if(state>>i&1)
		    {
		    	r=i/4,c=i%4;
		    	a[r][c]=1;
			}
		vis[r][c]=1;
		judge_res=1;
		dfs_map(r,c);
	    if(judge_res==5) res++;
	    return;
	}
	else
	{
		dfs(k+1,s+1,state|1<<k);
		dfs(k+1,s,state);
	}
}
int main()
{
	dfs(0,0,0);
	cout<<res<<endl;
	return 0;
}

答案:116
出现错误:vis[3][4]写成vis[4][3]woc!!!

8.四平方和

四平方和定理,又称为拉格朗日定理:
每个正整数都可以表示为至多4个正整数的平方和。
如果把0包括进去,就正好可以表示为4个数的平方和。

比如:
5 = 0^2 + 0^2 + 1^2 + 2^2
7 = 1^2 + 1^2 + 1^2 + 2^2
(^符号表示乘方的意思)

对于一个给定的正整数,可能存在多种平方和的表示法。
要求你对4个数排序:
0 <= a <= b <= c <= d
并对所有的可能表示法按 a,b,c,d 为联合主键升序排列,最后输出第一个表示法

程序输入为一个正整数N (N<5000000)
要求输出4个非负整数,按从小到大排序,中间用空格分开

例如,输入:
5
则程序应该输出:
0 0 1 2

再例如,输入:
12
则程序应该输出:
0 2 2 2

再例如,输入:
773535
则程序应该输出:
1 1 267 838

资源约定:
峰值内存消耗 < 256M
CPU消耗 < 3000ms

思路:emmmmm,目前我在网上看到的一类是三重循环枚举,还有一类是打表,不过不太理解。。。

9.密码脱落

X星球的考古学家发现了一批古代留下来的密码。
这些密码是由A、B、C、D 四种植物的种子串成的序列。
仔细分析发现,这些密码串当初应该是前后对称的(也就是我们说的镜像串)。
由于年代久远,其中许多种子脱落了,因而可能会失去镜像的特征。

你的任务是:
给定一个现在看到的密码串,计算一下从当初的状态,它要至少脱落多少个种子,才可能会变成现在的样子。

输入一行,表示现在看到的密码串(长度不大于1000)
要求输出一个正整数,表示至少脱落了多少个种子。

例如,输入:
ABCBA
则程序应该输出:
0

再例如,输入:
ABDCDCBABC
则程序应该输出:
3

资源约定:
峰值内存消耗 < 256M
CPU消耗 < 1000ms

思路:将输入的串反转,求两个串的最长公共子串!!!绝了!可惜我没有想出来:(

#include<iostream>
#include<algorithm>
#include<string>
using namespace std;
int dp[1010][1010],len;
void LCS(string &s1,string &s2)
{
	for(int i=1;i<=len;i++)
	{
		for(int j=1;j<=len;j++)
		{
			if(s1[i]==s2[j]) dp[i][j]=dp[i-1][j-1]+1;
			else dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
		}
	}
}
int main()
{
	string s1,s2;
	cin>>s1;
	len=s1.size();
	s1=" "+s1;
	s2=s1;
	reverse(s2.begin()+1,s2.end());
	LCS(s1,s2);
	cout<<len-dp[len][len]<<endl;
	return 0;
}
10.最大比例

X星球的某个大奖赛设了M级奖励。每个级别的奖金是一个正整数。
并且,相邻的两个级别间的比例是个固定值。
也就是说:所有级别的奖金数构成了一个等比数列。比如:
16,24,36,54
其等比值为:3/2

现在,我们随机调查了一些获奖者的奖金数。
请你据此推算可能的最大的等比值。

输入格式:
第一行为数字 N (0<N<100),表示接下的一行包含N个正整数
第二行N个正整数Xi(Xi<1 000 000 000 000),用空格分开。每个整数表示调查到的某人的奖金数额

要求输出:
一个形如A/B的分数,要求A、B互质。表示可能的最大比例系数

测试数据保证了输入格式正确,并且最大比例是存在的。

例如,输入:
3
1250 200 32

程序应该输出:
25/4

再例如,输入:
4
3125 32 32 200

程序应该输出:
5/2

再例如,输入:
3
549755813888 524288 2

程序应该输出:
4/1

资源约定:
峰值内存消耗 < 256M
CPU消耗 < 3000ms

思路:求最大公比,反复相除,直到只剩一个数(由于写的时候debug很久,心情烦躁,所以代码很乱啊啊啊)

#include<iostream>
#include<algorithm>
using namespace std;
typedef long long ll;
typedef pair<ll,ll> p;
const int N=105;
ll num[N];
p a[N];
ll gcd(ll n,ll m)
{
	return n%m==0?m:gcd(m,n%m);
}
bool cmp(p x,p y)
{
	return x.first*1.0/x.second<y.first*1.0/y.second;
	//其实也可这样写 return x.first<y.first; 
}

int main()
{
	int n;
	cin>>n; 
	ll t;
	for(int i=0;i<n;i++) cin>>num[i];
	sort(num,num+n);
    n=unique(num,num+n)-num;
	for(int i=0;i<n-1;i++)
	{
		t=gcd(num[i+1],num[i]);
		a[i].first=num[i+1]/t;
		a[i].second=num[i]/t;
	}	
	sort(a,a+n-1,cmp);  //关于a元素个数很容易弄错 
    n=unique(a,a+n-1)-a; 
	while(n>1)
	{
		for(int i=0;i<n-1;i++)
		{
			a[i].first=a[i+1].first/a[i].first;
			a[i].second=a[i+1].second/a[i].second;		
		}
		n--;//很容易忘 
		sort(a,a+n,cmp);
	    n=unique(a,a+n)-a; 
	}
	cout<<a[0].first<<"/"<<a[0].second<<endl;
	return 0;
}

遇到错误的地方大致都标出来了:(,一定要谨慎!!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值