第4讲 文字或算式题目的解题技巧(2)

1.下列乘法算式中:赛软件 * 比赛 = 软件比拼

其中每个汉字代表1个数字(0~9)。相同的汉字代表相同的数字,不同的汉字代表不同的数字。

试编程确定使得整个算式成立的数字组合,如有多种情况,请给出所有可能的答案。

#include<stdio.h>

#include<math.h>

int main()

{

int a,b,c,d,e,f,g; 

int x,y,z,k=1;

for(a=0;a<=9;a++){

for(b=0;b<=9;b++){

for(c=0;c<=9;c++){

for(d=0;d<=9;d++){

for(e=0;e<=9;e++){

for(f=0;f<=9;f++){

for(g=0;g<=9;g++){

                            if(a!=b && a!=c&& a!=d&&a!=e && a!=f&& a!=g && 

b!=c && b!=d && b!=e && b!=f && b!=g &&

c!=d && c!=e && c!=f && c!=g

 && d!=e && d!=f && d!=g &&

  e!=f && e!=g &&

   f!=g

   && k!=a &&k!=b &&k!=c &&k!=d &&k!=e &&k!=f &&k!=g ) {

    x=a*1000+b*100+c*10+d;

    y=k*1000+e*100+f*10+b;

    z=k*10000+e*1000+c*100+b*10+g;  }

if(x+y==z){

printf("%d+%d=%d",x,y,z);

return 0; }}}}}}}}

return 0;}

2.题目2:2012年——古堡算式

福尔摩斯到某古堡探险,看到门上写着一个奇怪的算式:

    ABCDE * ? = EDCBA

    他对华生说:“ABCDE应该代表不同的数字,问号也代表某个数字!”

    华生:“我猜也是!”

于是,两人沉默了好久,还是没有算出合适的结果来。

请你利用计算机的优势,找到破解的答案。

把 ABCDE 所代表的数字写出来。

#include <stdio.h>

int main(){

int a,b,c,d,e;

int x,y,z;

for(a=1;a<10;a++){

for(b=1;b<10;b++){

for(c=0;c<10;c++){

for(d=1;d<10;d++){

for(e=0;e<10;e++){

if(a!=b  &&  a!=c  &&  a!=d  &&  a!=e  &&  b!=c  && b!=d  &&  b!=e  &&  c!=d  &&  c!=e  &&  d!=e){

   x=a*100+b*10+c;

   y=d*10+a;

   z=b*1000+c*100*d*10+e;

   if(x*y==z){

       printf("%d*%d=%d\n",x,y,z); } } } } } } }

return 0; }

3.小明是个急性子,上小学的时候经常把老师写在黑板上的题目抄错了。

有一次,老师出的题目是:36 x 495 = ?

他却给抄成了:396 x 45 = ?

但结果却很戏剧性,他的答案竟然是对的!!

因为 36 * 495 = 396 * 45 = 17820

类似这样的巧合情况可能还有很多,比如:27 * 594 = 297 * 54

假设 a b c d e 代表1~9不同的5个数字(注意是各不相同的数字,且不含0)

答案是:142

#include<stdio.h>

#include<math.h>

int main()

{

int a,b,c,d,e;

int x1,y1,x2,y2;

int i=0;

for(a=1;a<=9;a++)  {

for(b=1;b<=9;b++)  {

for(c=1;c<=9;c++)  {

for(d=1;d<=9;d++)  {

for(e=1;e<=9;e++)  {

if(a!=b&&a!=c&&a!=d&&a!=e&&

b!=c&&b!=e&&b!=d&&

c!=d&&c!=e&&

d!=e)  {

                    x1=a*10+b;

                    y1=c*100+d*10+e;

                    x2=a*100+d*10+b;

                    y2=c*10+e;

                    if(x1*y1==x2*y2)  {

                     i++;  } } } } } } }

printf("%d",i);

return 0;  }

4.题目4:2015年——三羊献瑞

观察下面的加法算式:

 

其中,相同的汉字代表相同的数字,不同的汉字代表不同的数字。

请你填写“三羊献瑞”所代表的4位数字(答案唯一),不要填写任何多余内容。

#include<stdio.h>

#include<math.h>

int main()  {

int a,b,c,d,e,f,g; 

int x,y,z,k=1;

for(a=0;a<=9;a++)  {

for(b=0;b<=9;b++)  {

for(c=0;c<=9;c++)  {

for(d=0;d<=9;d++)  {

for(e=0;e<=9;e++)  {

for(f=0;f<=9;f++)  {

for(g=0;g<=9;g++)  {

                            if(a!=b && a!=c&& a!=d&&a!=e && a!=f&& a!=g && 

b!=c && b!=d && b!=e && b!=f && b!=g &&

c!=d && c!=e && c!=f && c!=g

 && d!=e && d!=f && d!=g &&

  e!=f && e!=g &&

   f!=g

   && k!=a &&k!=b &&k!=c &&k!=d &&k!=e &&k!=f &&k!=g )  {

    x=a*1000+b*100+c*10+d;

    y=k*1000+e*100+f*10+b;

    z=k*10000+e*1000+c*100+b*10+g;   }

if(x+y==z)  {

printf("%d+%d=%d",x,y,z);

return 0;  } } } } } } } }

return 0;}

5.题目1:2015年——加法变乘法

我们都知道:1+2+3+ ... + 49 = 1225

现在要求你把其中两个不相邻的加号变成乘号,使得结果为2015。

比如:1+2+3+...+10*11+12+...+27*28+29+...+49 = 2015 就是符合要求的答案。

请你寻找另外一个可能的答案,并把位置靠前的那个乘号左边的数字提交(对于示例,就是提交10)。

#include <stdio.h>

int main()

{

int i,j;

for(i=1;i<=48;i++)  {

for(j=i+2;j<=48;j++)  {

if(i*(i+1)+j*(j+1)==2015-1225-i-(i+1)-j-(j+1))

   printf("%d %d %d %d\n",i,i+1,j,j+1);  } }

return 0; } 

6.题目2:625

625这个数字很特别,625的平方等于390625,刚好其末3位是625本身。除了625,还有其它的3位数有这个特征吗?

