vijos题解

Vijos题解

题库地址:https://vijos.org/p

 

P1001 谁拿了最多奖学金

题意:按照指定要求计算奖学金,直接用if判断即可

#include<iostream>
using namespace std;
struct STUDENT{
  string name;
  int n1,n2;
  char xsgb,xb;
  int lw;
  int jxj;
};
int main(){
  int n;
  int sum=0;
  cin>>n;
  struct STUDENT stu[n];
  struct STUDENT zdz;
  zdz.jxj=0;
  for(int i=0;i<n;i++)stu[i].jxj=0;
  for(int i=0;i<n;i++){
    cin>>stu[i].name>>stu[i].n1>>stu[i].n2>>stu[i].xsgb>>stu[i].xb>>stu[i].lw;
    if(stu[i].n1>80&&stu[i].lw>=1)stu[i].jxj+=8000;
    if(stu[i].n1>85&&stu[i].n2>80)stu[i].jxj+=4000;
    if(stu[i].n1>90)stu[i].jxj+=2000;
    if(stu[i].n1>85&&stu[i].xb=='Y')stu[i].jxj+=1000;
    if(stu[i].n2>80&&stu[i].xsgb=='Y')stu[i].jxj+=850;
    if(stu[i].jxj>zdz.jxj)zdz=stu[i];
    sum+=stu[i].jxj;
  }
  cout<<zdz.name<<endl<<zdz.jxj<<endl<<sum;
  return 0;
  
}
View Code

 

P1007 绕钉子的长绳子

题意:求两点距离

 

#include<bits/stdc++.h>
using namespace std;
const double pi=3.14159265359;
struct POINT{
  double x;
  double y;
};
int n;
double r,ans=0;
POINT p[100001];
inline double dis(POINT a,POINT b){
  double x=a.x-b.x;
  double y=a.y-b.y;
  return sqrt(x*x+y*y);//两点距离
}
int main(){
  cin>>n>>r;
  for(int i=0;i<n;i++)cin>>p[i].x>>p[i].y;
  for(int i=1;i<n;i++){
    ans+=dis(p[i],p[i-1]);
  }
  ans+=dis(p[0],p[n-1]);//第一个和最后一个之间
  ans+=2*pi*r;//绕钉子的长度
  cout<<fixed<<setprecision(2)<<ans<<endl;
  return 0;
}
View Code

 

 

 

 

P1025 小飞侠的游园方案

题意:同P1104,01背包问题,只是输入的顺序改了

#include<bits/stdc++.h>
using namespace std;
int Time[10000],Value[10000];
int dp[100000];
int main(){
    int a,n;
    cin>>n>>a;
    for(int i=0;i<n;i++){
        cin>>Value[i]>>Time[i];
    }
    for(int i=0;i<n;i++){
        for (int j=a;j>=Time[i];j--){
            dp[j]=max(dp[j],dp[j-Time[i]]+Value[i]);
        }
    }
    cout<<dp[a];
    return 0;
}
View Code

 

 P1040 高精度乘法

题意:标准高精度乘法 

#include<iostream>
#include<cstring>
using namespace std;
string sa,sb;
int a[100000],b[100000],c[100000],lena,lenb,lenc;
int main() {
  cin>>sa;
  cin>>sb;  //读入数字a和b,将其存入字符串sa和sb中
  lena=sa.size();
  lenb=sb.size(); //求出2个数字的位数长度
  for (int i=1;i<=lena;i++) a[i]=sa[lena-i]-48;
  for (int i=1;i<=lenb;i++) b[i]=sb[lenb-i]-48;
  //将字符串中的存储的数字字符转换为数字逆序存于a、b数组中
  //a、b数组是从第1格开始存储,方便后续计算
  for (int i=1;i<=lena;i++)
    for (int j=1;j<=lenb;j++) {
      c[i+j-1]+=a[i]*b[j];
      c[i+j]+=c[i+j-1]/10;
      c[i+j-1]%=10; 
    }
  lenc=lena+lenb;  
  while (c[lenc]==0 && lenc>1) lenc--;
  //从高位开始找非零值,确定乘积的位数 
  for (int i=lenc;i>0;i--) cout<<c[i];  //输出答案
  return 0;
}
View Code

 

P1058 粘贴文本

题意:剪切和粘贴操作,模拟

Windows操作系统中有一个“剪贴板”的功能,把待粘贴的东西临时存放在那里。我们也可以使用类似的思路。主要步骤:

第一步,把文本放进剪贴板。

第二步,把已经剪切掉的部分后面的东西前移上去。

第三步,把要粘贴的地方所有后面的文本后移,腾出位子留给要粘贴的文本。

第四步,把剪贴板上的文本粘贴进去。

所有步骤都可以使用memmove操作。关于memmove和memcpy的区别可以自行网上搜索。

代码:

 

