本文介绍: C:贪心,肯定是积个大的秒后面数量多的,然后操作最后一个数的时候,尽量别浪费当前计数器的数,要分奇偶性和1,比如8,前面已经有2了,那么再操作2次,计数器变成4,操作计数器秒掉,如果当前是8,前面计数器是3,为了不浪费计数器最后一次肯定是直接消灭而不是使用计数器,所以先-1变成偶数(如果不这样做,最后计数器会多1,次数可能会增加),再换成偶数操作即可,特判1。如果当前a[i]可以整除2^x,然后加上2^(x-1),那么下次这个数就不能整除2^x假设 5是新增节点我们怎么操作

A:

这种操作题,每次先想这个操作什么性质

对于2^0来说可以操作 第1位

 对于2^1来说可以操作 第1-2位

对于2^2来说可以操作 第1-4位 (第3位无法单独修改

对于2^3来说可以操作 第1-8位(第5 6 7位无法单独修改

可以观察到我们要求无法修改的数要按顺序才能满足

#include<bits/stdc++.h>
using namespace std;
const int N = 1e6+10,mod= 998244353;
#define int long long
typedef long long LL;
typedef pair<int, int> PII;
const long long inf=1e17;
int n,m,k;
vector<int&gt; g[N];
int a[N];
void solve()
{
    cin&gt;>n;
    for(int i=1;i<=n;i++){
        cin>>a[i];
    }
    for(int i=2;i<=5;i++)
    {
        bool ok=true;
       // cout<<pow(2,i-1)<<" "<<pow(2,i)<<"n";
        for(int j=pow(2,i-1)+1;j<pow(2,i);j++){
            if(j+1>n||j>n) break;
            if(a[j]<=a[j+1]) continue;
            ok=false;
        }    
        if(!ok){
            cout<<"NOn";return ;
        }
    }
    cout<<"YESn";
}

signed main()
{
    cin.tie(0);cout.tie(0);ios::sync_with_stdio(0);
    int t=1;
    cin>>t;
    while(t--) solve();
}

B:

操作题还是想操作当前x每个数组a每个数有啥影响

如果当前a[i]可以整除2^x,然后加上2^(x-1),那么下次这个数就不能整除2^x

那么他就会变成2^(x-1)的倍数了,他的因子包含2^x,所以不会再操作

然后x最多30个数,去重后操作即可

#include<bits/stdc++.h>
using namespace std;
const int N = 1e6+10,mod=1e9+7;
#define int long long
typedef long long LL;
typedef pair<int, int> PII;
const long long inf=1e17;
int n,m,k;
vector<int> g[N];
int a[N];

class BitTree {
    public:
	vector<int> tree;
	int n;
    BitTree(int _n) : n(_n) {
	    tree.resize(n+1);
	    fill(tree.begin(),tree.end(),0);
	}
	inline int lowbit(int x) { return x&amp;-x; }
	inline void Modify(int x,int v) {
		for(;x<=n;x+=lowbit(x)) tree[x]+=v;
	}
	inline int q(int x) {
		int ret=0;
		if(x<=0) return 0;
		for(;x;x-=lowbit(x)) ret+=tree[x];
		return ret;
	}
	inline int Query(int l,int r) {
		return q(r)-q(l-1);
	}
};
int l[N],r[N];
void solve()
{
    vector<int> b;
    map<int,int> mp;
    cin>>n>>m;
    for(int i=1;i<=n;i++) cin>>a[i];
    for(int i=1;i<=m;i++){
        int x;cin>>x;
        if(mp.count(x)) continue;
        mp[x]++;
        b.push_back(x);
    }
    for(int i=1;i<=n;i++){
        int x=a[i];
        for(auto y:b){
            if(x%(1<<y)==0){
                x+=(1<<y-1);
            }
        }
        cout<<x<<" n"[i==n];
    }
    
}

signed main()
{
    cin.tie(0);cout.tie(0);ios::sync_with_stdio(0);
    int t=1;
    cin>>t;
    while(t--) solve();
}

C:贪心,肯定是积个大的秒后面数量多的,然后当操作最后一个数时候,尽量别浪费当前计数器的数,要分奇偶性和1,比如8,前面已经有2了,那么再操作2次,计数器变成4,操作个计数器秒掉,如果当前是8,前面计数器是3,为了不浪费计数器,最后一次肯定是直接消灭而不是使用计数器,所以先-1变成偶数(如果不这样做,最后计数器会多1,次数可能会增加),再换成偶数操作即可,特判1

#include<bits/stdc++.h>
using namespace std;
const int N = 1e6+10,mod= 998244353;
#define int long long
typedef long long LL;
typedef pair<int, int> PII;
const long long inf=1e17;
int n,m,k;
vector<int> g[N];
int a[N];
void solve()
{
    cin>>n;
    for(int i=1;i<=n;i++) cin>>a[i];
    
    sort(a+1,a+1+n);
    int res=0;
    int l=1,r=n;
    int sum=0;
    while(l<=r)
    {
        if(a[l]==0)
        {
            l++;
            continue;
        }
        if(l==r)
        {
            if(a[l]==1){
                res++;break;
            }
            if((a[l]-sum)%2==1)
            {
                res+=(a[l]-sum)/2+1;
                res++;
            }
            else{
                res+=(a[l]-sum)/2+1;
            }
            cout<<res<<"n";
            return ;
        }
        if(a[r]<=sum+a[l])
        {
            int x=a[r]-sum;
            res+=x+1;
            a[l]-=x;
            r--;
            sum=0;
        }    
        else
        {
            sum+=a[l];
            res+=a[l];
            l++;    
        }
    }
    cout<<res<<"n";
}

signed main()
{
    cin.tie(0);cout.tie(0);ios::sync_with_stdio(0);
    int t=1;
    cin>>t;
    while(t--) solve();
}

D:套路题,求两个区间即可

然后我画图应该能看懂

g的函数的值最多有两个不同,因为3 4 5…的增长比2增长快,所以最多两个

可以观察到g的值怎么求

比如8 到 15

用3的倍数求

16到31用 4倍数求他们的g的值,然后乘上求区间个数即可

#include<bits/stdc++.h>
using namespace std;
const int N = 1e6+10,mod=1e9+7;
#define int long long
typedef long long LL;
typedef pair<int, int> PII;
const long long inf=1e17;
int n,m,k;
vector<int> g[N];
int a[N];
void solve()
{
    auto get=[&amp;](int x){
        int res=0;
        int c=2;
        for(int i=4;i<=x;i*=2)
        {
            int cnt=0;
            int r=min(x,i*2-1);
            __int128 now=1;
            while(now<=i) cnt++,now*=c;
            c++;
            if(now>r){
                res+=(r-i+1)%mod*(cnt-1)%mod;
            }
            else{
                int t=r-now+1;
                t%=mod;
                res+=t*cnt%mod+(now-i)%mod*(cnt-1);
            }
        }
        return res%mod;
    };
    int l,r;
    cin>>l>>r;
    cout<<((get(r)-get(l-1))%mod+mod)%mod<<"n";
}

signed main()
{
    cin.tie(0);cout.tie(0);ios::sync_with_stdio(0);
    int t=1;
    cin>>t;
    while(t--) solve();
}

F:首先肯定要把数dfs一遍弄出来把,不然鬼知道子树哪些

然后我们把树画出来

假设 5是新增节点,我们怎么操作,

直接前面5节点的数全部减成0即可

然后就是差分咯,因为子树增加是增加整个区间

#include<bits/stdc++.h>
using namespace std;
const int N = 1e6+10,mod=1e9+7;
#define int long long
typedef long long LL;
typedef pair<int, int> PII;
const long long inf=1e17;
int n,m,k;
vector<int> g[N];
int a[N];

class BitTree {
    public:
	vector<int> tree;
	int n;
    BitTree(int _n) : n(_n) {
	    tree.resize(n+1);
	    fill(tree.begin(),tree.end(),0);
	}
	inline int lowbit(int x) { return x&amp;-x; }
	inline void Modify(int x,int v) {
		for(;x<=n;x+=lowbit(x)) tree[x]+=v;
	}
	inline int q(int x) {
		int ret=0;
		if(x<=0) return 0;
		for(;x;x-=lowbit(x)) ret+=tree[x];
		return ret;
	}
	inline int Query(int l,int r) {
		return q(r)-q(l-1);
	}
};
int l[N],r[N];
void solve()
{
    cin>>n;
    for(int i=1;i<=n*2+10;i++)
    g[i].clear();

    vector<array<int,3>>query;
    int now=1;
    for(int i=0;i<n;i++){
        int op;cin>>op;
        if(op==1){
            now++;
            int v;cin>>v;
            query.push_back({op,v,0});
            g[v].push_back(now);
        }
        else{
            int v,x;cin>>v>>x;
            query.push_back({op,v,x});
        }
    }
    int dfn=0;
    function<void(int)> dfs=[&amp;](int u){
        l[u]=++dfn;
        for(auto v:g[u]){
            dfs(v);
        }
        r[u]=++dfn;
    };
    dfs(1);
    BitTree tr(n*2+10);
    now=1;
    for(auto [op,v,x]:query){
        if(op==1){//v增加一个子节点
            now++;
            int t=tr.q(l[now]);
            tr.Modify(l[now],-t);
            tr.Modify(r[now],t);
        }
        else
        {
            tr.Modify(l[v],x);
            tr.Modify(r[v],-x);
        }
    }
    for(int i=1;i<=now;i++){
        cout<<tr.q(l[i])<<" n"[i==now];
    }
}

signed main()
{
    cin.tie(0);cout.tie(0);ios::sync_with_stdio(0);
    int t=1;
    cin>>t;
    while(t--) solve();
}

原文地址:https://blog.csdn.net/qq_61657632/article/details/134694806

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任

如若转载,请注明出处:http://www.7code.cn/show_43424.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱suwngjj01@126.com进行投诉反馈,一经查实,立即删除

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注