请编写程序,寻找所有这样的3位数:它的平方的末3位是这个数字本身。

输出结果中,从小到大,每个找到的数字占一行。比如那个625就输出为:625

#include <stdio.h>

#include <math.h>

int main(){

long long n,x,y;

for(n=100;n<1000;n++) {

x=n*n;

y=x%1000;

if(n==y)   {

printf("%lld\n",n);  } }

return 0; }

7.题目 3:立方和问题。

考虑方程式:a^3 + b^3 = c^3 + d^3

其中:“^”表示乘方。a、b、c、d是互不相同的小于30的正整数。这个方程有很多解。比如:

a = 1,b=12,c=9,d=10 就是一个解。因为:1的立方加12的立方等于1729,而9的立方加10的立方也等于1729。

当然,a=12,b=1,c=9,d=10 显然也是解。

如果不计abcd交换次序的情况,这算同一个解。

请编写程序实现如下任务:找到所有小于30的不同的正整数解。把a b c d按从小到大排列,用逗号分隔,每个解占用1行。比如,刚才的解输出为:1,9,10,12    不同解间的顺序可以不考虑。

#include <stdio.h>

int main()

{

int a,b,c,d;

for(a=1;a<=30;a++)  {

for(b=1;b<30;b++)  {

for(c=a+1;c<=30;c++)  {

for(d=c+1;d<=30;d++)  {

if(a!=b && a!=c && a!=d && b!=c && b!=d && c!=d)  {

if(a*a*a+b*b*b==c*c*c+d*d*d)

   printf("%d,%d,%d,%d\n",a,c,d,b);  } } } } } } 

第8讲 递归函数的构建及其应用

 

题目1:递归函数构建——输出begin到end

编写一个递归函数,分行输出begin到end。然后在主函数中调用该函数进行测试。

#include<stdio.h>

int giao(int a,int b)

{

    if(a==b)   {

     printf("%d\n",b);

     return 0;   }

printf("%d\n",a);

return giao(a+1,b);  }

int main()

{

int a,b;

scanf("%d %d",&a,&b);

giao(a,b);  }

题目2:编写递归函数int mysum(int n),功能是实现求1到n之和并返回。然后在主函数中输入整数n,调用mysum函数求1到n之和并输出。

#include<stdio.h>

int giao (int n)

{

if(n==0) return 0;

if(n==1) return 1;

return giao(n-1)+n;  }

int main()  {

int n,sum=0;

scanf("%d",&n);

sum=giao(n);

printf("%d\n",sum);  } 

3.题目:编写递归函数求两个数的最大公约数

编写递归函数int gcd(int m,int n)计算m和n的最大公约数并返回。然后在主函数中输入2个整数并调用gcd函数求最大公约数并输出。

#include<stdio.h>

int giao (int m,int n)

{

if(m%n==0) return n;

return giao(n,m%n);  }

int main()  {

int m,n,x;

scanf("%d %d",&m,&n);

x=giao(m,n);

printf("%d\n",x);   }

4.题目.使用递归,求数组的前n项之和。

int mysum1( int a[ ],int n )   

//该递归函数是求数组a的前n个元素之和 

{

}

#include<stdio.h>

int giao (int a[],int n)

{

if(n==0) return 0;

if(n==1) return 1;

return giao(a,n-1)+a[n-1];  }

int main()