#include<bits/stdc++.h>
using namespace std;
const int maxn=100010;
int text[maxn],temp[maxn];
int n,k,a,b,c;
int main(){
    cin>>n>>k;
    for(int i=1;i<=n;i++)text[i]=i;
    for(int i=0;i<k;i++){
        cin>>a>>b>>c;
        int len=b-a+1;
        memmove(temp,text+a,sizeof(int)*len);//待粘贴数字
        memmove(text+a,text+b+1,sizeof(int)*(n-b));//把数字剪切掉
        memmove(text+c+len+1,text+c+1,sizeof(int)*(n-len-c));//粘贴位置数字后移
        memmove(text+c+1,temp,sizeof(int)*len);//粘贴进去
    }
    for(int i=1;i<=10;i++)cout<<text[i]<<endl;
    return 0;
}
     
View Code

 

P1066 弱弱的战壕

题意:判断一个点是否在另一个点的左下角,枚举

#include<bits/stdc++.h>
using namespace std;
const int MAXN=20000+10;
int n;
int x[MAXN],y[MAXN],ans[MAXN],ans1[MAXN];
int main(){
    cin>>n;
    for(int i=1;i<=n;i++)cin>>x[i]>>y[i];
    for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++){
            if(i!=j&&(x[i]>=x[j])&&(y[i]>=y[j]))ans[i]++;//统计第i个战壕的保护对象数量
        }
    }
    for(int i=1;i<=n;i++)ans1[ans[i]]++;//保护数量为ans[i]的战壕个数
    for(int i=0;i<n;i++)cout<<ans1[i]<<endl;
    return 0;
}
View Code

 

P1093 文科生的悲哀

题意:本质就是求斐波那契数列的第n项,递推法

#include<iostream>
using namespace std;
int n,fibo[100000];
int main(){
  cin>>n;
  fibo[1]=fibo[2]=1;
  for(int i=3;i<=n;i++)fibo[i]=(fibo[i-1]+fibo[i-2])%7654321;
  cout<<fibo[n];
  return 0;
}
View Code

 

P1096 津津的储蓄计划

题意:使用变量记录在身边的钱和存的钱

PS:存的钱就是所有的钱向下舍入到百位,例如320向下舍入到百位就是300,也就是整百的钱。

 

 

 

(一些关于向下舍入的文字,来源《30天自制操作系统》第十章)

#include<bits/stdc++.h>
using namespace std;
int a[12];
int main(){
    int money=0/*自己的*/,cun=0;/*存的钱*/
    for(int i=0;i<12;i++){
        cin>>a[i];
    }
    for(int i=0;i<12;i++){
        money+=300;
        money-=a[i];
        if(money<0){
            cout<<"-"<<i+1<<endl;
            return 0;
        }
        cun+=money/100*100;//舍入到整百位
        money%=100;
    }
    cout<<money+cun*1.2<<endl;
    return 0;
}
View Code

 

 

P1097 合并果子

题意:每次找最小的两堆进行合并,边合并边排序

#include <iostream>
#include <algorithm>
using namespace std;
int main(){
  int n;
  cin>>n;
  int a[n];
  for(int i=0;i<n;i++)cin>>a[i];
  int ans=0;
  sort(a,a+n);
  for(int i=1;i<n;i++){
    a[i]=a[i]+a[i-1];
    ans+=a[i];
    sort(a+i,a+n);
  }
  cout<<ans<<endl;
  return 0;
}
View Code

(来源CCF基础篇)维护小根堆

#include<bits/stdc++.h>
using namespace std;
int N,ans;
const int MAXN=10000+10;
int a[MAXN];
bool cmp(int a,int b){
  return a>b;
}
int main(){
  cin>>N;
  for(int i=0;i<N;i++)cin>>a[i];
  make_heap(a,a+N,cmp);//建小根堆
  ans=0;
  for(int i=N;i>1;i--){//每次合并2堆,一共N-1次
    pop_heap(a,a+i,cmp);
    pop_heap(a,a+i-1,cmp);//最小值弹到堆底
    a[i-2]+=a[i-1];
    ans+=a[i-2];
    push_heap(a,a+i-1,cmp);//新元素在a[i-2],插入堆中
  }
  cout<<ans<<endl;
  return 0;
}
View Code

 

P1103 校门外的树

题意:开数组把树的状态记录下来

#include <bits/stdc++.h>
using namespace std;
int main(){
    int l,m;
    cin>>l>>m;
    int left,right,road[100000];
    for(int i=1;i<=m;i++){
        cin>>left>>right;
        for(int j=left;j<=right;j++)road[j]=1;
    }
    int s=0;
    for(int i=0;i<=l;i++){
        if(!road[i])s++;
    }
    cout<<s;
    return 0;
}
View Code

 

P1104 采药

题意:标准01背包问题

