码迷,mamicode.com
首页 > 其他好文 > 详细

20170908校内训练

时间:2017-09-20 23:16:25      阅读:199      评论:0      收藏:0      [点我收藏+]

标签:pre   i++   blog   bcd   .com   src   dep   color   dfs   

技术分享

技术分享

技术分享

技术分享

题意:

学过博弈论的同学都知道Nim游戏后手必胜的条件是异或和为0
给定一棵树 ,支持修改单点点权,询问链上异或和

预处理每个点到根的路径的异或和
由于异或的特殊性质,在求链x->y的异或和的时候,我们只需要知道x到根的异或和,y到根的异或和,将他们异或起来,最后异或上lca处的值即可。

如图,查询两个灰色节点的异或和

技术分享

如果一个点的值被修改,那么它和它的子树到根的路径的异或和的值都会被修改

所以,我们维护一棵dfs序为下标的线段树,线段树的sum意义是它的区间的值的异或和

修改一个值,如果该值为x,要变成y,那么给线段树传进一个x^y的值,这样就能做到修改它和它的子树的值

因为一个数异或本身等于0,然后再异或y,就变成y了

修改的区间为dfs[x]~dfs[x]+size[x]-1

查询最近公共祖先我写的是树剖(常数小,又能顺便求dfs序)

技术分享
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int fa[500010],dep[500010],dfs[500010],top[500010],size[500010],son[500010],dfn=0;
int to[1000100],next[1000100],h[500010],k=0;int a[500100],val[500010],n,m;
int l[2000100],r[2000100],sum[2000100],lazy[2000100];
void ins(int u,int v){next[++k]=h[u];h[u]=k;to[k]=v;}
void dfs1(int x,int d,int f)
{
    dep[x]=d;fa[x]=f;size[x]=1;
    for(int i=h[x];i;i=next[i])
    {
        if(to[i]==f)continue;
        dfs1(to[i],d+1,x);size[x]+=size[to[i]];
        if(size[son[x]]<size[to[i]])son[x]=to[i];
    }
}
void dfs2(int x,int tp)
{
    top[x]=tp;dfs[x]=++dfn;a[dfs[x]]=a[dfs[fa[x]]]^val[x];
    if(son[x])dfs2(son[x],tp);
    for(int i=h[x];i;i=next[i])
    {
        if(to[i]==son[x]||to[i]==fa[x])continue;
        dfs2(to[i],to[i]);
    }
}
int lca(int u,int v)
{
    while(top[u]!=top[v])
    {
        if(dep[top[u]]<dep[top[v]])v=fa[top[v]];
        else u=fa[top[u]];
    }
    if(dep[u]<dep[v])return u;
    else return v;
}
void pushdown(int x)
{
    if(!lazy[x])return;
    lazy[x<<1]^=lazy[x];lazy[x<<1|1]^=lazy[x];
    sum[x<<1]^=lazy[x];sum[x<<1|1]^=lazy[x];
    lazy[x]=0;
}
void pushup(int x)
{
    sum[x]=sum[x<<1]^sum[x<<1|1];
}
void build(int x,int L,int R)
{
    l[x]=L;r[x]=R;lazy[x]=0;if(L==R){sum[x]=a[L];return;}
    build(x<<1,L,(L+R)/2);build(x<<1|1,(L+R)/2+1,R);pushup(x);
}
void add(int x,int L,int R,int k)
{
    if(L==l[x]&&R==r[x]){lazy[x]^=k;sum[x]^=k;return;}
    pushdown(x);int mid=(l[x]+r[x])/2;
    if(R<=mid)add(x<<1,L,R,k);
    else if(L>mid)add(x<<1|1,L,R,k);
    else {add(x<<1,L,mid,k);add(x<<1|1,mid+1,R,k);}
    pushup(x);
}
int query(int x,int k)
{
    if(l[x]==k&&r[x]==k){return sum[x];}
    pushdown(x);int mid=(l[x]+r[x])/2;
    if(k<=mid)return query(x<<1,k);else return query(x<<1|1,k);
}
int main()
{
    freopen("game.in","r",stdin);
    freopen("game.out","w",stdout);
    scanf("%d",&n);
    for(int i=1;i<=n;i++)scanf("%d",&val[i]);
    for(int i=1;i<n;i++)
    {
        int x,y;scanf("%d%d",&x,&y);ins(x,y);ins(y,x);
    }
    dfs1(1,1,0);dfs2(1,1);build(1,1,n);
    scanf("%d",&m);
    for(int i=1;i<=m;i++)
    {
        char c[2];int x,y;scanf("%s%d%d",c,&x,&y);
        if(c[0]==C){add(1,dfs[x],dfs[x]+size[x]-1,val[x]^y);val[x]=y;}
        else
        {
            int ans=query(1,dfs[x])^query(1,dfs[y])^val[lca(x,y)];
            puts(ans?"Yes":"No");
        }
    } 
    return 0;
}
View Code

技术分享

  技术分享

技术分享

看到此题,首先想到的是文理分科的模型,但是这题比文理分科还简单

我们发现相邻的格子要选不同的东西才有额外收益,而文理分科是选相同的东西

我们可以把矩阵黑白染色,然后将一种颜色割到S和T表示的含义交换。

这样就满足了题目要求