{

int n,sum=0,a[100];

scanf("%d",&n);

for(int i=0;i<n;i++)

scanf("%d",&a[i]);

sum=giao(a,n);

printf("%d\n",sum);

5.题目.使用递归,求数组的前n项之和。

int mysum2( int a[ ],int L,int R ) 

 //该递归函数是求数组a的第L到第R个元素之和 

{

}

#include<stdio.h>

int giao (int a[],int L,int R)

{

if(L>R) return 0;

return a[L]+giao(a,L+1,R);  }

int main()

{

int n,sum=0,a[100];

scanf("%d",&n);

for(int i=0;i<n;i++)

scanf("%d",&a[i]);

sum=giao(a,0,n-1);

printf("%d\n",sum);  } 

(以下题目要求用递归函数实现)

题目 1575: 蓝桥杯算法提高VIP-递归倒置字符数组

完成一个递归程序,倒置字符数组。并打印实现过程

递归逻辑为:

当字符长度等于1时,直接返回

否则,调换首尾两个字符,在递归地倒置字符数组的剩下部分

输入   字符数组长度及该数组 

输出

在求解过程中,打印字符数组的变化情况。 

最后空一行,在程序结尾处打印倒置后该数组的各个元素。 

样例输入  5 abcde

样例输出

ebcda

edcba

 

Edcba

#include <stdio.h>

#include <string.h>

void mychar(char a[],int l,int r)

{

    if(l>=r) return ;

    char t;

    t=a[l];

    a[l]=a[r];

    a[r]=t;

    puts(a);

    mychar(a,l+1,r-1);  }

int main()  {

    char a[1000];

    int n;

    scanf("%d\n",&n);

    gets(a);

    mychar(a,0,n-1);

    printf("\n");

    puts(a);

    return 0;  }

题目 1400: 教学楼的楼梯

沈理工的教学楼有多少级楼梯,你造吗?

那么问题来了,假设共M级,刚开始时你在第一级,若每次只能跨上一级或二级,要走上第M级,共有多少种走法?

输入  输入数据首先包含一个整数N,表示测试实例的个数,然后是N行数据,每行包含一个整数M(1<=M<=40),表示楼梯的级数。

输出   对于每个测试实例,请输出不同走法的数量

样例输入

2

2

3

样例输出

1

2

#include<stdio.h>

int main()

{

 int n,t,i;

 int f[40];

 scanf("%d",&n);

 f[0]=f[1]=1;

 for(i=2;i<40;i++)   {

  f[i]=f[i-1]+f[i-2];   }

 while(n--)  {

  scanf("%d",&t);

  if(t>=1&&t<=40)

   printf("%d\n",f[t-1]);   }

 return 0;  }

题目 1861: 程序员爬楼梯

程序员是善于思考的,有一天他在爬楼梯的时候想出一个问题。

  楼梯有 n 级。每次你只能爬 1 级或者 3 级,那么你有多少种方法爬到楼梯的顶部?

  开始的时候在0级楼梯,顶级在第n级。

输入  一行,一个n, 2<=n<=20。 

输出  一行,输出一个整数,表示爬到n级的方案数。

样例输入  3

样例输出  2

#include<stdio.h>

void my(int f[],int n)

{

 f[1]=f[2]=1;

 f[3]=2;

 for(int i=4;i<=20;i++)

   f[i]=f[i-1]+f[i-3];  }

int main()

{

 int n,f[20];

 scanf("%d",&n);

 my(f,n);

 printf("%d",f[n]);

 return 0;  }

题目 1257: 超级楼梯

有一楼梯共M级,刚开始时你在第一级,若每次只能跨上一级或二级,要走上第M级,共有多少种走法?

输入

输入数据首先包含一个整数N,表示测试实例的个数,然后是N行数据,每行包含一个整数M(1<=M<=40),表示楼梯的级数。

输出

对于每个测试实例,请输出不同走法的数量

样例输入

2

2

3

样例输出

1

2

#include<stdio.h>

void myfun(int f[],int n)

{

 f[0]=f[1]=1;

 for(int i=2;i<=40;i++) {

  f[i]=f[i-1]+f[i-2];   }}

int main()  {

 int n,t;

 int f[40];

 scanf("%d",&n);

 myfun(f,n);

 while(n--)  {

  scanf("%d",&t);

  if(t>=2&&t<=40)

   printf("%d\n",f[t-1]);   }

 return 0;  }

题目 1031: [编程入门]自定义函数之字符串反转

写一函数,使输入的一个字符串按反序存放,在主函数中输入并输出反序后的字符串(不包含空格)。

输入  一行字符

输出  逆序后的字符串

样例输入  123456abcdef 

样例输出  fedcba654321

#include<stdio.h>

#include<string.h>

void mychar(char a[],int L,int R)

{

 char t;

 if(L>=R) return ;

  t=a[L];

  a[L]=a[R];

  a[R]=t;

 mychar(a,L+1,R-1);

}

int main()  {

 char a[100];

 int len;

 gets(a);

 len=strlen(a);

    mychar(a,0,len-1);

 puts(a);

 return 0;  }

题目 1093: 字符逆序(C++)

将一个字符串str的内容颠倒过来,并输出。str的长度不超过100个字符。

输入   输入包括一行。 第一行输入的字符串。

输出   输出转换好的逆序字符串。

样例输入   I am a student

样例输出   tneduts a ma I

#include<bits/stdc++.h>

#include<string.h>

using namespace std;

int main()  {

     string a;

     getline(cin,a);

     reverse(a.begin(),a.end());

     cout<<a;

 return 0;  }

题目 1026: [编程入门]数字逆序输出

输入10个数字,然后逆序输出。

输入  十个整数

输出  逆序输出,空格分开

样例输入  1 2 3 4 5 6 7 8 9 0

样例输出  0 9 8 7 6 5 4 3 2 1

#include<stdio.h>

int main()  {

 int a[10],i,j;

 for(i=0;i<10;i++)

   scanf("%d",&a[i]);

 for(i=9;i>=0;i--)

   printf("%d ",a[i]);

 return 0;  }

题目 1931: 蓝桥杯算法提高VIP-逆序排列

编写一个程序,读入一组整数(不超过20个),并把它们保存在一个整型数组中。当用户输入0时,表示输入结束。然后程序将把这个数组中的值按逆序重新存放,并打印出来。例如:假设用户输入了一组数据:7 19 -5 6 2 0,那么程序将会把前五个有效数据保存在一个数组中,即7 19 -5 6 2,然后把这个数组中的值按逆序重新存放,即变成了2 6 -5 19 7,然后把它们打印出来。

输入  输入只有一行,由若干个整数组成,中间用空格隔开,最末尾的整数为0。

输出  输出也只有一行,即逆序排列后的整数,中间用空格隔开,末尾没有空格。

样例输入  7 19 -5 6 2 0

样例输出  2 6 -5 19 7

#include<stdio.h>

void myfun(int a[],int n)  {

 int i,t;

 for(i=0;i<n/2;i++)  {

  t=a[i];

  a[i]=a[n-i-1];

  a[n-i-1]=t;   } }

int main()  {

 int i,j,t,n;

 int a[20];

 for(i=0;i<20;i++)   {

  scanf("%d",&a[i]);

  if(a[i]==0)

    break;   }

 myfun(a,i);

 for(j=0;j<i;j++)

   printf("%d ",a[j]);

 return 0;  }

题目 1999: 回文判断

若一个正整数从左向右读与从右向左读都一样,我们就将其称之为回文数(例如12321、44、3都是回文数)。输入一个正整数,判断它是否是回文数,是则输出YES,否则输出NO。(提示:以字符串形式读取输入的整数)

输入  正整数

输出  YES或NO

样例输入  2332

样例输出  YES

#include<stdio.h>

#include<string.h>

int myhui(char a[],int l,int r)  {

 if(l>r) return 1;

 if(a[l]!=a[r]) return 0;

 return myhui(a,l+1,r-1);  }

int main()  {

 char a[1000];

 int b,len;

 gets(a);

 len=strlen(a);

 b=myhui(a,0,len-1);

 if(b) printf("YES\n");

 else  printf("NO\n");

 return 0;  }

题目 1200: 回文串

回文串是从左到右或者从右到左读起来都一样的字符串,试编程判别一个字符串是否为回文串。

输入  输入一个字符串。串长度<255.

输出  判别输入的字符串是否为回文串,是输出"Y",否则输出"N"。

样例输入  abcba

样例输出  Y

#include<stdio.h>

#include<string.h>

int myhui(char a[],int l,int r)  {

 if(l>r) return 1;

 if(a[l]!=a[r]) return 0;

 return myhui(a,l+1,r-1);  }

int main()  {

 char a[1000];

 int b,len;

 gets(a);

 len=strlen(a);

 b=myhui(a,0,len-1);

 if(b) printf("Y\n");

 else  printf("N\n");

 return 0;  }

第10讲 C++标准模板库(STL)介绍和使用

问题 1219: 数列排序(C++)

将一正整数序列{K1,K2,...,K9}重新排列成一个新的序列。新序列中,比K1小的数都在K1的前面(左面),比K1大的数都在K1的后面(右面)。

输入   输入有多行,第一行为N表示行数,每行9个整数.

输出   输出N行,按要求进行排序的结果.

样例输入

2

6 8 9 1 2 5 4 7 3

3 5 8 9 1 2 6 4 7

样例输出

3 4 5 2 1 6 8 9 7

2 1 3 5 8 9 6 4 7

#include<bits/stdc++.h>

using namespace std;

int main()

{

int n,k,ki,len;

vector <int> v;

cin>>n;

while(n--) {

cin>>k;

v.push_back(k);

for(int i=1;i<9;i++) {

cin>>ki;

if(ki>k) 

  v.push_back(ki);

else

  v.insert(v.begin(),ki); } 

 len=v.size();

for(int i=0;i<len;i++)

  cout<<v[i]<<' ';

cout<<endl;

    v.clear();  }

return 0;  }

问题 1487: [蓝桥杯][算法提高VIP]不同单词个数统计(C++)

编写一个程序,输入一个句子,然后统计出这个句子当中不同的单词个数。例如:对于句子“one  little  two  little  three  little  boys”,总共有5个不同的单词:one,  little,  two,  three,  boys。

说明:(1)由于句子当中包含有空格,所以应该用gets函数来输入这个句子;(2)输入的句子当中只包含英文字符和空格,单词之间用一个空格隔开;(3)不用考虑单词的大小写,假设输入的都是小写字符;(4)句子长度不超过100个字符。

输入  输入只有一行,即一个英文句子。

输出  输出只有一行,是一个整数,表示句子中不同单词的个数。

样例输入  one little two little three little boys

样例输出  5

#include<bits/stdc++.h>

using namespace std;

int main()  {  

   string ch;

   set <string> s;

   while(cin>>ch)  {

    s.insert(ch);  } 

   int len=s.size();

   cout<<len;

   return 0;  }

题目 2162: [信息学奥赛一本通-T1239]统计数字(C++)

某次科研调查时得到了n个自然数,每个数均不超过1500000000(1.5∗10^9)。已知不相同的数不超过10000个,现在需要统计这些自然数各自出现的次数,并按照自然数从小到大的顺序输出统计结果。

输入

第一行是整数 n,表示自然数的个数;

第2~n+1每行一个自然数。

输出

包含m行(m为n个自然数中不相同数的个数),按照自然数从小到大的顺序输出。每行输出两个整数,分别是自然数和该数出现的次数,其间用一个空格隔开。

样例输入

8

2

4

2

4

5

100

2

100

样例输出

2 3

4 2

5 1

100 2

#include<bits/stdc++.h>

using namespace std;

int main()  {  

   map <int,int> mp;

   int n,key;

   cin>>n;

   while(n--)  {

     cin>>key;

     mp[key]=mp[key]+1;  }

   map<int,int>::iterator it;

   for(it=mp.begin();it!=mp.end();it++)  {

     cout<<it->first<<" "<<it->second<<endl;  }

   return 0;  }

问题 1724: 后缀子串排序(C++)

题目描述

对于一个字符串,将其后缀子串进行排序,例如grain

其子串有:

grain

rain

ain

in

n

然后对各子串按字典顺序排序,即:ain,grain,in,n,rain

输入  每个案例为一行字符串。

输出  将子串排序输出

样例输入

grain

banana

样例输出

ain

grain

in

n

rain

a

ana

anana

banana

na

Nana

#include<bits/stdc++.h> 

using namespace std;

bool cmp(char *s1,char *s2)   {

    return strcmp(s1,s2)<0;  }

int main()   {

    char s[1000];  

    char *str[1000];   

    while(cin>>s)  {  

        int len=strlen(s);  

        for(int i=0;i<len;i++)  

            str[i]=s+i;

        sort(str,str+len,cmp);  

        for(int i=0;i<len;i++)   

            cout<<str[i]<<endl;   }

    return 0;  }

题目 2046: 输出全排列(C++)

给出一个数n,要求你输出1到n的全排列(要求字典序从小到大)(n<=10)

输入  一个数n

输出  包括若干行,每行包含n个数,表示相应的全排列值。

样例输入  4

样例输出

1 2 3 4

1 2 4 3

1 3 2 4

1 3 4 2

1 4 2 3

1 4 3 2

2 1 3 4

2 1 4 3

2 3 1 4

2 3 4 1

2 4 1 3

2 4 3 1

3 1 2 4

3 1 4 2

3 2 1 4

3 2 4 1

3 4 1 2

3 4 2 1

4 1 2 3

4 1 3 2

4 2 1 3

4 2 3 1

4 3 1 2

4 3 2 1

#include<bits/stdc++.h>

using namespace std;

int main()

{

 int a[10]={1,2,3,4,5,6,7,8,9,10};

 int n;

 scanf("%d",&n);

 do  {

   for(int i=0;i<n;i++)

     printf("%d ",a[i]);

   printf("\n");   }

while(next_permutation(a,a+n));

 return 0;  } 

题目 1913: [蓝桥杯][算法提高VIP]排列数(C++)

0、1、2三个数字的全排列有六种,按照字母序排列如下:

012、021、102、120、201、210

输入一个数n  求0~9十个数的全排列中的第n个(第1个为0123456789)。

输入  一行,包含一个整数n

输出  一行,包含一组10个数字的全排列

样例输入  1

样例输出  0123456789

#include<bits/stdc++.h>

#include<string.h>

using namespace std;

int main()  {

 int a[10]={0,1,2,3,4,5,6,7,8,9};

 int n,t=0;

 scanf("%d",&n);

 do  {

  t++;

  if(t==n)  {

   for(int i=0;i<10;i++)

         printf("%d",a[i]);

      break; }

 }while(next_permutation(a,a+10));

  return 0;  }

第11讲 并查集

题目 1732: 连通图(C++)

给定一个无向图和其中的所有边,判断这个图是否所有顶点都是连通的。

输入

每组数据的第一行是两个整数 n 和 m(0<=n<=1000)。n 表示图的顶点数目,m 表示图中边的数目。如果 n 为 0 表示输入结束。随后有 m 行数据,每行有两个值 x 和 y(0<x, y <=n),表示顶点 x 和 y 相连,顶点的编号从 1 开始计算。输入不保证这些边是否重复。

输出

对于每组输入数据,如果所有顶点都是连通的,输出"YES",否则输出"NO"。

样例输入

4 3

4 3

1 2

1 3

5 7

3 5

2 3

1 3

3 2

2 5

3 4

4 1

7 3

6 2

3 1

5 6

0 0

样例输出

YES

YES

NO

#include<bits/stdc++.h>

#include<string.h>

using namespace std;

int edge[1010][1010];

bool vis[1010];

int n,m;

int b,e;

void dfs(int step)  {

    vis[step]=true;

    for(int i=1;i<=n;++i) {

        if(vis[i]) continue;

        if(edge[step][i]) {

            vis[i]=true;

            dfs(i);    }    }    }

int main()  {

    cin >> n >> m;

    while(n!=0) {

        memset(edge,0,sizeof(edge));

        memset(vis,0,sizeof(vis));

        for(int i=0;i<m;++i) {

            cin >> b >> e;

            edge[b][e]=1;

            edge[e][b]=1;   }

        dfs(1);

        bool flag=true;

        for(int i=1;i<=n;++i) {

            if(!vis[i]) {

                flag=false;break;  }  }

        if(flag) cout << "YES" << endl;

        else cout << "NO" << endl;

        cin >> n >> m;  }

    return 0;  }

题目 2033: 网络互通(C++)

对于有n个住户的小区,我们决定拉网线构成一个网络。

每次,我们可以对两户人家进行连接,让他们之间进行通讯。当然,如果你在此基础之上继续拉网线,能通讯的人将会风一般地增加。

现在,我们已经给若干住户连上了网络,那么,请问某些住户之间是否能通信?

这里,如果两个住户可以连接,那么与其相连的住户同样能相互连接。

输入

第一行是两个数字n(n<100000),m(m<100000)表示住户个数与网线个数。

接下来m行,每行是两个数字xi,xj,表示xi与xj有网络连接。

接下来是两个数x,y表示询问的住户是否能通信。

输出

如果能通信输出Yes,否则输出No

样例输入

5 6

1 4

2 3

3 5

2 4

1 5

1 2

2 5

样例输出  Yes

#include<bits/stdc++.h>

using namespace std;

#define N 100005

int father[N];

void Init(int n)

{

for(int i=1;i<=n;i++)

 father[i]=i;


}

int Find(int x)

{

if(father[x]==x)

return x;

return Find(father[x]);

}

void Union(int x,int y)

{

int fx,fy;

fx=Find(x);

fy=Find(y);

if(x!=y)

father[fx]=fy;

}

int main()

{

int n,m;

scanf("%d%d",&n,&m);

Init(n);

while(m--)

{

int xj,xi;

scanf("%d%d",&xj,&xi);

Union(xj,xi);

}

int x,y;

scanf("%d%d",&x,&y);

if(Find(x)==Find(y))

printf("Yes\n");

else

printf("No\n");

return 0;  }

题目 1744: 畅通工程(C++)

某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇。省政府“畅通工程”的目标是使全省任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要互相间接通过道路可达即可)。问最少还需要建设多少条道路?