#include<bits/stdc++.h>
using namespace std;
int Time[10000],Value[10000];
int dp[100000];
int main(){
    int a,n;
    cin>>a>>n;
    for(int i=0;i<n;i++){
        cin>>Time[i]>>Value[i];
    }
    for(int i=0;i<n;i++){
        for (int j=a;j>=Time[i];j--){
            dp[j]=max(dp[j],dp[j-Time[i]]+Value[i]);
        }
    }
    cout<<dp[a];
    return 0;
}
View Code

 

P1113 不高兴的津津

题意:找最大值

#include <iostream>
using namespace std;
int bgx,a,b,zdz;
int main(){
  for(int i=0;i<7;i++){
    cin>>a>>b;
    if(a+b>8&&a+b>bgx)bgx=a+b,zdz=i+1;
  }
  cout<<zdz;
  return 0;
}
View Code

 

P1115 火星人

题意:求一组数的第N个排列,可以用STL的next_permutation函数

#include<bits/stdc++.h>
using namespace std;
int A[10000];
int n;
int a;
int main(){
  int n;
  cin>>n>>a;
  for(int i=0;i<n;i++)
    cin>>A[i];
  for(int i=0;i<a;i++)
    next_permutation(A,A+n);
  for(int i=0;i<n;i++)
    cout<<A[i]<<' ';
  return 0;
}
View Code

 

P1116 一元三次方程求解

题意:枚举一元方程的解

PS:算式的绝对值小于0.01就可以看做是等于

 

#include<bits/stdc++.h>
using namespace std;
int main(){
  float a,b,c,d,x;
  scanf("%f%f%f%f",&a,&b,&c,&d);
  for(int i=-10000;i<=10000;i++){//枚举解
    x=i/100.0;
    if(fabs(a*x*x*x+b*x*x+c*x+d)<=0.01) printf("%.2f ",x);//可以看做是等于0
  }
  return 0;
}
View Code

 

P1123 均分纸牌

题意: 模拟,使每堆纸牌到达平均值

#include<bits/stdc++.h>
using namespace std;
int main(){
    int n;cin>>n;
    int a[n],tot=0;
    for(int i=0;i<n;i++)cin>>a[i],tot+=a[i];
    int ave=tot/n;
    int ans=0;
    for(int i=0;i<n;i++){//模拟,把所有的移动到平均值
        if(a[i]>ave)a[i+1]+=a[i]-ave,a[i]=ave,ans++;//太多,向右移动
        else if(a[i]<ave)a[i+1]-=ave-a[i],a[i]=ave,ans++;//太少,向右移动
    }
    cout<<ans;
    
}
View Code

 

 

P1127 级数求和

题意:计算1+1/2+1/3+1/4+...+1/n,用一个while循环即可

#include<bits/stdc++.h>
using namespace std;
int main(){
    int k;
    cin>>k;
    double ans=0;
    int i=1;
    while(ans<k){
        ans+=1.0/i;
        ++i;
    }
    cout<<i-1;
}
View Code

 

P1130 数的计数

题意:用动态规划的方法,我们假设第i个数生成的个数为dp[i],那么它的左边还可以加上的数的个数就是dp[1]+dp[2]+...+dp[i/2],dp[1]=1

#include<bits/stdc++.h>
using namespace std;
int dp[1100];
int main(){
  int n;
  cin>>n;
  dp[1]=1;
  for(int i=1;i<=n;i++){
    int tot=0;
    for(int j=1;j<=i/2;j++)tot+=dp[j];
    dp[i]=tot+1;
  }
  cout<<dp[n];
}
View Code

 

 

P1199 核弹危机

题意:四层循环枚举左上角和右下角点,计算核弹范围内房屋数量最大值

#include<bits/stdc++.h>
using namespace std;
char a[10000][10000];
int m,n;
int main(){
  cin>>m>>n;
  for(int i=0;i<m;i++){
    for(int j=0;j<m;j++){
      cin>>a[i][j];
    }
  }
  int total=0,ans=0;
  for(int i=0;i<m;i++){//攻击的左上角点
    for(int j=0;j<m;j++){
      total=0;
      for(int k=i;k<=i+n-1;k++){//攻击的右下角点
        for(int l=j;l<=j+n-1;l++){
          if(a[k][l]=='#')total++;//房屋
          if(total>ans)ans=total;
        }
      }
    }
  }
  cout<<ans;
  return 0;
}
View Code

 

 

P1201 高低位交换

方法一:使用C++自带的位运算,先用<<和>>进行移位,得到高16位和低16位,然后用“或”运算进行合并

#include<bits/stdc++.h>
using namespace std;
int main(){
    unsigned int a;
    cin>>a;
    unsigned int b=(a<<16)|(a>>16);//直接用位运算
    cout<<b;
}
View Code

 方法二:模拟

