winter 2024 day2

2023 中国大学生程序设计竞赛(CCPC)新疆赛区(重现赛

H数学

思路:有四平方和定理知道,任意正整数可表示为不超过四个整数的平方和

 并且n的范围为1e5,可以枚举出f(x)值为1、2、3、4的平方数组合情况。

也可以dp,f[i]=min(f[i],f[i-k*k]+1)

#include<bits/stdc++.h>
using namespace std;
#define int long long
//#define int __int128
#define double long double
typedef pair<int,int>PII;
typedef pair<string,int>PSI;
typedef pair<string,string>PSS;
const int N=3e5+5,INF=0x3f3f3f3f,Mod=1e9+7,mod=998244353;
const int MAXN=1e8+5;
const double eps=1e-6;


void solve() {
    int n;
    cin >> n;
    vector<int> f(n + 5, INF);
    vector<int> g;
    for (int i = 1; i * i <= n; ++i)g.push_back(i * i);
    f[0] = 0;
    for (int i = 1; i <= n; ++i) {
        for (int j = 0; j < g.size(); ++j) {
            if (g[j] > i)break;
            f[i] = min(f[i], f[i - g[j]] + 1);
        }
    }
    int ans = 1;
    for (int i = 1; i <= n; ++i)ans = (ans * f[i]) % mod;
    cout << ans;
}

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

 

D在此同步

思路:想的相邻的两个数交换会让逆序对数改变一,那就找一个加一和一个减一的相邻对交换一下总逆序对数就不变,暴力找一遍就可以。后面抗体姐被自己整笑了,直接找个山峰或山谷(三个连续数),然后将其内部后移一位就行

...

 

BSang的奇妙冒险

思路:首先分析式子,因为是要从1到n,对于|i-j|整体看来就是n-1。对于|ai-aj|,这里题目求的是代价最小的方案数,那么很明显最小的代价是|a1-an|,直接一步到终点,因为在1到n的路途中出现山峰或山谷的话,只会增加代价,那么选择的ai应该是不递减或不递增的。问题就转化为求最长不下降子序列(a1<an)或最长不上升子序列(a1>an),翻转下后的求法就是一样的。

 

F任务安排

思路:首先没有序列a的话就是求拓扑排序,首先判断是否为拓扑序;其次,给每个点设置权值,求拓扑序时用优先队列即可。对于设置权值,把序列a按顺序递增给权值1、2、3....。还有一个问题,当序列a中的点,指向不在序列a中的点时可能会导致最后的拓扑序中序列a中的点不连续,那么在序列a中的点里,若其后缀为非序列a中的点,将其后缀的权值设为无穷大,这样就可以在当前拓扑序里已有序列a的点的情况下,优先选择权值小的,也就是在序列a中的点

 

E外接圆

思路:首先要知道的是三点确定圆,那么就来个n3的枚举确定圆,再来来个n的枚举判断剩余点是否在圆上,噢记得先去重了,记录每个点出现次数。还要注意所有点共线的情况,那就是两点确定一圆,找到出现次数最多的两点即可

...抄板子

 G动态二叉树

思路:考虑离线求,由于整个二叉树不会改变,可以先求出最终的二叉树。操作二问的是x右边的节点,其实就是bfs序中x的后继,可以先求出最终二叉树的所有节点对应的bfs序号。再重新依次操作,若是操作一,先找到该节点所在层数,在该层插入该节点的bfs序号,由于bfs序号是有序的,这里可以用set存储,二分找到该节点的后继。

#include<bits/stdc++.h>
using namespace std;
#define int long long
//#define int __int128
#define double long double
typedef pair<int,int>PII;
typedef pair<string,int>PSI;
typedef pair<string,string>PSS;
const int N=5e5+5,INF=0x3f3f3f3f,Mod=1e9+7,mod=998244353;
const int MAXN=1e8+5;
const double eps=1e-12;
struct E{
    int op,p;
    char c;
};

void solve() {
    int n;
    cin>>n;
    vector<E>ve(n);
    vector<int>L(n+5),R(n+5),id(n+5),to(n+5);
    for(int i=0,idx=2;i<n;++i){
        cin>>ve[i].op;
        if(ve[i].op==1){
            cin>>ve[i].p>>ve[i].c;
            if(ve[i].c=='L')L[ve[i].p]=idx++;
            else R[ve[i].p]=idx++;
        }else{
            cin>>ve[i].p;
        }
    }
    queue<int>q;
    int idx=1;
    q.push(1);
    while(q.size()){
        int c=q.size();
        while(c--){
            int t=q.front();q.pop();
            to[idx]=t,id[t]=idx++;
            if(L[t])q.push(L[t]);
            if(R[t])q.push(R[t]);
        }
    }
    set<int>g[N+5];
    vector<int>d(n+5);
    g[0].insert(1);
    for(int i=0,now=2;i<n;++i){
        if(ve[i].op==1){
            int fd=d[ve[i].p];
            d[now]=fd+1;
            g[d[now]].insert(id[now++]);
        }else{
            auto t=g[d[ve[i].p]].lower_bound(id[ve[i].p]+1);
            int ans=-1;
            if(t!=g[d[ve[i].p]].end())ans=to[*t];
            cout<<ans<<'\n';
        }
    }
}

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

 

A动态树

思路:对于一棵树的直径s到t,只有当新增的边连接s或t时树的直径才会增大。如果添加了一条边后,出现出现了多条直径,那么出现的新直径的端点一定有一边还是s或t,那么只需要求下新增的点分别到s和t的距离即可知道。这里求直径用倍增LCA,可动态加叶子。

#include<bits/stdc++.h>
using namespace std;
#define int long long
//#define int __int128
#define double long double
typedef pair<int,int>PII;
typedef pair<string,int>PSI;
typedef pair<string,string>PSS;
const int N=2e5+5,M=18,INF=0x3f3f3f3f,Mod=1e9+7,mod=998244353;
const int MAXN=1e8+5;
const double eps=1e-12;

vector<int>g[N];
vector<int>dep(N);
vector<vector<int>>f(N,vector<int>(M+2));

void dfs(int u){
    for(int i=0;i<g[u].size(); ++i){
        int to=g[u][i];
        if(dep[to])continue;
        f[to][0]=u;
        dep[to]= dep[u] + 1;
        for(int j=1;j<=M;++j)f[to][j]=f[f[to][j-1]][j-1];
        dfs(to);
    }
}

int lca(int a,int b){
    if(dep[a]<dep[b])swap(a,b);
    for(int i=M;i>=0;--i)
        if(dep[f[a][i]]>=dep[b])a=f[a][i];
    if(a==b){
        return a;
    }
    for(int i=M;i>=0;--i){
        if(f[a][i]!=f[b][i]){
            a=f[a][i],b=f[b][i];
        }
    }
    return f[a][0];
}

void init(){
    dep[1]=1;
    dfs(1);
}

void solve() {
    int n,ma=0;
    cin>>n;
    init();
    int s=1,t=1,ans=-1;
    bool ok=true;
    for(int i=1;i<=n;++i){
        int x;
        cin>>x;
        if(!ok)continue;
        g[x].push_back(i+1),g[i+1].push_back(x);
        dfs(x);
        if(x==s){
            s=i+1,ma++;
        }else if(x==t){
            t=i+1,ma++;
        }else{
            int a=lca(i+1,s),b=lca(i+1,t);
            int ad=dep[i+1]+dep[s]-2*dep[a],bd=dep[i+1]+dep[t]-2*dep[b];
            if(ad==ma||bd==ma){
                ok=false;
                ans=i;
            }
        }
    }
    cout<<ans;
}

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

 

J子段和二象性

思路:搞成图来想,(但是还是不懂T^T),,那就要有点和关系,看原数组的前缀和数组,在满足条件的情况下,Si<Si+n,Si<Si-m(因为每n个数和为正,每m个数和为负),那么点就看成是Si,根据大小关系,Si指向Si+n,Si指向Si-m,找到非环的最长的链。

对于n=2、m=3,可以这样构造,从0开始,S0→S2→S4→S1→S3→S0,直到构成了环为止,这时候的最大长度为最大编号减一(不懂)

打表可以发现答案为n+m-gcd(n,m)-1

证明

 

posted @ 2024-01-25 20:43  bible_w  阅读(13)  评论(0编辑  收藏  举报