输入

测试输入包含若干测试用例。每个测试用例的第1行给出两个正整数,分别是城镇数目N ( < 1000 )和道路数目M;随后的M行对应M条道路,每行给出一对正整数,分别是该条道路直接连通的两个城镇的编号。为简单起见,城镇从1到N编号。 

    注意:两个城市之间可以有多条道路相通,也就是说

    3 3

    1 2

    1 2

    2 1

    这种输入也是合法的

    当N为0时,输入结束,该用例不被处理。

输出  对每个测试用例,在1行里输出最少还需要建设的道路数目。

 

样例输入

5 3

1 2

3 2

4 5

0

样例输出   1

#include <cstdio>

#include <deque>

#include<bits/stdc++.h>

using namespace std;

const long long INF = 1000000000;

const int MAX_N = 1000;

class disjointSet {

public:

    int parent[MAX_N];

    disjointSet() {

        for (int i = 0; i < MAX_N; ++i) {

            parent[i] = i;  } }

    int find(int x) {

        if (parent[x] == x) {

            return x;  }

 else {

            return parent[x] = find(parent[x]); } }

    bool unite(int x, int y) {

        int rootX = find(x);

        int rootY = find(y);

        if (rootX == rootY)  {

            return false;  }

        parent[rootY] = rootX;

        return true;  }