技术分享
#include<iostream>
#include<cstdio>
#include<queue>
#include<cstring>
using namespace std;
const int INF=1999999999;
int r,c,d;
int n,m;
struct data{
int next,to,cap;
} g[2000000];
int iter[11000],h[11000],level[11000],k=1,head,tail,q[11000];
void add(int from,int to,int cap) {
    g[++k].next=h[from];
    h[from]=k;
    g[k].to=to;
    g[k].cap=cap;
    g[++k].next=h[to];
    h[to]=k;
    g[k].to=from;
    g[k].cap=0;
}
void bfs(int s) {
    memset(level,0,sizeof(level));
    head=tail=0;
    q[tail++]=s;
    level[s]=1;
    while(head!=tail) {
        int u=q[head++];
        for(int i=h[u]; i; i=g[i].next) {
            if(!level[g[i].to]&&g[i].cap) {
                level[g[i].to]=level[u]+1;
                q[tail++]=g[i].to;
            }
        }
    }
}
int dfs(int u,int t,int f) {
    if(u==t)return f;
    int used=0,w;
    for(int &i=iter[u]; i; i=g[i].next) {
        if(g[i].cap&&level[g[i].to]==level[u]+1) {
            w=f-used;
            w=dfs(g[i].to,t,min(w,g[i].cap));
            if(w) {
                g[i].cap-=w;
                g[i^1].cap+=w;
                used+=w;
                if(used==f)return f;
            }
        }
    }
    return used;
}
int dinic(int s,int t) {
    int flow=0;
    for(;;) {
        for(int i=0; i<=n; i++)iter[i]=h[i];
        bfs(s);
        if(!level[t])return flow;
        flow+=dfs(s,t,INF);
    }
}
int main()
{
    freopen("city.in","r",stdin);freopen("city.out","w",stdout);
    int r,c;scanf("%d%d",&r,&c);n=r*c;
    int S=0,T=n+1,tot=0;n++;
    for(int i=1;i<=r;i++)
    for(int j=1;j<=c;j++)
    {
        int u;scanf("%d",&u);tot+=u;
        if(i%2==j%2)add(S,c*(i-1)+j,u);
        else add(c*(i-1)+j,T,u);
    }
    for(int i=1;i<=r;i++)
    for(int j=1;j<=c;j++)
    {
        int u;scanf("%d",&u);tot+=u;
        if(i%2==j%2)add(c*(i-1)+j,T,u);
        else add(S,c*(i-1)+j,u);
    }
    for(int i=1;i<=r-1;i++)
    for(int j=1;j<=c;j++)
    {
        int u;scanf("%d",&u);tot+=u;
        add(c*(i-1)+j,c*i+j,u);
        add(c*i+j,c*(i-1)+j,u);
    }
    for(int i=1;i<=r;i++)
    for(int j=1;j<=c-1;j++)
    {
        int u;scanf("%d",&u);tot+=u;
        add(c*(i-1)+j,c*(i-1)+j+1,u);
        add(c*(i-1)+j+1,c*(i-1)+j,u);
    }
    printf("%d",tot-dinic(S,T));
    return 0;
}
View Code

技术分享

技术分享

技术分享

以下所有数组下标从0开始

我们先预处理出每个位置开始,下一个字母的位置。

即用next[i][j]表示从第i个位置开始,下一个字母j的位置(包括本身),不存在设为451,0对应a,1对应b……

注意next[452][0……25]=451,next[|S|][0……25]=451

考虑状压DP

用dp[S]表示S中字母组成的所有排列的最后位置的最大值

举个例子吧,S=13=1101,即acd三个字母组成的所有排列的最后位置的最大值

dp[S]=max(dp[S],next[dp[s^(1<<i)]+1][i])  (i∈S)dp[0]=-1;

用上面的那个例子,S=13=1101,则dp[S]=max(next[dp[12=1100]+1][0],next[dp[9=1001]+1][2],next[dp[5=0101]+1][3])

最后如果dp[1111……11]=451,则不是,否则是

时间复杂度O(2^n * n),发现无法通过n>21的数据

我们可以发现,最小长度至少为n*(n-1)+1  (当n=4时,即abcdabcdabcda)

而22*21+1>450,所以输出No

技术分享
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int dp[5000000],next[500][27],n;
string a;
int DP(int S)
{
    if(S==0||dp[S]!=-1)return dp[S];
    for(int i=0;i<n;i++)
        if((S>>i)&1)dp[S]=max(dp[S],next[DP(S^(1<<i))+1][i]);
    return dp[S];
}
int main()
{
    freopen("string.in","r",stdin);freopen("string.out","w",stdout);
    int T;scanf("%d",&T);
    while(T--)
    { 
        memset(dp,-1,sizeof(dp));dp[0]=-1;memset(next,0,sizeof(next));
        scanf("%d",&n);cin>>a;if(n>21){puts("NO");continue;}
        int m=a.size()-1;
        for(int i=0;i<n;i++)next[m+1][i]=451,next[452][i]=451;
        next[m][a[m]-a]=m;
        for(int i=m;i>=0;i--)
        {
            for(int j=0;j<n;j++)next[i][j]=next[i+1][j];
            next[i][a[i]-a]=i;
        }
        if(DP((1<<n)-1)==451)puts("NO");else puts("YES");
    }
    return 0;
 } 
View Code

20170908校内训练

标签:pre   i++   blog   bcd   .com   src   dep   color   dfs   

原文地址:http://www.cnblogs.com/lher/p/7565248.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!