萌新联赛第(三)场

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】

相关推荐
liuzhangfeiabc2 小时前
[luogu12541] [APIO2025] Hack! - 交互 - 构造 - 数论 - BSGS
c++·算法·题解
学习使我变快乐2 小时前
C++:迭代器
开发语言·c++·windows
好想有猫猫2 小时前
【Redis】List 列表
数据库·c++·redis·分布式·缓存·list
平和男人杨争争4 小时前
山东大学计算机图形学期末复习15——CG15
人工智能·算法·计算机视觉·图形渲染
superior tigre4 小时前
C++学习:六个月从基础到就业——C++11/14:其他语言特性
c++·学习
天堂的恶魔9465 小时前
C++ - 仿 RabbitMQ 实现消息队列(2)(Protobuf 和 Muduo 初识)
c++·rabbitmq·ruby
休息一下接着来5 小时前
进程间通信(IPC)常用方式对比
linux·c++·进程间通讯
虾球xz5 小时前
游戏引擎学习第288天:继续完成Brains
c++·学习·游戏引擎
John_ToDebug5 小时前
Chromium 浏览器核心生命周期剖析:从 BrowserProcess 全局管理到 Browser 窗口实例
c++·chrome·性能优化
爱coding的橙子5 小时前
每日算法刷题Day11 5.20:leetcode不定长滑动窗口求最长/最大6道题,结束不定长滑动窗口求最长/最大,用时1h20min
算法·leetcode·职场和发展