    bool same(int x, int y) {

        return find(x) == find(y);  }};

int disjointSetMain()

{

    int x, y, n, m;

    while (~scanf("%d", &n) && n != 0) {

        scanf("%d", &m);

        disjointSet parent = disjointSet();

        for (int i = 1; i <= m; ++i) {

            scanf("%d%d", &x, &y);

            parent.unite(x, y);  }

        int count = 0;

        for (int i = 1; i <= n; ++i) {

            if (parent.parent[i] == i) {

                count ++;   } }

        printf("%d\n", count - 1);  } }

int main()  {

    disjointSetMain();  }

P3367 【模板】并查集(C++)

如题,现在有一个并查集,你需要完成合并和查询操作。

输入格式

第一行包含两个整数 N,MN,M ,表示共有 NN 个元素和 MM 个操作。

接下来 MM 行,每行包含三个整数 Z_i,X_i,Y_iZi,Xi,Yi 。

当 Z_i=1Zi=1 时,将 X_iXi 与 Y_iYi 所在的集合合并。

当 Z_i=2Zi=2 时,输出 X_iXi 与 Y_iYi 是否在同一集合内,是的输出 Y ;否则输出 N 。

输出格式

对于每一个 Z_i=2Zi=2 的操作,都有一行输出,每行包含一个大写字母,为 Y 或者 N 。