十进制转换为二进制:循环做除法,倒取余数

二进制转换为十进制:按位组合,第32-i位的权值是2的i-1次方

其实这道题目在转换的时候不需要把二进制的顺序倒过来,在转换为十进制的时候也反一下,i位的权值为2的i-1次方,这样即可

#include<bits/stdc++.h>
using namespace std;
int binary[33];
int main(){
    int x=0,len=0;
    cin>>x;
    while(x){//十转二 
        binary[++len]=x%2;
        x/=2;
    }
    for(int i=1;i<=16;i++)swap(binary[i],binary[i+16]);
    unsigned int ans=0;
    for(int i=1;i<=32;i++)ans+=pow(2,i-1)*binary[i];//二转十 
    cout<<ans;
    return 0;
}
View Code

 

 

P1211 生日日数

题意:计算指定日期过10000天后的日期,注意判断闰年和日期进制,可以使用数组保存每个月的天数

判断闰年:“四年一闰,百年不闰,四百年又一闰”

每个月的天数:口诀:“一三五七八十腊,三十一日总不差” 。其中的“腊月”就是十二月。

另外注意保存天数的数组不可以用const,因为要根据闰年来修改2月是28天还是29天。

#include<bits/stdc++.h>
using namespace std;
int daynum[]={0,31,28,31,30,31,30,31,31,30,31,30,31};
int main(){
  int year,month,day;
  cin>>year>>month>>day;
  for(int i=0;i<10000;i++){
    if((year%4==0&&year%100!=0)||year%400==0)daynum[2]=29;
    else daynum[2]=28;
    ++day;
    if(day>daynum[month]){
      day=1;++month;
    }
    if(month>12){
      month=1;++year;
    }
  }
  printf("%d-%d-%d",year,month,day);
}
View Code

 

P1217 乒乓球

题意:根据不同比分制进行比分计算

PS:最后一次完整比赛输出完后还要再输出一次

#include<bits/stdc++.h>
using namespace std;
int main(){
  char c;string s;
  int w=0,l=0;
  while((c=getchar())!='E'){
    s+=c;
  }
  for(int i=0;i<s.size();i++){
    if(s[i]=='W')++w;
    else if(s[i]=='L')++l;
    if((w>=11||l>=11)&&abs(w-l)>=2)printf("%d:%d\n",w,l),w=0,l=0;
  }
  printf("%d:%d\n\n",w,l);
  w=0;l=0;
  for(int i=0;i<s.size();i++){
    if(s[i]=='W')++w;
    else if(s[i]=='L')++l;
    if((w>=21||l>=21)&&abs(w-l)>=2)printf("%d:%d\n",w,l),w=0,l=0;
  }
  printf("%d:%d\n\n",w,l);
  return 0;
}
View Code

 

P1223 麦林数

题意:高精度计算2^P-1,快速幂

 

#include<bits/stdc++.h>
using namespace std;
const int Max_M=510;
int n;
int Out[Max_M];
int Power[2*Max_M];

void DFS(int n){
  int i,j;
  if (n==0) return;
  DFS(n/2);
  for (i=1;i<=500;i++)
    for (j=1;j<=500;j++)//分拆为两个数 
      if(n%2==0) Power[i+j-1]=Power[i+j-1]+Out[i]*Out[j];//偶数,分拆两个 
      else
        Power[i+j-1]=Power[i+j-1]+Out[i]*Out[j]*2;//奇数,分拆两个,再加上剩余的2 
  for(i=1;i<=500;i++){//进位 
    Out[i]=Power[i]%10;
    Power[i+1]=Power[i+1]+Power[i]/10;
  }
  memset(Power,0,sizeof(Power));
}

int main(){
  //freopen("mason.in","r",stdin);freopen("mason.out","w",stdout);
  cin>>n;
  cout<<int((log(2)/log(10)*n)+1)<<endl;//对数换底,求位数 
  Out[1]=1;
  DFS(n);
  for(int i=500;i>=2;i--){
    cout<<Out[i];
    if (i%50==1) cout<<endl;
  }
  cout<<Out[1]-1<<endl;
  return 0;
}
View Code

 

 

 

 

P1229 分解因式

题意:计算因数个数

#include <iostream>
using namespace std;
int f(int x){
    int ans=0;
    for(int i=1;i<=x;i++)
        if(x%i==0)ans++;
    return ans;
}
int main(){
    int n;
    cin>>n;
    for(int i=1;i<=20000;i++){
        if(f(i)==n){
            cout<<i;return 0;
        }
    }
    cout<<"NO SOLUTION";
    return 0;
}
View Code

 

P1258 多项式表示

PS:与P1812有区别,模拟

 

