2021 ICPC 南京站+上海站 部分题解

小明 2025-04-27 23:02:28 5

CSDN话题���战赛第2期

()

再一次发现一个无比无比菜的自己。。。。

H. Crystalfly

本题难得倒不是思路,是代码的实现。看了下其他大佬的实现,真的惊呼太优雅了。

()

思路:

1.对于这样的一棵树来说,t的值只能取1到3,可发现2秒实在是没有价值。如果一个点是3s,那说明可去其他1s或3s的点拿完,再去拿这个点的值。

2.如果从x点出发,走向z点,而y点的值是3s,说明可以去完z点转向去y点,这样势必会放弃z点孩子结点的点数。

3.因此采用树形dp来记录各种状态。

//dp[u][0]表示不取以u为根的孩子的最大点数

//dp[u][1]表示取以u为根获得的最大点数

状态的转移:

一般情况:dp[u][1]=max(dp[u][1],dp[u][0]+a[v]);表示走向哪个孩子,获得的点数最大

如果一个孩子结点的t值为3:说明这个结点y的权值是可以拿到的,z是另外一个以u为根结点权值最大or次大的点:dp[u][0]+a[y]+dp[z][0]-(dp[z][1]-a[z]),减去的那一部分是dp[u][0]包含的重复的部分。

此外还需对t为3的情况分两种情况讨论下。

#include 
#define int long long
#define endl '\n'
#define ios (ios::sync_with_stdio(false),cin.tie(0),cout.tie(0))
using namespace std;
const int inf=1e18;
const int N=7e5+5;
int n,a[N],t[N],dp[N][2],p[N];
vectore[N];
//dp[u][0]表示不取以u为根的孩子的最大点数
//dp[u][1]表示取以u为根获得的最大点数
void dfs(int u,int fa)
{
    dp[u][1]=dp[u][0]=a[u];
    int mx1=-inf,mx2=-inf;
    for(int v:e[u])
    {
        if(v==fa) continue;
        dfs(v,u);
        dp[u][0]+=dp[v][1]-a[v];
        int tmp=dp[v][0]-dp[v][1]+a[v];
        if(tmp>mx1) mx2=mx1,mx1=tmp;     //最大值
        else if(tmp>mx2) mx2=tmp;       //次大值
    }
    for(int v:e[u])
    {
        if(v==fa) continue;
        dp[u][1]=max(dp[u][1],dp[u][0]+a[v]);
        if(t[v]==3)
        {
            if(dp[v][0]-dp[v][1]+a[v]==mx1)
                dp[u][1]=max(dp[u][1],dp[u][0]+mx2+a[v]);
            else
                dp[u][1]=max(dp[u][1],dp[u][0]+mx1+a[v]);
        }
    }
}
void solve()
{
    cin>>n;
    for(int i=1;ia[i];
    for(int i=1;i>t[i];
    for(int i=1;i
        int u,v;cin>u>>v;
        e[u].push_back(v);
        e[v].push_back(u);
    }
    dfs(1,0);
    cout
    ios;
    int T;cinT;
    while(T--)
        solve();
    return 0;
}

    cinn>>k;
    for(int i=1;i
        cin>a[i];a[i]+=2e6;
        e[a[i]].push_back(a[i]);
        e[a[i]+k].push_back(a[i]);
        g=max(g,a[i]+k);
        mx=max({mx,(int)e[a[i]].size(),(int)e[a[i]+k].size()});
    }
    if(!k)
    {
        cout
        if(e[i].size()==0) continue;
        for(int j=0;j
            sum[j+1][0]=sum[j][0]+(e[i][j]==i);
            sum[j+1][1]=sum[j][1]+(e[i][j]!=i);
        }
        int tmp=sum[1][0]-sum[1][1];
        for(int j=1;j
            tmp=max(tmp,sum[j-1][0]-sum[j-1][1]);
            ans=max(ans,sum[e[i].size()][0]+sum[j][1]-sum[j][0]+tmp);
        }
    }
    cout
    ios;
    //int T;cinT;
    //while(T--)
        solve();
    return 0;
}

    cinpq;
    int gcd=__gcd(p,q);
    p=p/gcd;
    q=q/gcd;
    int g=sqrt(p+2*q);
    if(g*g!=p+2*q)
    {
        cout
        cout
        cout
        cout
        cout
    //ios;
    int T;cinT;
    while(T--)
        solve();
    return 0;
}

    dp[u]=1;
    int cnt=0;
    for(int v:e[u])
    {
        if(v==fa) continue;
        if(!dfs(v,u)) cnt++;
        dp[u]=dp[u]*dp[v]%mod;
    }
    for(int i=1;i
    cinn;
    for(int i=1;i
        int u,v;cinuv;
        e[u].push_back(v);
        e[v].push_back(u);
    }
    dfs(1,0);
    cout
    //ios;
    //int T;cinT;
    //while(T--)
        solve();
    return 0;
}
The End
微信