输入

2 1 2

1 1 2

2 1 2

1 3 4

2 1 4

1 2 3

2 1 4

输出

Y

N

Y

 

#include<bits/stdc++.h>

using namespace std;

int i,j,k,n,m,s,ans,f[10010],p1,p2,p3;

int find(int k)  {

    if(f[k]==k)return k;

    return f[k]=find(f[k]);  }

int main()  {

    cin>>n>>m;

    for(i=1;i<=n;i++)

        f[i]=i;

    for(i=1;i<=m;i++)  {

        cin>>p1>>p2>>p3;

        if(p1==1)

            f[find(p2)]=find(p3);

        else

            if(find(p2)==find(p3))

                printf("Y\n");

            else

                printf("N\n");  }

    return 0;  }

P1551 亲戚

题目背景

若某个家族人员过于庞大,要判断两个是否是亲戚,确实还很不容易,现在给出某个亲戚关系图,求任意给出的两个人是否具有亲戚关系。

题目描述

规定:xx 和 yy 是亲戚,yy 和 zz 是亲戚,那么 xx 和 zz 也是亲戚。如果 xx,yy 是亲戚,那么 xx 的亲戚都是 yy 的亲戚,yy 的亲戚也都是 xx 的亲戚。

输入格式

第一行:三个整数 n,m,pn,m,p,(n,m,p \le 5000n,m,p≤5000),分别表示有 nn 个人,mm 个亲戚关系,询问 pp 对亲戚关系。

以下 mm 行:每行两个数 M_iMi,M_jMj,1 \le M_i,~M_j\le N1≤Mi, Mj≤N,表示 M_iMi 和 M_jMj 具有亲戚关系。

接下来 pp 行:每行两个数 P_i,P_jPi,Pj,询问 P_iPi 和 P_jPj 是否具有亲戚关系。

输出格式

pp 行,每行一个 Yes 或 No。表示第 ii 个询问的答案为“具有”或“不具有”亲戚关系。

输入

1 2

1 5

3 4

5 2

1 3

1 4

2 3

5 6

输出

Yes

No

#include<bits/stdc++.h>

using namespace std;

int n,m,q,f[10010],c,d,a,b;

int fd(int x)

{

 if(f[x]==x)

 return x;

 else 

 return  f[x]=fd(f[x]);  }

void hb(int x,int y)  {

 f[fd(y)]=fd(x);

 return ;  }

int main()  {

 scanf("%d%d%d",&n,&m,&q);

 for(int i=1;i<=n;i++)

 f[i]=i;

 for(int i=1;i<=m;i++)   {

      scanf("%d%d",&c,&d);

      hb(c,d);   }

 for(int i=1;i<=q;i++)  {

  scanf("%d%d",&a,&b);

  if(fd(a)==fd(b))

  printf("Yes\n");

  else

  printf("No\n");  }

 return 0;  }

P1892 [BOI2003]团伙

给定 nn 个人,他们之间有两个种关系,朋友与敌对。可以肯定的是:

与我的朋友是朋友的人是我的朋友

与我敌对的人有敌对关系的人是我的朋友

现在这 nn 个人进行组团,两个人在一个团队内当且仅当他们是朋友。

求最多的团体数。

输入格式

第一行一个整数 nn 代表人数。

第二行一个整数 mm 代表每个人之间的关系。

接下来 mm 行每行一个字符 optopt 与两个整数 p,qp,q

如果 optopt 为 F 代表 pp 与 qq 为朋友。

如果 optopt 为 E 代表 pp 与 qq 为敌人。

输出格式   一行一个整数代表最多的团体数。

输入

4

E 1 4

F 3 5

F 4 6

E 1 2

输出   3

 

#include<bits/stdc++.h>

using namespace std;

int n,m,f[1001],enm[1001];   

int find(int x)            

{

    while(f[x]!=x) x=f[x]; 

    return x;  }

void hebing(int x,int y)    {

    x=find(x);y=find(y);  

    if(x==y) return;

    f[y]=x;

    return;  }

int main()  {

     cin>>n>>m;

     for(int i=1;i<=n;i++)

        f[i]=i;

    for(int i=1;i<=m;i++)   {

        int p,q;

        char c;

        cin>>c>>p>>q;

        if(c=='F') hebing(p,q);   

        else {

                if(enm[p]==0) enm[p]=find(q);

              else hebing(q,enm[p]);   

              if(enm[q]==0) enm[q]=find(p);

              else hebing(p,enm[q]);   }  }

    int count[1001]={0};

    for(int i=1;i<=n;i++)

        count[find(i)]++;

    int cnt=0;

    for(int i=1;i<=n;i++)

        if(count[i]) cnt++;  

    cout<<cnt;   } 

第12讲 二分查找和快速幂

题目 1710: 数据结构-静态表的顺序查找
用顺序表或者线性链表表示静态查找表时,搜索函数可以采用顺序查找来实现。
通常顺序查找的查找过程是从表中的自后一个记录开始,逐个将记录的关键字和给定的查找值进行比较,如果某个记录的关键字与给定的值比较相等,则说明查找成功;否则如果直到第一个记录,所有的关键字都与给定的值不相等,说明表中没有响应的记录,查找失败。
其查找过程可以描述如下:

在本题中,读入一串整数,另外给定多次查询,判断每一次查询是否找到了相应的整数,如果找到则输出整数相应的位置。
输入
输入的第一行包含2个正整数n和k,分别表示共有n个整数和k次查询。其中n不超过500,k同样不超过500。
第二行包含n个用空格隔开的正整数,表示n个原始记录。
第三行包含k个用空格隔开的正整数,表示k次查询的目标。
输出
只有1行,包含k个整数,分别表示每一次的查询结果。如果在查询中找到了对应的整数,则输出其相应的位置,否则输出-1。
请在每个整数后输出一个空格,并请注意行尾输出换行。
样例输入   8 31 3 5 7 8 9 10 159 2 5
样例输出   5 -1 2 

#include<stdio.h>

int serach(int a[],int n,int x)  {

 for(int i=n-1;i>=0;i--)  {

  if(x==a[i]) return i;   }

 return -1;  }