#include<bits/stdc++.h>
using namespace std;
int main(){
  int flag=0,data;
  for(int i=8;i>=0;i--){
    scanf("%d",&data);
    if(data==0)continue;
    if(data*flag==0&&data<0)printf("-");//首项无空格
    else if(data<0) printf(" - ");
    if(data*flag>0) printf(" + ");
    if(data*data-1>0) printf("%d",data>0 ? data : -data);//负数已经有负号了,输出相反数
    if(i>1) printf("x^%d",i);//有次数
    else if (i==1) printf("x");
    else if (i==0&&(data==1||data==-1)) printf("1");
    flag=1;
  }
  if(flag==0)printf("0");
  return 0;//全零标记
}
View Code

 

P1282 佳佳的魔法照片

题意:多关键字排序

 

#include<bits/stdc++.h>
using namespace std;
struct people{
  int w,d;
}p[50010];
bool cmp(people a,people b){
  if(a.w!=b.w)return a.w>b.w;
  return a.d<b.d;
}
int main(){
  int n,k;
  cin>>n>>k;
  int E[10];
  for(int i=0;i<10;i++)cin>>E[i];
  for(int i=0;i<n;i++)cin>>p[i].w,p[i].d=i+1;
  sort(p,p+n,cmp);
  for(int i=0;i<n;i++)p[i].w+=E[i%10];
  sort(p,p+n,cmp);
  for(int i=0;i<k;i++)cout<<p[i].d<<' ';
}
View Code

 

P1307 黑皮的正方形

题意:计算总正方形的个数,如果边长为n,那么单个组成的正方形个数为n*n,四个拼成的正方形个数为(n-1)*(n-1),九个拼成的正方形个数为(n-2)*(n-2),以此类推,总数就是从1-n的平方相加。

#include<bits/stdc++.h>
using namespace std;
long long n,ans;
int main(){
    cin>>n;
    for(int i=1;i<=n;i++)ans+=i*i;
    cout<<ans;
}
View Code

 

 

P1316 明明的随机数

题意:先去掉重复的,然后做排序,直接STL的sort函数即可

#include<iostream>
#include<algorithm>
using namespace std;
int main(){
  int n;
  scanf("%d",&n);
  int f=n;
  int a[n];
  for(int i=0;i<n;i++)scanf("%d",a+i);
  for(int i=0;i<n;i++){
    for(int j=i+1;j<n;j++){
      if(i!=j && a[i]==a[j] && a[i]!=-1 && a[j]!=-1 )f--,a[j]=-1;//重复的标记为-1
    }
  }
  sort(a,a+n);
  printf("%d\n",f);
  for(int i=0;i<n;i++){
    if(a[i]!=-1)printf("%d ",a[i]);
  }
  return 0;
}
View Code

 

P1389 婚礼上的小杉

题意:多关键字排序

PS:本题的难点在于如何输入数据,我们可以同时使用cin(输入数据)以及getchar(判断行是否结束)

 

#include<bits/stdc++.h>
using namespace std;
struct LETTER{
  int id;
  string x;
}a[1001];
bool cmp(LETTER a, LETTER b){
  return a.id<b.id;
}
int main(){
  int n;
  for(;;){
    scanf("%d",&a[n++].id);
    if(getchar()=='\n')break;//第一行是否结束
  }
  for(int i=0;i<n;i++){
    cin>>a[i].x;
  }
  sort(a,a+n,cmp);
  for(int i=0;i<n;i++){
    cout<<a[i].x<<endl;
  }
  return 0;
}
View Code

 

 

P1398 奖学金

题意:多关键字排序,使用sort的第三个参数cmp函数

#include<iostream>
#include<algorithm>
#define sum(i) i.yw+i.sx+i.yy
using namespace std;
struct STU{
    int xh;
    int yw,sx,yy;
};
bool cmp(STU a,STU b){
    if(sum(a)!=sum(b))return sum(a)>sum(b);
    else if(a.yw!=b.yw)return a.yw>b.yw;
    else return a.xh<b.xh;
}
int main(){
  int n;
  scanf("%d",&n);
  struct STU stu[n];
  for(int i=0;i<n;i++)scanf("%d %d %d",&stu[i].yw,&stu[i].sx,&stu[i].yy),stu[i].xh=i+1;
  sort(stu,stu+n,cmp);
  for(int i=0;i<5;i++)printf("%d %d\n",stu[i].xh,sum(stu[i])) ;
  return 0;
}
View Code

 

P1431 守望者的逃离

题意:动态规划,综合考虑使用魔法和跑步的最大值

CCF书上的思路分析:

