C题
这道题用暴力去写想都不要想,一定超时,于是我们需要优化,下面是思路过程:

如图,本题只需找到x的因数个数和(n-x)的因数个数,这两个相乘,得到的就是对于这个x来说组合的个数,且x的取值为1~n,由题n取不到。
代码:
cpp
#include<bits/stdc++.h>
using namespace std;
const int N=100005;
int num[N];//记录各个数字的因数个数
int main(){
int n,ans=0;//ans方案种数
cin>>n;
for(int i=1;i<n;i++){ //遍历1~n的数字
for(int j=1;j*j<=i;j++){ //求其因数个数
if(i%j==0){
num[i]++; //由题知,A,B,C,D这四个数是有序的,因此每次判断都会有两个因数
if(j*j!=i)num[i]++; //所以下面要+1,除非这两个数相等只需加1次
}
}
}
for(int i=1;i<n;i++){ //遍历和为n,加数所有可能的取值
ans+=num[i]*num[n-i];//因数个数相乘即得对应的组合数
}
cout<<ans;
return 0;
}
H题
本题乍一看很简单,就是求出每一个子集的和即可,但如何不重不漏的求出每一个集合我是真的不会,好的,换思路,利用贪心思想,将原集合排序,从第一个最小的数开始,逐步扩展可以表示的子集和范围,同时找到第一个无法表示的整数。因为已经排好序了,所以对于元素个数相同的集合来说,总是最前面的最小,而且代码的判断条件是a[i]>sum+1,那么sum + 1
就无法被表示,因为当前的子集和范围无法覆盖到 sum + 1,直接输出
sum+1,相反,如果a[i]<=sum+1,说明当前子集很可能已经覆盖了sum+1,让a[i]与sum+1比较是否可以覆盖,同时也可以避免出现不同子集因元素个数的不同造成对应的子集和大小不同问题,因为如果a[i]>sum+1说明a[i]一定很大,那么毫无疑问元素多的集合的和一定比元素少的集合的和大,这样我们就可以实现由小到大逐步扩展子集和范围。
代码如下:
cpp
#include<bits/stdc++.h>
using namespace std;
const int N=100005;
long long a[N];
int main(){
int n;
cin>>n;
for(int i=0;i<n;i++){
cin>>a[i];
}
sort(a,a+n); //由小到大逐步排序
long long sum=a[0]; //记录当前子集和的最大值
for(int i=1;i<n;i++){
if(sum+1<a[i]){ //比较当前子集和+1是否可以被下一个集合元素表示出来
cout<<sum+1; //如果可以,输出,结束
return 0;
}
else sum+=a[i]; //否则,继续累加
}
cout<<sum+1; //如果一直没在上面结束的话,此时sum代表全集的和,+1即是答案
return 0;
}
D题
对于这道题需要知道一个知识点:
x+y=(x&y)+(x|y)
根据位运算性质,x&y和x|y满足:(x&y)&(x|y)=x&y
本题思路:
由于x&y=a;说明x>=a,y>=a,则x+y>=2a,即s>=2a------第一个判断条件
又由x&y=a,x+y=s和上述知识点知x|y=s-a。因此,a&(s-a)=a------第二个判断条件
如果第一个不满足的话,直接输出NO,程序结束,否则,继续判断第二个条件。
这样做的好处是不用一一枚举出来x,y的值
cpp
#include<bits/stdc++.h>
using namespace std;
int main(){
int t;
long long a,s;
cin>>t;
while(t--){
cin>>a>>s;
if(s<2*a){ //首先判断
cout<<"No"<<endl;
}
else{ //继续深入判断
if((a&(s-a))==a) cout<<"Yes"<<endl;
else cout<<"No"<<endl;
}
}
return 0;
}
比赛地址:https://www.nowcoder.com/acm/contest/110544。【邀请码:666666】