int main()  {

 int n,k;

 int a[505];

 scanf("%d%d",&n,&k);

 for(int i=0;i<n;i++)

   scanf("%d",&a[i]);

 while(k--)   {

  int x, pos;

  scanf("%d",&x);

  pos=serach(a,n,x);

  printf("%d ",pos);   }

 return 0; }

题目 1711: 数据结构-有序表的折半查找(1710的类似题)
用有序表表示静态查找表时,通常检索函数可以用折半查找来实现。
折半查找的查找过程是:首先确定待查记录所在的范围,然后逐步缩小范围直到找到或者确定找不到相应的记录为止。而每次需要缩小的范围均为上一次的一半,这样的查找过程可以被称为折半查找。
其查找过程可以描述如下:

在本题中读入一串有序的整数,另外给定多次查询,判断每一次查询是否找到了相应的整数,如果找到则输出整数相应的位置。
输入
输入的第一行包含2个正整数n和k,分别表示共有n个整数和k次查询。其中n不超过1000,k同样不超过1000。
第二行包含n个用空格隔开的正整数,表示n个有序的整数。输入保证这n个整数是从小到大递增的。
第三行包含k个用空格隔开的正整数,表示k次查询的目标。
输出
只有1行,包含k个整数,分别表示每一次的查询结果。如果在查询中找到了对应的整数,则输出其相应的位置,否则输出-1。
请在每个整数后输出一个空格,并请注意行尾输出换行。
样例输入  8 31 3 5 7 8 9 10 159 2 5
样例输出  5 -1 2 

#include <stdio.h>

int search(int a[],int n,int x)  {

for(int i=0;i<n;i++)  {

if(x==a[i])  return i;  }

return -1;  }

int main()  {

int a[1005];

int n,k;

scanf("%d%d",&n,&k);

for(int i=0;i<n;i++)

    scanf("%d",&a[i]);

while(k--)  {

int x,pos;

scanf("%d",&x);

pos=search(a,n,x);

printf("%d ",pos);  }

return 0;  }

P2249 【深基13.例1】查找

 

#include <stdio.h>

int a[1000005];

int search(int a[],int n,int x)  

{

int L,R,mid;

L=0; R=n-1;

while(L<=R)  {

mid=(L+R)/2;

if(x>a[mid]) L=mid+1; 

else  R=mid-1;  }

return a[L]==x? L+1 : -1;   }

int main()   {

int n,k;

scanf("%d%d",&n,&k);

for(int i=0;i<n;i++)

    scanf("%d",&a[i]);

while(k--)  {

int x,pos;

scanf("%d",&x);

pos=search(a,n,x);

printf("%d ",pos);  }

return 0;  }

题目 2163: 信息学奥赛一本通T1240-查找最接近的元素

在一个非降序列中,查找与给定值最接近的元素。

输入

第一行包含一个整数n,为非降序列长度。1 ≤ n ≤ 100000。

第二行包含n个整数,为非降序列各元素。所有元素的大小均在0-1,000,000,000之间。

第三行包含一个整数m,为要询问的给定值个数。1 ≤ m ≤ 10000。

接下来m行,每行一个整数,为要询问最接近元素的给定值。所有给定值的大小均在0-1,000,000,000之间。

输出

m行,每行一个整数,为最接近相应给定值的元素值,保持输入顺序。若有多个值满足条件,输出最小的一个。

样例输入

3

2 5 8

2

10

5

样例输出

8

5

#include<stdio.h> 

int a[1000005];

int serach(int a[],int n,int x)  {

 int L,R,mid;

 L=0;R=n-1;

 while(L<=R)  {

  mid=(L+R)/2;

  if(x>a[mid]) L=mid+1;

  else R=mid-1;   }

 if(L==0) return a[0];

 if(L==n) return a[n-1];

 if(x-a[L-1]<=a[L]-x) return a[L-1];

 else return a[L];   }

int main()  {

 int n,k;

 scanf("%d",&n); 

 for(int i=0;i<n;i++)

   scanf("%d",&a[i]);

 scanf("%d",&k);

 while(k--)   {

  int x, pos;

  scanf("%d",&x);

  pos=serach(a,n,x);

  printf("%d\n",pos);  }

 return 0;  }

题目 2088: 蓝桥杯算法提高VIP-快速幂

给定A, B, P,求(A^B) mod P。

输入

输入共一行。

第一行有三个数,N, M, P。

输出

输出共一行,表示所求。

共10组数据

对100%的数据,A, B为long long范围内的非负整数,P为int内的非负整数。

样例输入  2 5 3

样例输出  2

#include<stdio.h>

typedef long long LL;

LL qpow(LL a,LL b,LL p)

{

 if (b==0) return 1%p;

 if(b%2==1) return (a%p)* qpow(a,b-1,p)%p %p;

 else{

  LL t=qpow(a,b/2,p)%p;

  return (t*t)%p;  } }

int main()

{

 LL a,b,p,t;

 scanf("%lld%lld%lld",&a,&b,&p);

 t=qpow(a,b,p);

 printf("%lld",t);

 return 0;  }

P1226 【模板】快速幂||取余运算

 

#include <iostream>

using namespace std;

int qmi(int a, int b, int p)

{

    int res = 1;

    while (b)  {

        if (b & 1)

            res = (long long) res * a % p;

        b >>= 1;

        a = (long long) a * a % p;   }

    return res;  }

int main()  {

    int a, b, p;

    scanf("%d%d%d", &a, &b, &p);

    printf("%d^%d mod %d=%d",a,b,p,qmi(a, b, p));

    return 0;  }

————————————————————————

题目 1002: [编程入门]三个数最大值

编写一个程序,输入a、b、c三个值,输出其中最大值。

输入    一行数组,分别为a b c

输出    a b c其中最大的数

样例输入    10 20 30

样例输出    30

#include<stdio.h>

int main()

{

int a,b,c,max;

scanf("%d%d%d",&a,&b,&c);

max=a;

if(b>max) max=b;

if(c>max) max=c;

printf("%d\n",max);

return 0;  }

题目 1057: 二级C语言-分段函数