#include<iostream>
#include<cstring>
using namespace std;
#define MAXN 300010
int main(){
  int m,s,t,i;
  int f[MAXN];
  cin>>m>>s>>t;
  f[0]=0;
  for(int i=1;i<=t;i++){//计算只用闪烁法术时的每秒最大距离
    if(m>9)f[i]=f[i-1]+60,m-=10;
    else f[i]=f[i-1],m+=4;
  }
  for(int i=1;i<=t;i++){//计算加入跑步选择时的每秒最大距离
    if(f[i]<f[i-1]+17)f[i]=f[i-1]+17;
    if(f[i]>=s){//刚好离岛
      cout<<"Yes"<<endl<<i;return 0;
    }
  }
  cout<<"No"<<endl<<f[t];//不能离岛
  return 0;
}
View Code

 

 

P1484 ISBN号码

PS:10在ISBN中表示为'X'

#include<bits/stdc++.h>
using namespace std;
int main(){
  string s;
  cin>>s;
  for(int i=0;i<s.size();i++){
    if(s[i]!='-'&&s[i]!='X')s[i]-='0';
    if(s[i]=='X')s[i]=10;
  }
  int x=(s[0]*1+s[2]*2+s[3]*3+s[4]*4+s[6]*5+s[7]*6+s[8]*7+s[9]*8+s[10]*9)%11;
  if(x==s[12])printf("Right\n");
  else{
    for(int i=0;i<s.size();i++){
      if(s[i]!='-')s[i]+='0';
    }
    for(int i=0;i<s.size()-1;i++)cout<<s[i];
    if(x!=10)cout<<x;
    else cout<<'X';
  }
}
View Code

 

P1496 火柴棒等式

题意:枚举A,B,C

CCF书上的思路分析:

#include<bits/stdc++.h>
using namespace std;
int main(){
  int a[10]={6,2,5,5,4,5,6,3,7,6},ans=0,temp=0,k;
  //a[i]表示数字i需要几根火柴棒
  int num[2016];
  int n;
  cin>>n;
  num[0]=6;
  for(int i=1;i<2000;i++){//预处理0-2000数对应的火柴数目
    k=i;
    while(k){
      temp+=a[k%10];
      k/=10;
    }
    num[i]=temp;//num[i]存放i的火柴个数
    temp=0;
  }
  for(int i=0;i<999;i++){
    for(int j=0;j<999;j++){//枚举A,B
      if(num[i]+num[j]>=n)continue;//超过
      else if(num[i+j]+num[i]+num[j]+4==n){//C为A+B即i+j
        ans++;
      }
    }
  }
  cout<<ans;
  return 0;
}
      
View Code

 

 

P1597 2的幂次方

思路:设递归函数dfs(x)用于输出x的幂次方

最容易的思路:0不输出,1输出为2(0),2输出2,剩下的递归执行。

每一次递归:例如7,拆分为4+3,先拆出最大的是2的次方的数出来,输出4,再把3分拆输出。

对于3,拆分为2+1。

//flag用于标记输出时前面是否需要加+号
#include<bits/stdc++.h>
using namespace std;
int n;
void dfs(int n,bool flag){
    if(n==0)return;
    if(flag)cout<<"+";
    if(n==2)cout<<"2";
    else if(n==1)cout<<"2(0)";
    //else if(n==0)return;
    else{
        int i;
        for(i=0;;i++){
            if(pow(2,i)>n)break;
        }
        i--;
        if(i==1)cout<<"2";
        else{
            cout<<"2(";
            dfs(i,0);
            cout<<")";
        }
        dfs(n-pow(2,i),1);
    }
}
int main(){
    cin>>n;
    dfs(n,0);
}
View Code

 

 

P1762 统计变量 

题意:找':'和','的个数即可,因为变量名是用它们分隔的。另外变量都定义在主程序之前,所以读取到第一个begin就需要结束了

#include<bits/stdc++.h>
using namespace std;
int main(){
  string s;
  int ans=0;
  while(cin>>s){
    if(s=="begin")break;
    for(int i=0;i<s.size();i++){
      if(s[i]==','||s[i]==':')ans++;
    }
  }
  cout<<ans;
}
View Code

 

P1772 巧妙填数

题意:将1,2,...,9共9个数分成三组,分别组成三个三位数,且使这三个三位数构成1:2:3的比例。

可以使用枚举的方法,只枚举a,然后把b,c计算出来,数位分离并且用数组记录下每个数字出现的个数。

#include<bits/stdc++.h>
using namespace std;
int main(){
  int a,b,c;
  for(int a=123;a<=333;a++){
        b=2*a;c=3*a;
        char s[10];
        sprintf(s,"%d%d%d",a,b,c);
        int flag[10];
        memset(flag,0,sizeof(flag));
        for(int i=0;i<9;i++)
          flag[s[i]-'0']++;
        for(int i=1;i<=9;i++){
            if(flag[i]!=1)goto next;
        }
        cout<<a<<' '<<b<<' '<<c<<endl;
        next:
            continue;
  }
}
      
View Code

 

 

 

P1784 数字统计

题意:在数字中数位分离找到指定的数字,可以直接sprintf数字转字符串,然后逐位判断

