zoukankan      html  css  js  c++  java
  • Codeforces Round 264 (Div. 2)


    layout: post
    title: Codeforces Round 264 (Div. 2)
    author: "luowentaoaa"
    catalog: true
    tags:
    mathjax: true
    - codeforces
    - DP


    传送门

    A - Caisa and Sugar (签到)

    B - Caisa and Pylons (根据能量守恒定理hhh)

    C - Gargari and Bishops (模拟)

    题意

    给出一个棋盘你可以放两个皇后,皇后可以攻击和他一个对角线的对象。但是两个皇后不能攻击到同一个目标 求出皇后攻击目标后获得的最大值

    思路

    首先对于对角线正对角线的x+y都是固定的,反对角线的X-Y都是固定的

    然后我们就枚举对角线的值就行,

    通过观察发现不相交的对角线他们的x+y的奇偶不同。所以我们分别求出最大即可

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    const int maxn=2e3+50;
    
    ll a[maxn][maxn];
    ll maxL[maxn*2];
    ll maxR[maxn*2];
    int main(){
    
        int n;
        scanf("%d",&n);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                scanf("%d",&a[i][j]);
                maxL[i+j]+=a[i][j];
                maxR[i-j+n]+=a[i][j];
            }
        }
        ll even=0,odd=0,sx,sy,ex,ey;
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                if((i+j)&1){
                    if(maxL[i+j]+maxR[i-j+n]-a[i][j]>=odd){
                        odd=maxL[i+j]+maxR[i-j+n]-a[i][j];
                        sx=i;sy=j;
                    }
                }
                else{
                    if(maxL[i+j]+maxR[i-j+n]-a[i][j]>=even){
                        even=maxL[i+j]+maxR[i-j+n]-a[i][j];
                        ex=i;ey=j;
                    }
                }
            }
        }
        printf("%lld
    ",even+odd);
        printf("%lld %lld %lld %lld
    ",sx,sy,ex,ey);
        return 0;
    }
    

    D - Gargari and Permutations (DP)

    题意

    求出K个排列的LCS

    思路

    两种解法:

    DP:首先对于第一个排列他们之间的大小关系肯定是固定的,所以我们可以枚举第一个排列的大小关系,判断这个关系在其他k-1个排列中是否也符合 如果符合那么后者的答案就可以从前者身上转移而来 (dp[i]=max(dp[i],dp[j]+1))

    dfs: 前半部分相同,后面改成每一个符合的关系一起建图。对于每个点求出一个最长路。

    //dp
    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    const int maxn=2e3+50;
    int a[6][maxn];
    int id[6][maxn];
    int dp[maxn];
    int now[maxn];
    int k;
    bool ok(int u,int v){
        for(int i=2;i<=k;i++){
            if(id[i][u]<id[i][v])return false;
        }
        return true;
    }
    int main(){
        int n;
        cin>>n>>k;
        for(int j=1;j<=k;j++)
            for(int i=1;i<=n;i++){
                cin>>a[j][i];
                id[j][a[j][i]]=i;
            }
        for(int i=1;i<=n;i++){
            dp[i]=1;
        }
        for(int i=1;i<=n;i++){
            for(int j=1;j<i;j++){
                if(ok(a[1][i],a[1][j]))dp[a[1][i]]=max(dp[a[1][i]],dp[a[1][j]]+1);
            }
        }
        cout<<*max_element(dp+1,dp+n+1)<<endl;
        return 0;
    }
    
    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    const int maxn=2e3+50;
    #define bug cout<<"hello"<<endl;
    int a[6][maxn];
    int id[6][maxn];
    int dp[maxn];
    int now[maxn];
    int k;
    bool ok(int u,int v){
        for(int i=1;i<=k;i++){
            if(id[i][u]>id[i][v])return false;
        }
        return true;
    }
    vector<int>ve[maxn];
    int dfs(int u){
        if(dp[u]!=-1)return dp[u];
        int mx=0;
        for(int i=0;i<ve[u].size();i++){
            mx=max(mx,dfs(ve[u][i]));
        }
        return dp[u]=mx+1;
    }
    int main(){
        int n;
        cin>>n>>k;
        memset(dp,-1,sizeof(dp));
        for(int j=1;j<=k;j++)
            for(int i=1;i<=n;i++){
                cin>>a[j][i];
                id[j][a[j][i]]=i;
            }
    
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                if((i!=j)&&ok(i,j)){
                    ve[i].push_back(j);
                }
            }
        }
        int ans=0;
        for(int i=1;i<=n;i++){
            ans=max(ans,dfs(i));
        }
        cout<<ans<<endl;
        return 0;
    

    E - Caisa and Tree (暴力/素数筛选)

    题意

    给你一个树,让你求根(1)到一个点(v)之间路径上和v的值之间GCD不为1的且深度最大的点编号 有五十次修改一个点的操作

    思路

    看到50次修改就已经想到了暴力了。

    但是还是想用正常操作做。

    初始化每个值的素因子

    首先我们对于每个点存一个他的素因子,然后在每次修改的时候重新建图。从根结点开始开一个栈,让遍历到的点进去,判断这个点的素因子是否出现在刚才遍历的路径中。然后选一个最大的。后把当前这个点也入栈。遍历这个点的儿子们。然后回溯的时候把所有的素因子中包括他的弹开。

    //讲道理每个素因子开一个vector就可以了。可是一直过不去,其他人用map就过去了。。

    第二种暴力解法。对于每个操作离线存起来,然后在修改的时候再把之间的询问一次性遍历。

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    const int maxn=2e6+50;
    #define bug cout<<"hello"<<endl;
    int n,q;
    int a[maxn];
    int prime[maxn];
    bool check[maxn];
    int cnt;
    
    void init(int N){
        for(int i=2;i<=N;i++){
            if(!check[i]){
                prime[++cnt]=i;
            }
            for(int j=1;j<=cnt;j++){
                if(i*prime[j]>N)break;
                check[i*prime[j]]=true;
                if(i%prime[j]==0)break;
            }
        }
    }
    vector<int>ve[maxn];
    void gao(int id,int val){
        ve[id].clear();
        for(int i=1;prime[i]*prime[i]<=val&&i<=cnt;i++){
            if(val%prime[i]==0){
                ve[id].push_back(prime[i]);
                while(val%prime[i]==0)val/=prime[i];
            }
        }
        if(val!=1)ve[id].push_back(val);
    }
    int Laxt[maxn],Next[maxn],To[maxn],res;
    
    void add(int u,int v){
        Next[++res]=Laxt[u];Laxt[u]=res;To[res]=v;
    }
    
    int ans[maxn],dep[maxn];
    map<int,vector<int> >mp;
    
    void dfs(int u,int fa,int depp){
        ans[u]=-1,dep[u]=depp;
        for(int i=0;i<ve[u].size();i++){
            int v=ve[u][i];
            if(mp[v].size()==0){
                mp[v].push_back(u);
            }
            else{
                if(ans[u]==-1||dep[ans[u]]<dep[mp[v][mp[v].size()-1]])
                    ans[u]=mp[v][mp[v].size()-1];
                mp[v].push_back(u);
            }
        }
        for(int i=Laxt[u];i;i=Next[i]){
            int v=To[i];
            if(v==fa)continue;
            dfs(v,u,depp+1);
        }
        for(int i=0;i<ve[u].size();i++){
            int v=ve[u][i];
            mp[v].erase(mp[v].end()-1);
        }
    }
    
    int main(){
        init(2e6+10);
        scanf("%d%d",&n,&q);
        for(int i=1;i<=n;i++)scanf("%d",&a[i]),gao(i,a[i]);
    
        for(int i=1;i<n;i++){
            int u,v;scanf("%d%d",&u,&v);
            add(u,v);add(v,u);
        }
    
        dfs(1,-1,1);
        while(q--){
            int op,x,y;
            scanf("%d",&op);
            if(op==1){
                scanf("%d",&x);
                printf("%d
    ",ans[x]);
            }
            else{
                scanf("%d%d",&x,&y);
                a[x]=y;
                gao(x,a[x]);
                dfs(1,-1,1);
            }
        }
        return 0;
    }
    
    #include<bits/stdc++.h>
    using namespace std;
    
    const int maxn=1e5+50;
    int Laxt[maxn],Next[maxn<<1],To[maxn<<1],cnt;
    
    void add(int u,int v){
        Next[++cnt]=Laxt[u];Laxt[u]=cnt;To[cnt]=v;
    }
    
    int n,q;
    int a[maxn];
    int ans[maxn];
    vector<int>Q[maxn];
    vector<int>St;
    
    void dfs(int u,int fa){
        int k=-1;
        if(!Q[u].empty()){
            for(int i=St.size()-1;i>=0;i--){
                int v=St[i];
                if(__gcd(a[u],a[v])!=1){
                    k=v;break;
                }
            }
            for(int i=0;i<Q[u].size();i++)ans[Q[u][i]]=k;
            Q[u].clear();
        }
        St.push_back(u);
        for(int i=Laxt[u];i;i=Next[i]){
            int v=To[i];
            if(v==fa)continue;
            dfs(v,u);
        }
        St.pop_back();
    }
    
    int main(){
        int n,q;
        scanf("%d%d",&n,&q);
        for(int i=1;i<=n;i++)scanf("%d",&a[i]);
        for(int i=1;i<n;i++){
            int u,v;scanf("%d%d",&u,&v);
            add(u,v);add(v,u);
        }
        int res=0;
        for(int i=1;i<=q;i++){
            int op,x,y;
            scanf("%d",&op);
            if(op==1){
                scanf("%d",&x);
                Q[x].push_back(i);
                res++;
            }
            else{
                if(res)dfs(1,-1);
                ans[i]=-10086;
                scanf("%d%d",&x,&y);
                a[x]=y;
                res=0;
            }
        }
        if(res)dfs(1,-1);
        for(int i=1;i<=q;i++)if(ans[i]!=-10086)printf("%d
    ",ans[i]);
        return 0;
    }
    
  • 相关阅读:
    107、Java中String类之判断开头或结尾
    106、Java中String类之使用contains()方法判断子字符串是否存在
    105、Java中String类之利用indexOf()方法判断子字符串是否存在
    104、Java中String类之使用indexOf()等功能查找
    103、Java中String类之compareTo()方法
    102、Java中String类之相等判断(忽略大小写)
    101、Java中String类之判断是否由数字组成
    100、Java中String类之字符串转为大写
    099、Java中String类之字符数组与字符串的转换
    098、Java中String类之charAt()方法
  • 原文地址:https://www.cnblogs.com/luowentao/p/10556248.html
Copyright © 2011-2022 走看看