题目描述
有一个函数如下,写一程序,输入x,输出y值。保留两位小数
输入    无
输出    无
样例输入    1
样例输出    1.00 

#include<stdio.h>

int main()

{

 float x,y;

 scanf("%f",&x);

 if(x<1) y=x;

  else if(x<10) y=2*x-1;

   else if(x>=10) y=3*x-11;

 printf("%.2f\n",y);

 return 0;  }

题目 1058: 二级C语言-求偶数和

题目描述

编制程序,输入n个整数(n从键盘输入,n>0),输出它们的偶数和。

输入  无

输出  无

样例输入

2

1 2

样例输出

2

#include<stdio.h>

int main()

{

 int n,i,x;

 long long sum=0;

 scanf("%d",&n);

 for(i=1;i<=n;i++)  {

  scanf("%d",&x);

  if(x%2==0)

   sum=sum+x;  }

 printf("%lld",sum);

 return 0;  }

题目 1040: [编程入门]实数的打印

题目描述

请设计输出实数的格式,包括:⑴一行输出一个实数;⑵一行内输出两个实数;⑶一行内输出三个实数。实数用"6.2f"格式输出。

输入    一个实数,float范围

输出    输出3行,第一行打印一遍输入的数,第二行打印两遍,第三行打印三遍。 第二行和第三行,用空格分隔同一行的数字。 实数用"6.2f"格式输出。

样例输入    0.618

样例输出

  0.62

  0.62   0.62

  0.62   0.62   0.62

#include<stdio.h>

int main()

{

 float x;

 scanf("%f",&x);

 printf("%6.2f\n",x);

 printf("%6.2f %6.2f\n",x,x);

 printf("%6.2f %6.2f %6.2f",x,x,x);

 return 0;  }

题目 1434: 蓝桥杯历届试题-回文数字

观察数字:12321,123321  都有一个共同的特征,无论从左到右读还是从右向左读,都是相同的。这样的数字叫做:回文数字。
本题要求你找到一些5位或6位的十进制数字。满足如下要求:
该数字的各个数位之和等于输入的整数。

输入: 一个正整数  n  (10< n< 100),  表示要求满足的数位和

输出:

若干行,每行包含一个满足要求的5位或6位整数。 
数字按从小到大的顺序排列。 
如果没有满足条件的,输出:-1 

#include<stdio.h>

int main()

{

   int i,j,n,sum1,sum2,count=0;

   scanf("%d",&n);

   for(i=10000;i<=999999;i++)  {

     j=i;

     sum1=0;

     sum2=0;

     while(j>0)  {

       sum1=sum1*10+j%10;

       sum2+=j%10;

       j=j/10;  }

    if(sum1==i&&sum2==n)  {

      count++;

      printf("%d\n",i);  }  }

   if(count==0)  {

     printf("-1\n");  }

   return 0;  }

题目 1053: 二级C语言-平均值计算

输入10个整数,求它们的平均值,并输出大于平均值的数据的个数。

输入:  10个数

输出:  大于平均数的个数

样例输入:   1 2 3 4 5 6 7 8 9 10

样例输出:   5

#include<stdio.h>

int main()

{

 int i,sum=0,k,t=0;

 int a[10];

 for(i=0;i<10;i++)  {

  scanf("%d",&a[i]);

  sum+=a[i];  }

 k=sum/10;

 for(i=0;i<10;i++)  {

  if(a[i]>k) t++;  }

 printf("%d",t);

 return 0;  }

题目 1246: 第几天

这是一个很经典的题,给定一个日期,输出这个日期是该年的第几天。

输入

输入数据有多组,每组占一行,数据格式为YYYY/MM/DD组成(见样例) ,另外,可以向你确保所有的输入数据是合法的。

输出

对于每组输入数据,输出一行,表示该日期是该年的第几天。

样例输入:

1985/1/20

2006/3/12

样例输出:

20

71

#include <stdio.h>

#include <stdlib.h>

int main()

{

    int year,month,Day;

    while(scanf("%d/%d/%d",&year,&month,&Day)!=EOF)   {

        int i,day[12]={31,28,31,30,31,30,31,31,30,31,30,31};

        for(i=0;i<month-1;i++)

            Day+=day[i];

        if((year%4==0&&year%100!=0)||year%400==0)

            Day++;

        printf("%d\n",Day);  }

    return 0;  }

题目 1065: 二级C语言-最小绝对值

题目描述

输入10个数,找出其中绝对值最小的数,将它和最后一个数交换,然后输出这10个数。

输入    十个数

输出    交换后的十个数

样例输入    10 2 30 40 50 60 70 80 90 100

样例输出    10 100 30 40 50 60 70 80 90 2

#include<stdio.h>

#include<math.h>

int main()

{

   int i,a[10],min,t;

   for(i=0;i<10;i++)

     scanf("%d",&a[i]);

    min=a[0];

    for(i=0;i<10;i++)

     if(fabs(a[i])<fabs(min))

       min=a[i];

    for(i=0;i<10;i++)

     if(a[i]==min)  {

      t=a[i];

      a[i]=a[9];

      a[9]=t;

      break;  }

  for(i=0;i<10;i++)

    printf("%d ",a[i]);

 return 0;  }

题目 1126: C语言训练-字符串正反连接

所给字符串正序和反序连接,形成新串并输出

输入   任意字符串(长度<=50)

输出   字符串正序和反序连接所成的新字符串

样例输入   123abc

样例输出   123abccba321

#include<stdio.h>

#include<string.h>

int main()

{

 char s[60],t[120];

 int i,j;

 gets(s);

 for(i=0,j=0;s[i]!='\0';i++,j++) {

  t[j]=s[i]; }

 for(i=i-1;i>=0;i--,j++) {

  t[j]=s[i]; }

 t[j]='\0';

 printf("%s",t);

 return 0;  }

题目 1206: 字符串问题

字符串处理在计算机中有很多复杂的操作,但是这些复杂的操作都是由基本的字符串操作复合而成,要求编写一字符串颠倒的程序,把字符串中的字符颠倒位置。

输入    输入一

点赞(0)
 

0.0分

0 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论