#include<bits/stdc++.h>
using namespace std;
int main(){
    int a,b,ans=0;
    char s[15];
    cin>>a>>b;
    for(int i=a;i<=b;i++){
        sprintf(s,"%d",i);
        for(int j=0;j<15;j++){
            if(s[j]=='2')ans++;
        }
    }
    cout<<ans<<endl;
    return 0;
}
View Code

手动数位分离

 

#include<bits/stdc++.h>
using namespace std;
int main(){
  int a,b,ans=0;
  char s[15];
  cin>>a>>b;
  for(int i=a;i<=b;i++){
    int temp=i;
    while(i>0){
      if(i%10==2)ans++;
      i/=10;
    }
    i=temp;
  }
  cout<<ans<<endl;
  return 0;
}
View Code

 

 

 

P1788 第k大

 题意:排序,找第k大的数,注意sort默认从小到大排序

#include<bits/stdc++.h>
using namespace std;
int main(){
  int n,k;
  cin>>n>>k;
  int a[n];
  for(int i=0;i<n;i++)cin>>a[i];
  sort(a,a+n);
  cout<<a[n-k];
}
View Code

二分法(来源NOIP2008普及组初赛的完善程序,有微调)

每一次把数组分为两部分,大于基准数和小于基准数

#include <iostream>
using namespace std;
int a[1000001],n,ans =-1;
void swap(int &a,int &b){
  int c;
  c=a;a=b;b=c;
}
int FindKth(int left, int right, int n){
  int tmp,value,i,j;
  if (left==right) return left;//区间只有一个数 
  tmp=rand()%(right-left)+left;//随机的中间数 
  swap(a[tmp],a[left]);
  value=a[left];//基准数 
  i=left;
  j=right;
  while (i<j){//对区间进行二分 
    while (i<j && a[j]<value) j--;
    if (i<j) {a[i]=a[j]; i++;} else break;
    while (i<j && a[i]>value) i++;
    if (i<j) {a[j]=a[i]; j--;} else break;
  }
  a[i]=value;//i为排序下来的中间数 
  if (i<n) return FindKth(i+1,right,n);//在左区间继续 
  if (i>n) return FindKth(left,i-1,n); //在右区间继续 
  return i;//i=n的情况 
}
int main(){
  int n,k;
  cin>>n>>k;
  for(int i=1;i<=n;i++)
    cin>>a[i];
  ans=FindKth(1,n,k);
  cout<<a[ans];
  return 0;
}
View Code

 

 

P1812 多项式输出

题意:按照题目指定的格式输出多项式

注意事项:

如果是n次,不需要输出+号。

从n-1次到2次,直接输出。注意,输出负数的时候,会自动带上负号,输出正数不会自动输出+号,需要自己写。例如,

cout<<a;//输出-1

cout<<"-"<<a;//输出--1

1次需要注意不用输出x后面的^和次数

0次只需要输出系数。

如果系数为1或-1,不输出1。

#include<bits/stdc++.h>
using namespace std;
int main(){
  int n,a;
  cin>>n;
  int i;
  for(i=n;i>=0;i--){
    cin>>a;
    if(a==0)continue;
    if(i==n){
      if(a!=1 && a!=-1)cout<<a<<"x^"<<i;
      if(a==1)cout<<"x^"<<i;
      if(a==-1)cout<<"-x^"<<i;
      continue;
    }
    else if(i==0){
      if(a>0){
        cout<<"+"<<a;continue;
      }
      if(a<0){
        cout<<a;continue;
      }
    }
    else if(i==1){
      if(a>0){
        if(a!=1)cout<<"+"<<a<<"x";
        else cout<<"+x";
      }
      if(a<0){
        if(a!=-1)cout<<a<<"x";
        else cout<<"-x";
      }
}
    else {
      if(a>0){
        if(a!=1)cout<<"+"<<a<<"x^"<<i;
        else cout<<"+x^"<<i;
      }
      if(a<0){
        if(a!=-1)cout<<a<<"x^"<<i;
        else cout<<"-x^"<<i;
      }
      continue;
    }
  }
  return 0;
}
View Code

 

P1813 分数线划定

题意:多关键字排序

#include<bits/stdc++.h>
using namespace std;
int N,M,n,m;
struct Node{
  int k,s;
}P[10001];
bool Compare(Node a,Node b){
  if(a.s==b.s)return a.k<b.k;
  return a.s>b.s;
}
int main(){
  scanf("%d %d",&N,&M);
  for(int i=1;i<=N;i++){
    scanf("%d %d",&P[i].k,&P[i].s);
  }
  sort(P+1,P+N+1,Compare);
  m=floor(M*1.5);
  for(int i=1;;i++){
    if(P[i].s>=P[m].s)++n;
    else break;
  }
  printf("%d %d\n",P[m].s,n);
  for(int i=1;;i++){
    if(P[i].s>=P[m].s)printf("%d %d\n",P[i].k,P[i].s);
    else break;
  }
  return 0;
}
View Code

 

P1848 计数问题

题意:同P1784

sprintf数位分离法

#include<iostream>
using namespace std;
int main(){
  int n,x,ans=0;
  cin>>n>>x;
  char s[8];
  for(int i=1;i<=n;i++){
    sprintf(s,"%8d",i);
    for(int j=0;j<8;j++){
      if(s[j]-'0'==x)ans++;
    }
  }
  cout<<ans;
}
View Code

手动数位分离法:

使用一个while循环,不断取除以10的余数,除一位后就去掉这位,从下一位继续

#include<iostream>
using namespace std;
int main(){
  int n,x,ans=0;
  cin>>n>>x;
  for(int i=1;i<=n;i++){
    int temp=i;
    while(i>0){
      if(i%10==x)ans++;
      i/=10;
    }
    i=temp;
  }
  cout<<ans;
}
View Code

 CCF程序设计基础篇书上的题解,也是手动数位分离法

 

#include<iostream>
using namespace std;
int main(){
  int ans=0,i,n,x,tmp;
  cin>>n>>x;
  for(int i=1;i<=n;i++){//重复对1-n中的每一个数进行计数
    tmp=i;
    while(tmp>=1){//对于数字逐位判断,计数
      if(tmp%10==x)ans++;//如果当前的末尾数字是x,则计数器加一
      tmp/=10;//为下一次取得末尾数字做准备
    }
  }
  cout<<ans;
  return 0;
}
View Code

 

 

P1911 珠心算

题意:找出集合中一个数等于其他两数之和,三层循环枚举

#include<bits/stdc++.h>
using namespace std;
int n,*a,s,t,u,ans;
bool f[10001]; 
int main(){
  cin>>n;
  a=(int*)malloc(n*sizeof(int));
  for(int i=0;i<n;i++)
    cin>>a[i];
  for(int i=0;i<n;i++){
    s=a[i];
    for(int j=0;j<n;j++){
      if(j==i)continue;//不同的数 
      t=a[j];
      for(int k=0;k<n;k++){
        if(k==i||k==j)continue;
        u=a[k];
        if(s+t==u && !f[u]){
          ans++;f[u]=1;//已经算过的
      }
    }
  }
  }
 cout<<ans<<endl;
  return 0;
}
View Code

 

P1912 比例简化

题意:将 A 比 B 化简为 A’比 B’,要求在 A’和 B’均不大于 L 且 A’和 B’互质(两个整数的最大公约数是 1)的前提下,A’/B’ ≥ A/B 且 A’/B’ - A/B 的值尽可能小。

#include<bits/stdc++.h>
using namespace std;
int gcd(int x,int y){
    if(y==0)return x;
    return gcd(y,x%y);
}
int main(){
  int a,b,l,a1=100000,b1=1;
  cin>>a>>b>>l;
  for(int i=1;i<=l;i++){
    for(int j=1;j<=l;j++){
      if(gcd(i,j)==1 && i*b>=j*a)
        if(i*b1<=j*a1)a1=i,b1=j;//分数的大小比较可以用十字相乘法
    }
  }
  cout<<a1<<' '<<b1;
  return 0;
}
View Code

 

P1913 螺旋矩阵

题意:模拟,按照顺序构造符合条件的螺旋矩阵

#include<stdio.h>
int n,x,y;
int min(int n1,int n2){
  if(n1<n2)return n1;
  else return n2;
}
int solve(){
  scanf("%d%d%d",&n,&x,&y);
  int layer=min(x,n+1-x);//确定层,n方阵,如n=7,x=5就是第二层
  layer=min(layer,min(y,n+1-y));
  int num=0,side=n-1;
  for(int i=1;i<layer;i++,side-=2){
    int layer_total;
    if(side==0)layer_total=1;
    else layer_total=side*4;//当前层一圈总个数
    num+=layer_total;
  }
  num++;
  int nx=layer,ny=layer;
  if(nx==x && ny==y)return num;
  for(int i=1;i<=side;i++){
    ny++;num++;//y列向右递增
    if(nx==x && ny==y)return num;
  }
  for(int i=1;i<=side;i++){
    nx++;num++;//x行向下递增
    if(nx==x && ny==y)return num;
  }
  for(int i=1;i<=side;i++){
    ny--;num++;//y列向左递增
    if(nx==x && ny==y)return num;
  }
  for(int i=0;i<side;i++){
    nx--;num++;//x行向上递增
    if(nx==x && ny==y)return num;
  }
  return -1;
}
int main(){
  int res=solve();printf("%d",res);
  return 0;
}
View Code

 

posted @ 2021-07-11 14:57  计算机知识杂谈  阅读(132)  评论(0编辑  收藏  举报