zoukankan      html  css  js  c++  java
  • Templates

    Game theory

    Nim

    #include<bits/stdc++.h>
    using namespace std;
    int main(){
    	int T;
    	scanf("%d",&T);
    	while(T--){
    		int n,x,res=0;
    		scanf("%d",&n);
    		while(n--){
    			scanf("%d",&x);
    			res^=x;
    		}
    		puts(res==0?"No":"Yes");
    	}
    	return 0;
    }
    

    SG function

    
    

    Graph theory

    SPFA

    void spfa(int start){
    	for(int i=1;i<=n;i++)dis[i]=INF,vis[i]=0;
    	dis[start]=0;
    	q[qhead=qtail=1]=start;
    	while(qhead<=qtail){
    		int u=q[qhead++];
    		vis[u]=0;
    		for(int i=head[u];i;i=e[i].next){
    			int v=e[i].to;
    			if(dis[u]+e[i].w<dis[v]){
    				dis[v]=dis[u]+e[i].w;
    				if(!vis[v]){
    					vis[v]=1;
    					q[++qtail]=v;
    				}
    			}
    		}
    	}
    }
    

    SPFA check negative cycles

    void spfa(){
        for(int i=1;i<=n;i++)cnt[i]=vis[i]=0,dis[i]=INF;
        dis[1]=0;
        queue<int> q;
        q.push(1);
        while(!q.empty()){
            int u=q.top();
            q.pop();
            vis[u]=0;
            for(int i=head[u];i;i=e[i].next){
                int v=e[i].to,w=e[i].w;
                D w=e[i].a*dis[u]+e[i].b;
                if(dis[u]+w<dis[v]&&cnt[v]<=n){
                    dis[v]=w;
                    if(++cnt[v]>n)dis[v]=-INF;
                    if(!vis[v]){
                        vis[v]=1;
                        q.push(v);
                    }
                }
            }
        }
    }
    

    Dijkstra

    void dijkstra(int st){
    	for(int i=1;i<=n;i++){
    		dis[i]=INF;
    		vis[i]=0;
    	}
    	dis[st]=0;
    	priority_queue<P,vector<P>,greater<P> > q;
    	q.push(P(0,st));
    	while(!q.empty()){
    		int u=q.top().second,v;
    		q.pop();
    		if(vis[u])continue;
    		vis[u]=1;
    		for(int i=head[u];i;i=e[i].next){
    			v=e[i].to;
    			if(dis[u]+e[i].w<dis[v]&&!vis[v]){
    				dis[v]=dis[u]+e[i].w;
    				q.push(P(dis[v],v));
    			}
    		}
    	}
    }
    

    Floyd

    for(int k=1;k<=n;k++){
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                dis[i][j]=min(dis[i][j],dis[i][k]+dis[k][j]);
            }
        }
    }
    

    Floyd求路径最值限制问题

    例:过路费

    有一天你来到了一个奇怪的国家,它有 (N) 个城市,城市之间有若干条双向道路连接,每条道路都有一定的费用,经过城市也要一定的费用。从一个城市到达另一个城市的总花费为路径上费用最大的城市费用(包括起点和终点)加上路径上所有的道路的费用。给出 (Q) 次询问,分别回答每次询问中两城市间的最少花费。保证城市之间可以互达。

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=303,INF=0x3f3f3f3f;
    int n,m,Q,p[maxn],a[maxn],g[maxn][maxn],dp[maxn][maxn];
    bool cmp(int x,int y){return a[x]<a[y];}
    void floyd(int k){
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                g[i][j]=min(g[i][j],g[i][k]+g[k][j]);
            }
        }
    }
    int main(){
        scanf("%d%d",&n,&m);
        for(int i=1;i<=n;i++)scanf("%d",a+i);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++)dp[i][j]=g[i][j]=INF;
            g[i][i]=0;
            p[i]=i;
        }
        sort(p+1,p+n+1,cmp);
        for(int i=1;i<=m;i++){
            int u,v,w;
            scanf("%d%d%d",&u,&v,&w);
            g[u][v]=g[v][u]=min(g[u][v],w);
        }
        for(int l=1,r=1;l<=n;l=r){
            for(;r<=n&&a[p[l]]==a[p[r]];r++){
                floyd(p[r]);
            }
            for(int _i=1;_i<r;_i++){
                int i=p[_i];
                for(int _j=1;_j<r;_j++){
                    int j=p[_j];
                    dp[i][j]=min(dp[i][j],g[i][j]+a[p[l]]);
                }
            }
        }
        scanf("%d",&Q);
        while(Q--){
            int x,y;
            scanf("%d%d",&x,&y);
            printf("%d
    ",dp[x][y]);
        }
        return 0;
    }
    

    SCC

    void tarjan(int u){
        dfn[u]=low[u]=++cntdfn;
        stk[++top]=u;
        for(int i=head[u];~i;i=e[i].next){
            int v=e[i].to;
            if(!dfn[v]){
                tarjan(v);
                low[u]=min(low[u],low[v]);
            }
            else if(!incyc[v]){
                low[u]=min(low[u],dfn[v]);
            }
        }
        if(dfn[u]==low[u]){
            numcyc[++cntcyc]=0;
            int x;
            do{
                x=stk[top];
                top--;
                numcyc[cntcyc]++;
                incyc[x]=cntcyc;
            }while(x!=u);
        }
    }
    void getcyc(){
        for(int i=1;i<=n;i++){
            if(!dfn[i])tarjan(i);
        }
    }
    

    topological-sort

    void topsort(){
    	queue<int> q;
    	for(int i=1;i<=n;i++){
    		if(indeg[i]==0){
    			q.push(i);
    		}
    	}
    	while(!q.empty()){
    		int u=q.front();
    		q.pop();
    		for(int i=head[u];i;i=e[i].next){
    			int v=e[i].to;
    			if(--indeg[v]==0){
    				q.push(v);
    			}
    		}
    	}
    }
    

    bipartite-graph max-match (Hungary algorithm)

    bool dfs(int u){
        for(int i=0;i<int(g[u].size());i++){
            int v=g[u][i];
            if(vis[v])continue;
            vis[v]=1;
            if(!c[v]||dfs(c[v])){
            	d[u]=v;
                c[v]=u;
                return 1;
            }
        }
        return 0;
    }
    int match(){
        int ret=0;
        for(int i=1;i<=n1;i++){
            for(int j=1;j<=n2;j++)vis[j]=0;
            if(dfs(i))ret++;
        }
        return ret;
    }
    int main(){
        scanf("%d%d%d",&n1,&n2,&m);
        for(int i=1;i<=m;i++){
            int u,v;
            scanf("%d%d",&u,&v);
            g[u].push_back(v);
        }
        printf("%d
    ",match());
        return 0;
    }
    

    bipartite-graph max-match in minimum lexicographic order (Hungary algorithm)

    bool dfs(int u){
        for(int i=0;i<int(g[u].size());i++){
            int v=g[u][i];
            if(vis[v])continue;
            vis[v]=1;
            if(!c[v]||dfs(c[v])){
            	d[u]=v;
                c[v]=u;
                return 1;
            }
        }
        return 0;
    }
    int match(){
        int ret=0;
        for(int i=n1;i>=1;i--){
            for(int j=1;j<=n2;j++)vis[j]=0;
            if(dfs(i))ret++;
        }
        return ret;
    }
    int main(){
        scanf("%d%d%d",&n1,&n2,&m);
        for(int i=1;i<=m;i++){
            int u,v;
            scanf("%d%d",&u,&v);
            g[u].push_back(v);
        }
        for(int i=1;i<=n1;i++)sort(g[i].begin(),g[i].end());
        printf("%d
    ",match());
        return 0;
    }
    

    bipartite-graph perfect-match in maximum weight (KM-algorithm)

    bool dfs(int u){
        visd[u]=1;
        for(int v=1;v<=n;v++){
            int tmp=ld[u]+lc[v]-g[u][v];
            if(!visc[v]&&!tmp){
                visc[v]=1;
                if(!c[v]||dfs(c[v])){
                    c[v]=u;
                    return 1;
                }
            }
            else if(tmp<slack[v]){
                slack[v]=tmp;
            }
        }
        return 0;
    }
    int km(){
        for(int i=1;i<=n;i++){
            c[i]=lc[i]=ld[i]=0;
            for(int j=1;j<=n;j++){
                ld[i]=max(ld[i],g[i][j]);
            }
        }
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++)slack[j]=INF;
            while(1){
                for(int j=1;j<=n;j++)visc[j]=visd[j]=0;
                if(dfs(i))break;
                int res=INF;
                for(int v=1;v<=n;v++){
                    if(!visc[v]){
                        res=min(res,slack[v]);
                    }
                }
                for(int u=1;u<=n;u++){
                    if(visd[u]){
                        ld[u]-=res;
                    }
                }
                for(int v=1;v<=n;v++){
                    if(visc[v]){
                        lc[v]+=res;
                    }
                    else{
                        slack[v]-=res;
                    }
                }
            }
        }
        int ans=0;
        for(int i=1;i<=n;i++)ans+=ld[i]+lc[i];
        return ans;
    }
    

    network-flow (Dinic)

    bool bfs(){
    	for(int i=1;i<=n;i++)dep[i]=0,head1[i]=head[i];
    	dep[s]=1;
    	int *qhead=q,*qtail=q;
    	*qtail++=s;
    	while(qhead!=qtail){
    		int u=*qhead++;
    		for(int i=head[u];~i;i=e[i].next){
    			int v=e[i].to;
    			if(e[i].w&&dep[v]==0){
    				dep[v]=dep[u]+1;
    				*qtail++=v;
    			}
    		}
    	}
    	return dep[t]!=0;
    }
    int dfs(int u,int low){
    	if(low==0||u==t)return low;
    	int flow=0;
    	for(int &i=head1[u];~i;i=e[i].next){
    		int v=e[i].to;
    		if(e[i].w&&dep[v]==dep[u]+1){
    			int tmp=dfs(v,min(low,e[i].w));
    			if(tmp==0)dep[v]=0;
    			else{
    				flow+=tmp;
    				low-=tmp;
    				e[i].w-=tmp;
    				e[i^1].w+=tmp;
    				if(low==0)break;
    			}
    		}
    	}
    	return flow;
    }
    int dinic(){
    	int ans=0;
    	while(bfs())ans+=dfs(s,INF);
    	return ans;
    }
    int main(){
    	int m;
    	scanf("%d%d%d%d",&n,&m,&s,&t);
    	for(int i=1;i<=n;i++)head[i]=-1;
    	cnte=-1;
    	while(m--){
    		int u,v,w;
    		scanf("%d%d%d",&u,&v,&w);
    		addedge(u,v,w);
    	}
    	printf("%d",dinic());
    	return 0;
    }
    

    network-flow (SPFA+EK)

    bool spfa(){
    	for(int i=1;i<=n;i++)flow[i]=dis[i]=INF,vis[i]=0;
    	pre[t]=-1;
    	dis[s]=0;
    	queue<int> q;
    	q.push(s);
    	while(!q.empty()){
    		int u=q.front();
    		q.pop();
    		vis[u]=0;
    		for(int i=head[u];~i;i=e[i].next){
    			int v=e[i].to;
    			if(e[i].w>0&&dis[u]+e[i].cost<dis[v]){
    				dis[v]=dis[u]+e[i].cost;
    				pre[v]=u;
    				edg[v]=i;
    				flow[v]=min(flow[u],e[i].w);
    				if(!vis[v]){
    					vis[v]=1;
    					q.push(v);
    				}
    			}
    		}
    	}
    	return pre[t]!=-1;
    }
    void micmxf(){
    	while(spfa()){
    		int x=t;
    		while(x!=s){
    			e[edg[x]].w-=flow[t];
    			e[edg[x]^1].w+=flow[t];
    			x=pre[x];
    		}
    		maxflow+=flow[t];
    		mincost+=flow[t]*dis[t];
    	}
    }
    int main(){
    	int m;
    	scanf("%d%d%d%d",&n,&m,&s,&t);
    	for(int i=0;i<=n;i++)head[i]=-1;
    	cnte=-1;
    	for(int i=1;i<=m;i++){
    		int u,v,w,cost;
    		scanf("%d%d%d%d",&u,&v,&w,&cost);
    		add(u,v,w,cost);
    		add(v,u,0,-cost);
    	}
    	micmxf();
    	printf("%d %d
    ",maxflow,mincost);
    	return 0;
    }
    

    MST (Prim)

    void prim(){
        for(int i=1;i<=n;i++)dis[i]=INT_MAX;
        priority_queue<pii,vector<pii>,greater<pii> > q;
        q.push(make_pair(0,1));
        while(!q.empty()){
            int d=q.top().first,u=q.top().second;
            q.pop();
            if(vis[u])continue;
            ans+=d;
            vis[u]=1;
            cnt++;
            for(int i=head[u];i;i=e[i].next){
                int v=e[i].to;
                if(dis[v]>e[i].w&&!vis[v]){
                    dis[v]=e[i].w;
                    q.push(make_pair(dis[v],v));
                }
            }
        }
    }
    

    MST (Kruskal)

    void kruskal(){
        sort(e+1,e+m+1);
        ufset s(n);
        for(int i=1;i<=m;i++){
            int u=e[i].from,v=e[i].to;
            if(s.same(u,v))continue;
            ans+=e[i].w;
            s.Union(u,v);
            if(++tot==n-1)break;
        }
    }
    

    Strictly second-best MST

    #include<bits/stdc++.h>
    #define maxn 100003
    #define maxm 300003
    #define INF 100000000000000000ll
    using namespace std;
    typedef long long D;
    D n,m,lg[maxn];
    namespace G{
        struct edge{
            D from,to,w;
            bool vis;
            bool operator <(const edge& x)const{return w<x.w;}
        }e[maxm];
        D cnte;
        void add(D u,D v,D w){e[++cnte]=(edge){u,v,w,0};}
    }
    namespace MST{
        struct edge{
            D to,next,w;
        }e[maxn<<1];
        D head[maxn],cnte,dep[maxn],fa[maxn][23],mx[maxn][23][2];
        void add(D u,D v,D w){e[++cnte]=(edge){v,head[u],w},head[u]=cnte;}
        void calc(D &X0,D &X1,D Y0,D Y1,D Z0,D Z1){
            X0=max(Y0,Z0);
            if(Y0!=Z0)X1=max(min(Y0,Z0),max(Y1,Z1));
            else X1=max(Y1,Z1);
        }
        void init(D u,D last){
            fa[u][0]=last;
            dep[u]=dep[last]+1;
            for(D i=1;(1<<i)<=dep[u];i++){
                D up=fa[u][i-1];
                fa[u][i]=fa[up][i-1];
                calc(mx[u][i][0],mx[u][i][1],mx[u][i-1][0],mx[u][i-1][1],mx[up][i-1][0],mx[up][i-1][1]);
            }
            for(D i=head[u];i;i=e[i].next){
                D v=e[i].to;
                if(v==last)continue;
                mx[v][0][0]=e[i].w;
                init(v,u);
            }
        }
        D lca(D x,D y){
            if(dep[x]<dep[y])swap(x,y);
            while(dep[x]>dep[y]){
                x=fa[x][lg[dep[x]-dep[y]]];
            }
            if(x==y)return x;
            for(D i=lg[dep[x]];i>=0;i--){
                if(fa[x][i]!=fa[y][i]){
                    x=fa[x][i];
                    y=fa[y][i];
                }
            }
            return fa[x][0];
        }
        D query(D x,D y,D w){
            D l=lca(x,y);
            D ans0=0,ans1=0;
            while(x!=l){
                calc(ans0,ans1,ans0,ans1,mx[x][lg[dep[x]-dep[l]]][0],mx[x][lg[dep[x]-dep[l]]][1]);
                x=fa[x][lg[dep[x]-dep[l]]];
            }
            while(y!=l){
                calc(ans0,ans1,ans0,ans1,mx[y][lg[dep[y]-dep[l]]][0],mx[y][lg[dep[y]-dep[l]]][1]);
                y=fa[y][lg[dep[y]-dep[l]]];
            }
            if(w!=ans0)return w-ans0;
            if(ans1)return w-ans1;
            return INF;
        }
    }
    D f[maxn];
    D find(D x){
        if(x!=f[x])f[x]=find(f[x]);
        return f[x];
    }
    D kruskal(){
        D cnt=0,ret=0;
        sort(G::e+1,G::e+m+1);
        for(D i=1;i<=n;i++)f[i]=i;
        for(D i=1;i<=m;i++){
            D u=G::e[i].from,v=G::e[i].to,fu=find(u),fv=find(v);
            if(fu!=fv){
                f[fu]=fv;
                ret+=G::e[i].w;
                G::e[i].vis=1;
                MST::add(u,v,G::e[i].w);
                MST::add(v,u,G::e[i].w);
                cnt++;
                if(cnt>=n)break;
            }
        }
        return ret;
    }
    signed main(){
        scanf("%lld%lld",&n,&m);
        for(D i=1;i<=m;i++){
            D u,v,w;
            scanf("%lld%lld%lld",&u,&v,&w);
            G::add(u,v,w);
        }
        D ans1=kruskal(),ans2=INF;
        for(D i=2;i<maxn;i++)lg[i]=lg[i-1]+((1<<(lg[i-1]+1))==i);
        MST::init(1,0);
        for(D i=1;i<=m;i++){
            if(!G::e[i].vis){
                D u=G::e[i].from,v=G::e[i].to;
                ans2=min(ans2,MST::query(u,v,G::e[i].w));
            }
        }
        printf("%lld
    ",ans1+ans2);
        return 0;
    }
    

    LCA (Heavy-light decomposition)

    void initdep(int u,int last){
    	sz[u]=1;
    	fa[u]=last;
    	int mx=0;
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==fa[u])continue;
    		dep[v]=dep[u]+1;
    		initdep(v,u);
    		sz[u]+=sz[v];
    		if(sz[v]>mx)mx=sz[v],son[u]=v;
    	}
    }
    void initdfn(int u,int tp){
    	dfn[u]=++cntdfn;
    	num[cntdfn]=u;
    	top[u]=tp;
    	if(son[u]){
    		initdfn(son[u],tp);
    	}
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==fa[u]||v==son[u])continue;
    		initdfn(v,v);
    	}
    	rig[u]=cntdfn;
    }
    int lca(int x,int y){
    	while(top[x]!=top[y]){
    		if(dep[top[x]]<dep[top[y]])swap(x,y);
    		x=fa[top[x]];
    	}
    	if(dep[x]<dep[y])swap(x,y);
    	return y;
    }
    

    LCA (ST RMQ)

    void init(int u,int last){
        st[++cntdfn][0]=u;
        dfn[u]=cntdfn;
        for(int i=head[u];i;i=e[i].next){
            int v=e[i].to;
            if(v==last)continue;
            dep[v]=dep[u]+1;
            init(v,u);
            st[++cntdfn][0]=u;
        }
    }
    int MIN(int x,int y){return dep[x]<dep[y]?x:y;}
    void initst(){
        for(int len=1;(1<<len)<=cntdfn;len++){
            for(int i=1;i+(1<<len)-1<=cntdfn;i++){
                st[i][len]=MIN(st[i][len-1],st[i+(1<<(len-1))][len-1]);
            }
        }
    }
    int lca(int x,int y){
        if(dfn[x]>dfn[y])swap(x,y);
        int tmp=lg[dfn[y]-dfn[x]+1];
        return MIN(st[dfn[x]][tmp],st[dfn[y]-(1<<tmp)+1][tmp]);
    }
    

    LCA (doubling)

    void init(int u){
        for(int i=1;(1<<i)<=dep[u];i++)fa[u][i]=fa[fa[u][i-1]][i-1];
        for(int i=head[u];i;i=e[i].next){
            int v=e[i].to;
            if(v==fa[u][0])continue;
            fa[v][0]=u;
            dep[v]=dep[u]+1;
            init(v);
        }
    }
    int lca(int x,int y){
        if(dep[x]<dep[y])swap(x,y);
        while(dep[x]>dep[y])x=fa[x][lg[dep[x]-dep[y]]];
        if(x==y)return x;
        for(int i=lg[dep[x]];i>=0;i--)if(fa[x][i]!=fa[y][i])x=fa[x][i],y=fa[y][i];
        return fa[x][0];
    }
    

    Virtual tree

    例:[SDOI2011]消耗战

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=250003,maxm=500003,INF=0x3f3f3f3f;
    struct edge{int to,next,w;}e[maxn<<1];
    int head[maxn],cnte;
    void add(int u,int v,int w){e[++cnte].to=v,e[cnte].w=w,e[cnte].next=head[u],head[u]=cnte;}
    int n,lg[maxn],fa[maxn][23],cntdfn,dfn[maxn],rig[maxn],dep[maxn],p[maxn<<2],
    mi[maxn],stk[maxn],top;
    long long dp[maxn];
    bool vis[maxn];
    void init(int u,int last){
    	dfn[u]=++cntdfn;
    	fa[u][0]=last;
    	for(int i=1;(1<<i)<=dep[u];i++)fa[u][i]=fa[fa[u][i-1]][i-1];
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==last)continue;
    		dep[v]=dep[u]+1;
    		mi[v]=min(mi[u],e[i].w);
    		init(v,u);
    	}
    	rig[u]=++cntdfn;
    }
    int lca(int x,int y){
    	if(dep[x]<dep[y])swap(x,y);
    	while(dep[x]>dep[y])x=fa[x][lg[dep[x]-dep[y]]];
    	if(x==y)return x;
    	for(int i=lg[dep[x]];i>=0;i--)if(fa[x][i]!=fa[y][i])x=fa[x][i],y=fa[y][i];
    	return fa[x][0];
    }
    bool cmp(int x,int y){return (x>0?dfn[x]:rig[-x])<(y>0?dfn[y]:rig[-y]);}
    int main(){
    	for(int i=2;i<maxn;i++)lg[i]=lg[i-1]+((1<<(lg[i-1]+1))==i);
    	scanf("%d",&n);
    	for(int i=1;i<n;i++){
    		int u,v,w;
    		scanf("%d%d%d",&u,&v,&w);
    		add(u,v,w),add(v,u,w);
    	}
    	mi[1]=INF;
    	init(1,0);
    	int Q;
    	scanf("%d",&Q);
    	while(Q--){
    		int m;
    		scanf("%d",&m);
    		for(int i=1;i<=m;i++)scanf("%d",p+i),vis[p[i]]=1,dp[p[i]]=mi[p[i]];
    		sort(p+1,p+m+1,cmp);
    		for(int i=2;i<=m;i++){
    			int l=lca(p[i-1],p[i]);
    			if(vis[l])continue;
    			vis[l]=1;
    			p[++m]=l;
    		}
    		for(int i=m;i>=1;i--)p[++m]=-p[i];
    		if(!vis[1]){
    			p[++m]=1,p[++m]=-1;
    			vis[1]=1;
    		}
    		sort(p+1,p+m+1,cmp);
    		top=0;
    		for(int i=1;i<m;i++){
    			if(p[i]>0){
    				stk[++top]=p[i];
    			}
    			else{
    				int u=stk[top--],last=stk[top];
    				dp[last]+=min(dp[u],(long long)mi[u]);
    				vis[u]=0,dp[u]=0;
    			}
    		}
    		printf("%lld
    ",dp[1]);
    		vis[1]=0,dp[1]=0;
    	}
    	return 0;
    }
    

    Tree-plus-cycle (containing multiple edges)

    bool init(int u){
    	stk[++top]=u;
    	instk[u]=1;
    	for(int i=head[u];i;i=e[i].next){
    		int id=(i+1)>>1;
    		if(vis[id])continue;
    		vis[id]=1;
    		int v=e[i].to;
    		if(instk[v]){
    			int x;
    			do{
    				x=stk[top--];
    				p[++tot]=x;
    			}while(x!=v);
    			return 1;
    		}
    		if(init(v))return 1;
    	}
    	top--;
    	return 0;
    }
    

    Divide-and-conquer on tree

    例:【模板】点分治

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=10003,INF=0x3f3f3f3f;
    struct edge{int to,next,w;}e[maxn<<1];
    int head[maxn],cnte;
    void add(int u,int v,int w){e[++cnte].to=v,e[cnte].w=w,e[cnte].next=head[u],head[u]=cnte;}
    int n,m,k,dis[maxn],cntdis;
    bool vis[maxn];
    namespace GG{
    	int sz[maxn],G,szG;
    	int inittrsz(int u,int last){
    		int trsz=1;
    		for(int i=head[u];i;i=e[i].next){
    			int v=e[i].to;
    			if(v==last||vis[v])continue;
    			trsz+=inittrsz(v,u);
    		}
    		return trsz;
    	}
    	void getG1(int u,int last,int trsz){
    		int ret=0;
    		sz[u]=1;
    		for(int i=head[u];i;i=e[i].next){
    			int v=e[i].to;
    			if(v==last||vis[v])continue;
    			getG1(v,u,trsz);
    			ret=max(ret,sz[v]);
    			sz[u]+=sz[v];
    		}
    		ret=max(ret,trsz-sz[u]);
    		if(ret<szG){
    			szG=ret;
    			G=u;
    		}
    	}
    	int getG(int u){
    		int trsz=inittrsz(u,0);
    		szG=INF;
    		getG1(u,0,trsz);
    		return G;
    	}
    }
    void work(int u,int last,int depth){
    	dis[++cntdis]=depth;
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==last||vis[v])continue;
    		work(v,u,depth+e[i].w);
    	}
    }
    int solve(int u,int kk){
    	cntdis=0;
    	work(u,0,0);
    	sort(dis+1,dis+cntdis+1);
    	int ans=0;
    	pair<int*,int*> tmp;
    	for(int i=1;i<=cntdis;i++){
    		tmp=equal_range(dis+i+1,dis+cntdis+1,kk-dis[i]);
    		ans+=tmp.second-tmp.first;
    	}
    	return ans;
    }
    bool dfs(int g){
    	int ans=solve(g,k);
    	vis[g]=1;
    	for(int i=head[g];i;i=e[i].next){
    		int v=e[i].to;
    		if(vis[v])continue;
    		ans-=solve(v,k-(e[i].w<<1));
    	}
    	if(ans)return 1;
    	for(int i=head[g];i;i=e[i].next){
    		int v=e[i].to;
    		if(vis[v])continue;
    		int gg=GG::getG(v);
    		if(dfs(gg))return 1;
    	}
    	return 0;
    }
    int main(){
    	scanf("%d%d",&n,&m);
    	for(int i=1;i<n;i++){
    		int u,v,w;
    		scanf("%d%d%d",&u,&v,&w);
    		add(u,v,w),add(v,u,w);
    	}
    	int g=GG::getG(1);
    	while(m--){
    		scanf("%d",&k);
    		for(int i=1;i<=n;i++)vis[i]=0;
    		puts(dfs(g)?"AYE":"NAY");
    	}
    	return 0;
    }
    

    DSU on tree

    例:Codeforces 600E

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=100003;
    struct edge{int to,next;}e[maxn<<1];
    int head[maxn],cnte;
    void add(int u,int v){e[++cnte].to=v,e[cnte].next=head[u],head[u]=cnte;}
    int n,a[maxn],sz[maxn],son[maxn],buc[maxn],mxcnt;
    long long sum,ANS[maxn];
    void init(int u,int last){
    	sz[u]=1;
    	int res=0;
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==last)continue;
    		init(v,u);
    		sz[u]+=sz[v];
    		if(sz[v]>res)res=sz[v],son[u]=v;
    	}
    }
    void add(int u,int last,int heavy){
    	buc[a[u]]++;
    	if(buc[a[u]]>mxcnt)mxcnt=buc[a[u]],sum=a[u];
    	else if(buc[a[u]]==mxcnt)sum+=a[u];
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==last||v==heavy)continue;
    		add(v,u,heavy);
    	}
    }
    void del(int u,int last){
    	buc[a[u]]--;
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==last)continue;
    		del(v,u);
    	}
    }
    void dfs(int u,int last,bool keep){
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==last||v==son[u])continue;
    		dfs(v,u,0);
    	}
    	if(son[u])dfs(son[u],u,1);
    	add(u,last,son[u]);
    	ANS[u]=sum;
    	if(!keep){
    		mxcnt=sum=0;
    		del(u,last);
    	}
    }
    int main(){
    	scanf("%d",&n);
    	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);
    	}
    	init(1,0);
    	dfs(1,0,0);
    	for(int i=1;i<=n;i++)printf("%lld ",ANS[i]);
    	return 0;
    }
    

    Data structures

    Heavy-light-decomposition

    namespace SEG{
        struct node{
            long long sum,z;
        }t[maxn<<2];
        void pushdown(int p,int l,int r){
            if(l==r){t[p].z=0;return;}
            if(t[p].z){
                int mid=(l+r)>>1;
                PlusEqual(t[p<<1].sum,L(t[p].z*(mid-l+1)));
                PlusEqual(t[p<<1|1].sum,L(t[p].z*(r-mid)));
                PlusEqual(t[p<<1].z,t[p].z);
                PlusEqual(t[p<<1|1].z,t[p].z);
                t[p].z=0;
            }
        }
        void pushup(int p,int l,int r){
            t[p].sum=L(t[p<<1].sum+t[p<<1|1].sum);
        }
        void build(int p,int l,int r){
            t[p].z=0;
            if(l==r){
                t[p].sum=a[num[l]];
                return;
            }
            int mid=(l+r)>>1;
            build(p<<1,l,mid);
            build(p<<1|1,mid+1,r);
            pushup(p,l,r);
        }
        void change(int p,int l,int r,int pos,long long k){
            pushdown(p,l,r);
            if(l==r){
                PlusEqual(t[p].sum,k);
                return;
            }
            int mid=(l+r)>>1;
            if(pos<=mid)change(p<<1,l,mid,pos,k);
            else change(p<<1|1,mid+1,r,pos,k);
            pushup(p,l,r);
        }
        void change(int p,int l,int r,int seg_l,int seg_r,long long k){
            pushdown(p,l,r);
            if(seg_l<=l&&r<=seg_r){
                PlusEqual(t[p].sum,L(k*(r-l+1)));
                t[p].z=k;
                return;
            }
            int mid=(l+r)>>1;
            if(seg_l<=mid)change(p<<1,l,mid,seg_l,seg_r,k);
            if(seg_r>mid)change(p<<1|1,mid+1,r,seg_l,seg_r,k);
            pushup(p,l,r);
        }
        long long query(int p,int l,int r,int seg_l,int seg_r){
            pushdown(p,l,r);
            if(seg_l<=l&&r<=seg_r){
                return t[p].sum;
            }
            int mid=(l+r)>>1;
            long long ret=0;
            if(seg_l<=mid)PlusEqual(ret,query(p<<1,l,mid,seg_l,seg_r));
            if(seg_r>mid)PlusEqual(ret,query(p<<1|1,mid+1,r,seg_l,seg_r));
            return ret;
        }
    }
    
    void initdep(int u,int last,int depth){
        fa[u]=last;
        dep[u]=depth;
        sz[u]=1;
        int mxsz=0;
        for(int i=head[u];~i;i=e[i].next){
            int v=e[i].to;
            if(v==fa[u])continue;
            initdep(v,u,depth+1);
            sz[u]+=sz[v];
            if(sz[v]>mxsz)mxsz=sz[v],son[u]=v;
        }
    }
    void initdfn(int u,int tp){
        dfn[u]=++cntdfn;
        num[cntdfn]=u;
        top[u]=tp;
        if(son[u]){
            initdfn(son[u],tp);
            for(int i=head[u];~i;i=e[i].next){
                int v=e[i].to;
                if(v==fa[u]||v==son[u])continue;
                initdfn(v,v);
            }
        }
        rig[u]=cntdfn;
    }
    void change(int u,long long w){
        SEG::change(1,1,n,dfn[u],w);
    }
    void change(int x,int y,long long w){
        while(top[x]!=top[y]){
            if(dep[top[x]]<dep[top[y]])swap(x,y);
            SEG::change(1,1,n,dfn[top[x]],dfn[x],w);
            x=fa[top[x]];
        }
        if(dep[x]<dep[y])swap(x,y);
        SEG::change(1,1,n,dfn[y],dfn[x],w);
    }
    void changesub(int u,long long w){
        SEG::change(1,1,n,dfn[u],rig[u],w);
    }
    long long query(int x,int y){
        long long ans=0;
        while(top[x]!=top[y]){
            if(dep[top[x]]<dep[top[y]])swap(x,y);
            PlusEqual(ans,SEG::query(1,1,n,dfn[top[x]],dfn[x]));
            x=fa[top[x]];
        }
        if(dep[x]<dep[y])swap(x,y);
        PlusEqual(ans,SEG::query(1,1,n,dfn[y],dfn[x]));
        return ans;
    }
    long long querysub(int x){
        return SEG::query(1,1,n,dfn[x],rig[x]);
    }
    

    LCT

    例:QTREE

    #include<stdio.h>
    #define maxn 20003
    void swap(int *x,int *y){int t=*x;*x=*y;*y=t;}
    int max(int x,int y){return x>y?x:y;}
    
    struct node{
        int val,mx,fa,son[2],z;
    }t[maxn];
    int isroot(int p){return t[t[p].fa].son[0]!=p&&t[t[p].fa].son[1]!=p;}
    int chk(int p){return t[t[p].fa].son[1]==p;}
    void reverse(int p){if(p==0)return;swap(&t[p].son[0],&t[p].son[1]);t[p].z^=1;}
    void pushup(int p){t[p].mx=max(max(t[t[p].son[0]].mx,t[t[p].son[1]].mx),t[p].val);}
    void pushdown(int p){
        if(t[p].z){
            reverse(t[p].son[0]);
            reverse(t[p].son[1]);
            t[p].z=0;
        }
    }
    void pushall(int p){if(!isroot(p))pushall(t[p].fa);pushdown(p);}
    void rotate(int p){
        int q=t[p].fa,r=t[q].fa,k=chk(p),s=t[p].son[!k];
        t[q].son[k]=s,t[s].fa=q;
        if(!isroot(q))t[r].son[chk(q)]=p;
        t[p].fa=r;
        t[p].son[!k]=q,t[q].fa=p;
        pushup(q);
        pushup(p);
    }
    void splay(int p){
        pushall(p);
        while(!isroot(p)){
            int q=t[p].fa;
            if(!isroot(q)){
                if(chk(p)==chk(q))rotate(q);
                else rotate(p);
            }
            rotate(p);
        }
    }
    void new_node(int p,int fa,int k){t[p].fa=fa,t[p].mx=t[p].val=k;}
    
    void access(int x){for(int y=0;x;y=x,x=t[x].fa)splay(x),t[x].son[1]=y,pushup(x);}
    void change(int x,int k){splay(x),t[x].val=k,pushup(x);}
    void makeroot(int x){access(x),splay(x),reverse(x);}
    int query(int x,int y){makeroot(x),access(y),splay(y);return t[y].mx;}
    
    struct edge{int to,next;}e[maxn<<1];
    int head[maxn],cnte;
    void add(int u,int v){e[++cnte].to=v,e[cnte].next=head[u],head[u]=cnte;}
    int n,a[maxn],ed[maxn];
    void dfs(int u,int last){
        new_node(u,last,a[u]);
        for(int i=head[u];i;i=e[i].next){
            int v=e[i].to;
            if(v==last)continue;
            dfs(v,u);
        }
    }
    void init(){
        for(int i=0;i<=(n<<1);i++)t[i].val=t[i].mx=t[i].fa=t[i].son[0]=t[i].son[1]=t[i].z=head[i]=a[i]=ed[i]=0;
        cnte=0;
    }
    char mo[13];
    int main(){
        int T;
        scanf("%d",&T);
        while(T--){
            scanf("%d",&n);
            init();
            for(int i=1;i<n;i++){
                int u,v,w;
                scanf("%d",&u),scanf("%d",&v),scanf("%d",&w);
                add(u,i+n),add(i+n,u),add(i+n,v),add(v,i+n);
                ed[i]=i+n;
                a[i+n]=w;
            }
            n=n+n-1;
            dfs(1,0);
            while(scanf("%s",mo),*mo!='D'){
                int x,y;
                scanf("%d",&x),scanf("%d",&y);
                if(*mo=='Q'){
                    printf("%d
    ",query(x,y));
                }
                else{
                    change(ed[x],y);
                }
            }
        }
        return 0;
    }
    

    QTREE7

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=100003,INF=0x3f3f3f3f;
    template<typename tp>
    void read(tp& x){
    	x=0;
    	char c=getchar();
    	bool sgn=0;
    	while((c<'0'||c>'9')&&c!='-')c=getchar();
    	if(c=='-')sgn=1,c=getchar();
    	while(c>='0'&&c<='9')x=(x<<3)+(x<<1)+c-'0',c=getchar();
    	if(sgn)x=-x;
    }
    template<typename tp>
    void write(tp x){
    	if(x<0)putchar('-'),write(-x);
    	else{
    		if(x>=10)write(x/10);
    		putchar(x%10+'0');
    	}
    }
    
    struct edge{int to,next;}e[maxn<<1];
    int head[maxn],cnte;
    void add(int u,int v){e[++cnte].to=v,e[cnte].next=head[u],head[u]=cnte;}
    int n,fa[maxn],val[maxn],a[maxn];
    
    struct LCT{
    	struct node{
    		int mx,fa,son[2];
    	};node t[maxn];
    	multiset<int,greater<int> > light[maxn];
    	bool isroot(int p){return t[t[p].fa].son[0]!=p&&t[t[p].fa].son[1]!=p;}
    	bool chk(int p){return t[t[p].fa].son[1]==p;}
    	void pushup(int p){
    		t[p].mx=max(max(max(t[t[p].son[0]].mx,t[t[p].son[1]].mx),a[p]),light[p].empty()?-INF:*light[p].begin());
    	}
    	void rotate(int p){
    		int q=t[p].fa,r=t[q].fa,k=chk(p),s=t[p].son[!k];
    		t[q].son[k]=s,t[s].fa=q;
    		if(!isroot(q))t[r].son[chk(q)]=p;
    		t[p].fa=r;
    		t[p].son[!k]=q,t[q].fa=p;
    		pushup(q);
    		pushup(p);
    	}
    	void splay(int p){
    		while(!isroot(p)){
    			int q=t[p].fa,r=t[q].fa;
    			if(!isroot(q)){
    				if(chk(p)==chk(q))rotate(q);
    				else rotate(p);
    			}
    			rotate(p);
    		}
    	}
    	int find(int p){
    		while(t[p].son[0])p=t[p].son[0];
    		splay(p);
    		return p;
    	}
    
    	void access(int x){
    		for(int y=0;x;y=x,x=t[x].fa){
    			splay(x);
    			light[x].insert(t[t[x].son[1]].mx);
    			t[x].son[1]=y;
    			light[x].erase(light[x].lower_bound(t[t[x].son[1]].mx));
    			pushup(x);
    		}
    	}
    	void link(int x){
    		if(fa[x]==0)return;
    		access(fa[x]),splay(fa[x]),splay(x);
    		t[x].fa=fa[x];
    		light[fa[x]].insert(t[x].mx);
    		pushup(fa[x]);
    	}
    	void cut(int x){
    		if(fa[x]==0)return;
    		access(x),splay(x);
    		t[x].son[0]=t[t[x].son[0]].fa=0;
    		pushup(x);
    	}
    	int query(int x){
    		access(x),splay(x);
    		int tmp=find(x);
    		return val[tmp]==val[x]?t[tmp].mx:t[t[tmp].son[1]].mx;
    	}
    
    	void init(int n){for(int i=0;i<=n;i++)t[i].mx=-INF,t[i].fa=t[i].son[0]=t[i].son[1]=0;}
    }tree[2];
    
    void dfs(int u,int last){
    	fa[u]=last;
    	for(int i=head[u];i;i=e[i].next){
    		int v=e[i].to;
    		if(v==last)continue;
    		dfs(v,u);
    	}
    }
    
    int main(){
    	read(n);
    	tree[0].init(n);
    	tree[1].init(n);
    	for(int i=0;i<=n;i++)a[i]=-INF;
    	for(int i=1;i<n;i++){
    		int u,v;
    		read(u),read(v);
    		add(u,v),add(v,u);
    	}
    	for(int i=1;i<=n;i++)read(val[i]);
    	for(int i=1;i<=n;i++)read(a[i]),tree[0].t[i].mx=tree[1].t[i].mx=a[i];
    	dfs(1,0);
    	for(int i=1;i<=n;i++)tree[val[i]].link(i);
    	int Q;
    	read(Q);
    	while(Q--){
    		int mo,x;
    		read(mo),read(x);
    		if(mo==1){
    			tree[val[x]].cut(x);
    			val[x]^=1;
    			tree[val[x]].link(x);
    		}
    		else if(mo==2){
    			int y;
    			read(y);
    			tree[0].access(x),tree[0].splay(x),tree[1].access(x),tree[1].splay(x);
    			a[x]=y;
    			tree[0].pushup(x),tree[1].pushup(x);
    		}
    		else{
    			write(tree[val[x]].query(x)),putchar('
    ');
    		}
    	}
    	return 0;
    }
    

    【模板】动态树

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=300003;
    template<typename tp>
    void read(tp& x){
        x=0;
        char c=getchar();
        bool sgn=0;
        while((c<'0'||c>'9')&&c!='-')c=getchar();
        if(c=='-')sgn=1,c=getchar();
        while(c>='0'&&c<='9')x=(x<<3)+(x<<1)+c-'0',c=getchar();
        if(sgn)x=-x;
    }
    template<typename tp>
    void write(tp x){
        if(x<0)putchar('-'),write(-x);
        else{
            if(x>=10)write(x/10);
            putchar(x%10+'0');
        }
    }
    
    int n,a[maxn];
    
    struct node{
        int sum,fa,son[2];
        bool z;
    }t[maxn];
    bool isroot(int p){return t[t[p].fa].son[0]!=p&&t[t[p].fa].son[1]!=p;}
    bool chk(int p){return t[t[p].fa].son[1]==p;}
    void reverse(int p){if(p==0)return;swap(t[p].son[0],t[p].son[1]);t[p].z^=1;}
    void pushup(int p){
        t[p].sum=t[t[p].son[0]].sum^t[t[p].son[1]].sum^a[p];
    }
    void pushdown(int p){
        if(t[p].z){
            reverse(t[p].son[0]);
            reverse(t[p].son[1]);
            t[p].z=0;
        }
    }
    void pushall(int p){if(!isroot(p))pushall(t[p].fa);pushdown(p);}
    void rotate(int p){
        int q=t[p].fa,r=t[q].fa,k=chk(p),s=t[p].son[!k];
        t[q].son[k]=s,t[s].fa=q;
        if(!isroot(q))t[r].son[chk(q)]=p;
        t[p].fa=r;
        t[p].son[!k]=q,t[q].fa=p;
        pushup(q);
        pushup(p);
    }
    void splay(int p){
        pushall(p);
        while(!isroot(p)){
            int q=t[p].fa;
            if(!isroot(q)){
                if(chk(p)==chk(q))rotate(q);
                else rotate(p);
            }
            rotate(p);
        }
    }
    
    void access(int x){for(int y=0;x;y=x,x=t[x].fa)splay(x),t[x].son[1]=y,pushup(x);}
    void change(int x,int k){splay(x),a[x]=k,pushup(x);}
    void makeroot(int x){access(x),splay(x),reverse(x);}
    int findroot(int x){
        access(x),splay(x);
        while(t[x].son[0])pushdown(x),x=t[x].son[0];
        splay(x);
        return x;
    }
    int query(int x,int y){makeroot(x),access(y),splay(y);return t[y].sum;}
    bool link(int x,int y){
        makeroot(x);
        if(findroot(y)==x)return 0;
        t[x].fa=y;
        return 1;
    }
    bool cut(int x,int y){
        makeroot(x);
        if(findroot(y)!=x||t[y].fa!=x||t[y].son[0])return 0;
        t[y].fa=t[x].son[1]=0;
        pushup(x);
        return 1;
    }
    
    int main(){
        int Q;
        read(n),read(Q);
        for(int i=1;i<=n;i++)read(a[i]),t[i].sum=a[i];
        while(Q--){
            int mo,x,y;
            read(mo),read(x),read(y);
            if(mo==0){
                write(query(x,y)),putchar('
    ');
            }
            else if(mo==1){
                link(x,y);
            }
            else if(mo==2){
                cut(x,y);
            }
            else{
                change(x,y);
            }
        }
        return 0;
    }
    

    单调队列

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=1000003;
    int a[maxn],n,k,q[maxn],qhead,qtail;
    int main(){
    	scanf("%d%d",&n,&k);
    	for(int i=1;i<=n;i++)scanf("%d",a+i);
    	qhead=1,qtail=0;
    	for(int i=1;i<=n;i++){
    		while(qhead<=qtail&&a[i]<=a[q[qtail]])qtail--;
    		q[++qtail]=i;
    		while(qhead<=qtail&&i-q[qhead]>=k)qhead++;
    		if(i>=k)printf("%d ",a[q[qhead]]);
    	}
    	puts("");
    	qhead=1,qtail=0;
    	for(int i=1;i<=n;i++){
    		while(qhead<=qtail&&a[i]>=a[q[qtail]])qtail--;
    		q[++qtail]=i;
    		while(qhead<=qtail&&i-q[qhead]>=k)qhead++;
    		if(i>=k)printf("%d ",a[q[qhead]]);
    	}
    	return 0;
    }
    

    Heap

    template<typename tp>
    class heap{
    	tp h[1000002];
        int cnt;
    	public:
    	heap():cnt(0){}
    	void push(const tp &x){
    		h[++cnt]=x;
    		int p=cnt;
    		while(p>=2&&h[p>>1]>h[p]){
    			swap(h[p>>1],h[p]);
    			p>>=1;
    		}
    	}
    	void pop(){
    		swap(h[1],h[cnt]);
    		cnt--;
    		int p=1;
    		while((p<<1)<=cnt){
    			p<<=1;
    			if(p<cnt&&h[p|1]<h[p])p|=1;
    			if(h[p]<h[p>>1])swap(h[p],h[p>>1]);
    			else break;
    		}
    	}
    	const tp &top(){
    		return h[1];
    	}
    };
    

    单调栈

    for(int i=1;i<=n;i++){
        int top=0;
        for(int j=1;j<=n;j++){
            if(b[i][j])a[j]=i;
            int tot=1;
            while(top&&a[j]>=a[stk[top]]){
                D tmp=at(a[stk[top]]+1,stk[top]-cnt[top]+1,i,j-1);
                if(tmp>ans){
                    ans=tmp;
                    x1=a[stk[top]]+1,y1=stk[top]-cnt[top]+1,x2=i,y2=j-1;
                }
                tot+=cnt[top];
                top--;
            }
            stk[++top]=j;
            cnt[top]=tot;
        }
        while(top){
            D tmp=at(a[stk[top]]+1,stk[top]-cnt[top]+1,i,n);
            if(tmp>ans){
                ans=tmp;
                x1=a[stk[top]]+1,y1=stk[top]-cnt[top]+1,x2=i,y2=n;
            }
            top--;
        }
    }
    

    左偏树

    int new_node(int val){
    	t[++cnt].val=val;
    	return cnt;
    }
    void merge(int &p,int x,int y){
    	if(x==0||y==0){p=x+y;return;}
    	if(a[t[x].val]!=a[t[y].val]?a[t[x].val]>a[t[y].val]:t[x].val>t[y].val)swap(x,y);
    	p=x;
    	merge(R(p),R(x),y);
    	if(dis[L(p)]<dis[R(p)])swap(L(p),R(p));
    	dis[p]=dis[R(p)]+1;
    }
    void pop(int &p){
    	merge(p,L(p),R(p));
    }
    

    DSU

    int f[maxn];
    int find(int x){return x!=f[x]?f[x]=find(f[x]):x;}
    void Union(int x,int y){int fx=find(x),fy=find(y);if(fx!=fy)f[fx]=fy;}
    bool same(int x,int y){return find(x)==find(y);}
    

    ST algorithm

    void init(){
    	for(int i=1;i<=n;i++)lg[i]=lg[i>>1]+1;
    	for(int len=1;(1<<len)<=n;len++){
    		for(int i=1;i+(1<<len)-1<=n;i++){
    			st[i][len]=max(st[i][len-1],st[i+(1<<(len-1))][len-1]);
    		}
    	}
    }
    int query(int l,int r){
    	int tmp=lg[r-l+1];
    	return max(st[l][tmp],st[r-(1<<tmp)+1]fe[tmp]);
    }
    

    Fenwick tree

    int t[maxn];
    void add(int pos,int k){while(pos<=n)t[pos]+=k,pos+=pos&-pos;}
    int query(int pos){int ret=0;while(pos)ret+=t[pos],pos-=pos&-pos;return ret;}
    

    Segment tree

    merge

    void merge(int &p,int q){
    	if(!p||!q){p|=q;return;}
        cnt++;
        sz[cnt]=sz[p]+sz[q];
        p=cnt;
    	merge(L[p],L[q]);
    	merge(R[p],R[q]);
    }
    

    树套树

    #include<bits/stdc++.h>
    #define L(p) t[p].son[0]
    #define R(p) t[p].son[1]
    using namespace std;
    const int maxn=100005,maxv=100000000,INF=2147483647;
    struct treap_node{
    	int val,son[2],sz;
    }t[4000005];
    int cnt,n,a[maxn];
    struct TREAP{
    	int root;
    	TREAP():root(0){}
    	void O(int p){
    		if(p==0)return;
    		O(L(p));
    		printf("%d ",t[p].val);
    		O(R(p));
    	}
    	void DEBUG(){
    		printf("###### DEBUG root:%d ######
    ",root);
    		O(root);
    		printf("
    ######  ######
    ");
    	}
    	void pushup(int p){t[p].sz=t[L(p)].sz+t[R(p)].sz+1;}
    	void split(int p,int k,int &x,int &y){
    		if(p==0){x=y=0;return;}
    		if(t[p].val<=k){
    			x=p;
    			split(R(p),k,R(p),y);
    		}
    		else{
    			y=p;
    			split(L(p),k,x,L(p));
    		}
    		pushup(p);
    	}
    	void merge(int &p,int x,int y){
    		if(x==0||y==0){p=x+y;return;}
    		if(rand()%(t[x].sz+t[y].sz)<t[x].sz){
    			merge(R(x),R(x),y);
    			pushup(x);
    			p=x;
    		}
    		else{
    			merge(L(y),x,L(y));
    			pushup(y);
    			p=y;
    		}
    	}
    	int new_node(int k){
    		cnt++;
    		t[cnt].val=k;
    		t[cnt].sz=1;
    		return cnt;
    	}
    	void insert(int k){
    		int p,x,y;
    		split(root,k,x,y);
    		p=new_node(k);
    		merge(p,x,p);
    		merge(root,p,y);
    	}
    	void erase(int k){
    		int x,y,z;
    		split(root,k,x,y);
    		split(x,k-1,z,x);
    		merge(z,z,L(x));
    		merge(y,R(x),y);
    		merge(root,z,y);
    	}
    	int rnk(int k){
    		int x,y;
    		split(root,k-1,x,y);
    		int ret=t[x].sz+1;
    		merge(root,x,y);
    		return ret;
    	}
    	int kth(int p,int k){
    		while(1){
    			if(k<=t[L(p)].sz)p=L(p);
    			else{
    				k-=(t[L(p)].sz+1);
    				if(k<=0)return p;
    				p=R(p);
    			}
    		}
    	}
    	int pre(int k){
    		int x,y;
    		split(root,k-1,x,y);
    		if(t[x].sz==0)return -INF;
    		int ret=kth(x,t[x].sz);
    		merge(root,x,y);
    		return t[ret].val;
    	}
    	int nxt(int k){
    		int x,y;
    		split(root,k,x,y);
    		if(t[y].sz==0)return INF;
    		int ret=kth(y,1);
    		merge(root,x,y);
    		return t[ret].val;
    	}
    };
    struct SEG{
    	struct seg_node{
    		TREAP tr;
    	}t[maxn<<2];
    	void build(int p,int l,int r){
    		for(int i=l;i<=r;i++)t[p].tr.insert(a[i]);
    		if(l==r)return;
    		int mid=(l+r)>>1;
    		build(p<<1,l,mid);
    		build(p<<1|1,mid+1,r);
    	}
    	void change(int p,int l,int r,int pos,int k){
    		t[p].tr.erase(a[pos]);
    		t[p].tr.insert(k);
    		if(l==r)return;
    		int mid=(l+r)>>1;
    		if(pos<=mid)change(p<<1,l,mid,pos,k);
    		else change(p<<1|1,mid+1,r,pos,k);
    	}
    	int _rnk(int p,int l,int r,int seg_l,int seg_r,int k){
    		if(seg_l<=l&&r<=seg_r)return t[p].tr.rnk(k)-1;
    		int mid=(l+r)>>1,ret=0;
    		if(seg_l<=mid)ret+=_rnk(p<<1,l,mid,seg_l,seg_r,k);
    		if(seg_r>mid)ret+=_rnk(p<<1|1,mid+1,r,seg_l,seg_r,k);
    		return ret;
    	}
    	int rnk(int seg_l,int seg_r,int k){
    		return _rnk(1,1,n,seg_l,seg_r,k)+1;
    	}
    	int kth(int seg_l,int seg_r,int k){
    		int l=0,r=maxv,mid,ans=-1;
    		while(l<=r){
    			mid=(l+r)>>1;
    			if(rnk(seg_l,seg_r,mid)<=k)ans=mid,l=mid+1;
    			else r=mid-1;
    		}
    		return ans;
    	}
    	int pre(int p,int l,int r,int seg_l,int seg_r,int k){
    		if(seg_l<=l&&r<=seg_r)return t[p].tr.pre(k);
    		int mid=(l+r)>>1,ret=-INF;
    		if(seg_l<=mid)ret=max(ret,pre(p<<1,l,mid,seg_l,seg_r,k));
    		if(seg_r>mid)ret=max(ret,pre(p<<1|1,mid+1,r,seg_l,seg_r,k));
    		return ret;
    	}
    	int nxt(int p,int l,int r,int seg_l,int seg_r,int k){
    		if(seg_l<=l&&r<=seg_r)return t[p].tr.nxt(k);
    		int mid=(l+r)>>1,ret=INF;
    		if(seg_l<=mid)ret=min(ret,nxt(p<<1,l,mid,seg_l,seg_r,k));
    		if(seg_r>mid)ret=min(ret,nxt(p<<1|1,mid+1,r,seg_l,seg_r,k));
    		return ret;
    	}
    }seg;
    int main(){
    	srand(19260817);
    	int Q,mo,x,y,z;
    	scanf("%d%d",&n,&Q);
    	for(int i=1;i<=n;i++)scanf("%d",a+i);
    	seg.build(1,1,n);
    	while(Q--){
    		scanf("%d%d%d",&mo,&x,&y);
    		switch(mo){
    			case 1:scanf("%d",&z),printf("%d
    ",seg.rnk(x,y,z));break;
    			case 2:scanf("%d",&z),printf("%d
    ",seg.kth(x,y,z));break;
    			case 3:seg.change(1,1,n,x,y),a[x]=y;break;
    			case 4:scanf("%d",&z),printf("%d
    ",seg.pre(1,1,n,x,y,z));break;
    			case 5:scanf("%d",&z),printf("%d
    ",seg.nxt(1,1,n,x,y,z));break;
    		}
    	}
    	return 0;
    }
    

    Treap

    维护权值

    #include<bits/stdc++.h>
    #define maxn 100003
    using namespace std;
    struct node{
    	int val,son[2],sz;
    }t[maxn];
    int cnt,root;
    void pushup(int p){t[p].sz=t[t[p].son[0]].sz+t[t[p].son[1]].sz+1;}
    void split(int p,int k,int &x,int &y){
    	if(p==0){x=y=0;return;}
    	if(t[p].val<=k){
    		x=p;
    		split(t[p].son[1],k,t[p].son[1],y);
    	}
    	else{
    		y=p;
    		split(t[p].son[0],k,x,t[p].son[0]);
    	}
    	pushup(p);
    }
    void merge(int &p,int x,int y){
    	if(x==0||y==0){p=x+y;return;}
    	if(rand()%(t[x].sz+t[y].sz)<t[x].sz){
    		merge(t[x].son[1],t[x].son[1],y);
    		pushup(x);
    		p=x;
    	}
    	else{
    		merge(t[y].son[0],x,t[y].son[0]);
    		pushup(y);
    		p=y;
    	}
    }
    int new_node(int k){
    	cnt++;
    	t[cnt].val=k;
    	t[cnt].sz=1;
    	return cnt;
    }
    void insert(int k){
    	int p,x,y;
    	split(root,k,x,y);
    	p=new_node(k);
    	merge(p,x,p);
    	merge(root,p,y);
    }
    void erase(int k){
    	int x,y,z;
    	split(root,k,x,y);
    	split(x,k-1,z,x);
    	merge(z,z,t[x].son[0]);
    	merge(y,t[x].son[1],y);
    	merge(root,z,y);
    }
    int rnk(int k){
    	int x,y;
    	split(root,k-1,x,y);
    	int ret=t[x].sz+1;
    	merge(root,x,y);
    	return ret;
    }
    int kth(int p,int k){
    	while(1){
    		if(k<=t[t[p].son[0]].sz)p=t[p].son[0];
    		else{
    			k-=(t[t[p].son[0]].sz+1);
    			if(k<=0)return p;
    			p=t[p].son[1];
    		}
    	}
    }
    int pre(int k){
    	int x,y;
    	split(root,k-1,x,y);
    	int ret=kth(x,t[x].sz);
    	merge(root,x,y);
    	return ret;
    }
    int nxt(int k){
    	int x,y;
    	split(root,k,x,y);
    	int ret=kth(y,1);
    	merge(root,x,y);
    	return ret;
    }
    int main(){
    	srand(19260817);
    	int Q;
    	scanf("%d",&Q);
    	while(Q--){
    		int mo,x;
    		scanf("%d%d",&mo,&x);
    		switch(mo){
    			case 1:insert(x);break;
    			case 2:erase(x);break;
    			case 3:printf("%d
    ",rnk(x));break;
    			case 4:printf("%d
    ",t[kth(root,x)].val);break;
    			case 5:printf("%d
    ",t[pre(x)].val);break;
    			case 6:printf("%d
    ",t[nxt(x)].val);break;
    		}
    	}
    	return 0;
    }
    

    维护序列

    #include<bits/stdc++.h>
    #define L(p) t[p].son0
    #define R(p) t[p].son1
    using namespace std;
    const int maxn=100003;
    struct node{
    	int son0,son1,sz;
    	bool z;
    }t[maxn];
    int cnt,root;
    void pushup(int p){t[p].sz=t[L(p)].sz+t[R(p)].sz+1;}
    void solve(int p){if(p)swap(L(p),R(p)),t[p].z^=1;}
    void pushdown(int p){
    	if(t[p].z){
    		solve(L(p)),solve(R(p));
    		t[p].z=0;
    	}
    }
    void output(int p){
    	if(p==0)return;
    	pushdown(p);
    	output(L(p));
    	printf("%d ",p);
    	output(R(p));
    }
    void O(int p=root){
    	output(p);
    	puts("");
    }
    void split(int p,int k,int &x,int &y){
    	if(p==0){x=y=0;return;}
    	pushdown(p);
    	if(t[L(p)].sz+1<=k){
    		x=p;
    		split(R(p),k-t[L(p)].sz-1,R(p),y);
    	}
    	else{
    		y=p;
    		split(L(p),k,x,L(p));
    	}
    	pushup(p);
    }
    void merge(int &p,int x,int y){
    	if(x==0||y==0){p=x+y;return;}
    	if(rand()%(t[x].sz+t[y].sz)<t[x].sz){
    		pushdown(x);
    		merge(R(x),R(x),y);
    		pushup(x);
    		p=x;
    	}
    	else{
    		pushdown(y);
    		merge(L(y),x,L(y));
    		pushup(y);
    		p=y;
    	}
    }
    void build(int &p,int l,int r){
    	if(l>r)return;
    	p=(l+r)>>1;
    	build(L(p),l,p-1);
    	build(R(p),p+1,r);
    	pushup(p);
    }
    void reverse(int l,int r){
    	int x,y,z;
    	split(root,l-1,x,y);
    	split(y,r-l+1,y,z);
    	solve(y);
    	merge(y,x,y);
    	merge(root,y,z);
    }
    int main(){
    	srand(19260817);
    	int n,Q;
    	scanf("%d%d",&n,&Q);
    	build(root,1,n);
    	while(Q--){
    		int x,y;
    		scanf("%d%d",&x,&y);
    		reverse(x,y);
    	}
    	O();
    	return 0;
    }
    

    persistent

    #include<bits/stdc++.h>
    #define L(p) t[p].son[0]
    #define R(p) t[p].son[1]
    using namespace std;
    namespace FASTIO{...}
    const int maxn=200003;
    struct node{
    	int val,son[2],sz;
    	long long sum;
    	bool z;
    }t[20000003];
    int cnt,root[maxn];
    int new_node(int k){
    	cnt++;
    	t[cnt].val=t[cnt].sum=k;
    	t[cnt].sz=1;
    	L(cnt)=R(cnt)=0;
    	return cnt;
    }
    void pushup(int p){
    	t[p].sz=t[L(p)].sz+t[R(p)].sz+1;
    	t[p].sum=t[L(p)].sum+t[R(p)].sum+t[p].val;
    }
    void pushdown(int p){
    	if(t[p].z){
    		int l=0,r=0;
    		if(L(p))t[r=++cnt]=t[L(p)],t[r].z^=1;
    		if(R(p))t[l=++cnt]=t[R(p)],t[l].z^=1;
    		L(p)=l,R(p)=r;
    		t[p].z=0;
    	}
    }
    void output(int p){
    	if(p==0)return;
    	pushdown(p);
    	output(L(p));
    	printf("val:%d sum:%lld son0:%d son1:%d
    ",t[p].val,t[p].sum,t[L(p)].val,t[R(p)].val);
    	output(R(p));
    }
    void O(int p){
    	printf("### O ###
    ");
    	puts("############");
    	output(p);
    	puts("############");
    }
    void split(int p,int k,int &x,int &y){
    	if(p==0){x=y=0;return;}
    	pushdown(p);
    	if(t[L(p)].sz+1<=k){
    		x=++cnt;
    		t[x]=t[p];
    		split(R(p),k-t[L(p)].sz-1,R(x),y);
    		pushup(x);
    	}
    	else{
    		y=++cnt;
    		t[y]=t[p];
    		split(L(p),k,x,L(y));
    		pushup(y);
    	}
    }
    void merge(int &p,int x,int y){
    	if(x==0||y==0){p=x+y;return;}
    	if(rand()%(t[x].sz+t[y].sz)<t[x].sz){
    		pushdown(x);
    		merge(R(x),R(x),y);
    		pushup(x);
    		p=x;
    	}
    	else{
    		pushdown(y);
    		merge(L(y),x,L(y));
    		pushup(y);
    		p=y;
    	}
    }
    void insert(int u,int v,int pos,int k){
    	int x,y;
    	split(root[v],pos,x,y);
    	merge(x,x,new_node(k));
    	merge(root[u],x,y);
    }
    void erase(int u,int v,int pos){
    	int x,y,z;
    	split(root[v],pos-1,x,y);
    	split(y,1,y,z);
    	merge(root[u],x,z);
    }
    void reverse(int u,int v,int l,int r){
    	int x,y,z;
    	split(root[v],l-1,x,y);
    	split(y,r-l+1,y,z);
    	t[y].z^=1;
    	merge(y,x,y);
    	merge(root[u],y,z);
    }
    long long query(int v,int l,int r){
    	int x,y,z;
    	split(root[v],l-1,x,y);
    	split(y,r-l+1,y,z);
    	long long ret=t[y].sum;
    	merge(y,x,y);
    	merge(root[v],y,z);
    	return ret;
    }
    int main(){
    	srand(19260817);
    	int Q,v,mo;
    	long long x,y,last=0;
    	read(Q);
    	for(int i=1;i<=Q;i++){
    		read(v),read(mo),read(x);
    		root[i]=root[v];
    		x^=last;
    		if(mo==1){
    			read(y);
    			y^=last;
    			insert(i,v,x,y);
    		}
    		if(mo==2){
    			erase(i,v,x);
    		}
    		if(mo==3){
    			read(y);
    			y^=last;
    			reverse(i,v,x,y);
    		}
    		if(mo==4){
    			read(y);
    			y^=last;
    			write(last=query(v,x,y)),pc('
    ');
    		}
    	}
    	flush();
    	return 0;
    }
    

    String

    KMP

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=1000003;
    int kmp[maxn];
    char s1[maxn],s2[maxn];
    int main(){
    	scanf("%s%s",s1+1,s2+1);
    	int l1=strlen(s1+1),l2=strlen(s2+1),pos=0;
    	for(int i=2;i<=l2;i++){
    		while(pos&&s2[pos+1]!=s2[i])pos=kmp[pos];
    		if(s2[pos+1]==s2[i])pos++;
    		kmp[i]=pos;
    	}
    	pos=0;
    	for(int i=1;i<=l1;i++){
    		while(pos&&s1[i]!=s2[pos+1])pos=kmp[pos];
    		if(s1[i]==s2[pos+1])pos++;
    		if(pos==l2){
    			printf("%d
    ",i-l2+1);
    			pos=kmp[pos];
    		}
    	}
    	for(int i=1;i<=l2;i++)printf("%d ",kmp[i]);
    	return 0;
    }
    

    Hash

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=10003;
    char s[maxn];
    int n,ha[maxn];
    int hash(){
    	static const int p1=15797,p2=1000000009;
    	int ret=0;
    	for(int i=0;s[i];i++){
    		ret=(1ll*ret*p1+s[i])%p2;
    	}
    	return ret;
    }
    int main(){
    	scanf("%d",&n);
    	for(int i=1;i<=n;i++){
    		scanf("%s",s);
    		ha[i]=hash();
    	}
    	sort(ha+1,ha+n+1);
    	printf("%d",int(unique(ha+1,ha+n+1)-ha-1));
    	return 0;
    }
    

    Suffix array

    #include<bits/stdc++.h>
    #define maxn 1000003
    #define mod 1000000007
    #define O(a) printf(#a": ");for(int j=1;j<=n;j++)printf("%d ",a[j]);puts("")
    #define DEBUG printf("%d:
    ",(int)__LINE__);O(sa);O(rnk);O(tp);O(buc);O(height)
    using namespace std;
    char s[maxn];
    int rnk[maxn],sa[maxn],tp[maxn],buc[maxn],n,m,height[maxn],a[maxn],f[maxn];
    long long sum[maxn];
    void bucsort(){
    	for(int i=1;i<=m;i++)buc[i]=0;
    	for(int i=1;i<=n;i++)buc[rnk[i]]++;
    	for(int i=1;i<=m;i++)buc[i]+=buc[i-1];
    	for(int i=n;i>=1;i--)sa[buc[rnk[tp[i]]]--]=tp[i];
    }
    void sufsort(){
    	m=128;
    	for(int i=1;i<=n;i++)rnk[i]=s[i],tp[i]=i;
    	bucsort();
    	for(int t=1,cnt=0;cnt<n;m=cnt,t<<=1){
    //update tp
    		cnt=0;
    		for(int i=n-t+1;i<=n;i++)tp[++cnt]=i;
    		for(int i=1;i<=n;i++){
    			if(sa[i]>t)tp[++cnt]=sa[i]-t;
    		}
    //update sa
    		bucsort();
    //update rnk
    		for(int i=1;i<=n;i++)tp[i]=rnk[i];
    		rnk[sa[1]]=cnt=1;
    		for(int i=2;i<=n;i++){
    			rnk[sa[i]]=(tp[sa[i]]==tp[sa[i-1]]&&tp[sa[i]+t]==tp[sa[i-1]+t]?cnt:++cnt);
    		}
    	}
    }
    void geth(){
    	int k=0;
    	for(int i=1;i<=n;i++)rnk[sa[i]]=i;
    	for(int i=1;i<=n;i++){
    		if(rnk[i]==1)continue;
    		if(k)k--;
    		int j=sa[rnk[i]-1];
    		while(i+k<=n&&j+k<=n&&s[i+k]==s[j+k])k++;
    		height[rnk[i]]=k;
    	}
    }
    bool cmp(int x,int y){return height[x]>height[y];}
    int find(int x){return x!=f[x]?f[x]=find(f[x]):f[x];}
    int main(){
    	scanf("%s",s+1);
    	n=strlen(s+1);
    	sufsort(),geth();
    	for(int i=1;i<=n;i++)a[i]=f[i]=i,sum[i]=1;
    	sort(a+1,a+n+1,cmp);
    	long long ans=0;
    	for(int i=n,j=1;i>=1;i--){
    		for(;j<=n&&height[a[j]]==i;j++){
    			int x=find(sa[a[j]-1]),y=find(sa[a[j]]);
    			sum[x]+=sum[y];
    			ans=max(ans,(long long)i*sum[x]);
    			f[y]=x;
    		}
    	}
    	printf("%lld
    ",ans);
    	return 0;
    }
    

    Manacher

    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=22000005;
    char t[maxn>>1],s[maxn];
    int p[maxn];
    int main(){
    	scanf("%s",t+1);
    	int N=strlen(t+1),n=0,mid=0,r=0,ans=1;
    	s[++n]=-2,s[++n]=-1;
    	for(int i=1;i<=N;i++)s[++n]=t[i],s[++n]=-1;
    	for(int i=1;i<=n;i++){
    		if(i<=r)p[i]=min(p[(mid<<1)-i],r-i+1);
    		while(s[i+p[i]]==s[i-p[i]])p[i]++;
    		if(i+p[i]-1>r)r=i+p[i]-1,mid=i;
    		ans=max(ans,p[i]);
    	}
    	printf("%d
    ",ans-1);
    	return 0;
    }
    

    Geometry

    const double eps=1e-10,INF=1e17;
    typedef double tp;
    int sgn(tp x){return x<-eps?-1:x>eps;}
    // const double pi=acos(-1);
    struct point{
    	tp x,y;
    	point(){}
    	point(tp _x,tp _y):x(_x),y(_y){}
    	void write()const{printf("%.4lf %.4lf
    ",x,y);}
    	bool operator<(point p)const{return sgn(x-p.x)==0?sgn(y-p.y)<0:sgn(x-p.x)<0;}
    	bool operator==(point p)const{return sgn(x-p.x)==0&&sgn(y-p.y)==0;}
    	point operator+(point p)const{return point(x+p.x,y+p.y);}
    	point operator-(point p)const{return point(x-p.x,y-p.y);}
    	point operator*(tp k)const{return point(x*k,y*k);}
    	double norm()const{return x*x+y*y;}
    	// double abs()const{return sqrt(x*x+y*y);}
    };
    tp dot(point u,point v){return u.x*v.x+u.y*v.y;}
    tp cross(point u,point v){return u.x*v.y-v.x*u.y;}
    double dist(point p1,point p2){return sqrt(double(dot(p1-p2,p1-p2)));}
    
    point inter(point P,point v,point Q,point w){
    	point u=P-Q;
    	if(sgn(cross(v,w))==0){
    		if(sgn(cross(Q-P,v))==0)return point(-INF,-INF); //chong he
    		return point(INF,INF); //ping xing
    	}
    	tp t=cross(w,u)/cross(v,w);
    	return P+v*t; //OK
    }
    
    bool seg_inter(point p1,point p2,point p3,point p4){
    	return min(p1.x,p2.x)<=max(p3.x,p4.x)&&
    	min(p3.x,p4.x)<=max(p1.x,p2.x)&&
    	min(p1.y,p2.y)<=max(p3.y,p4.y)&&
    	min(p3.y,p4.y)<=max(p1.y,p2.y)&&
    	sgn(cross(p2-p1,p3-p1))*sgn(cross(p2-p1,p4-p1))<=0&&
    	sgn(cross(p4-p3,p1-p3))*sgn(cross(p4-p3,p2-p3))<=0;
    }
    bool line_seg_inter(point l1,point l2,point p1,point p2){
    	return sgn(cross(l2-l1,p1-l1))*sgn(cross(l2-l1,p2-l1))<=0;
    }
    
    int stk[maxn],cntcon;
    point con[maxn];
    void convex_hull(point a[],int n){
    	cntcon=0;
    	int top=2;
    	stk[1]=1,stk[2]=2;
    	for(int i=3;i<=n;i++){
    		while(top>1&&cross(a[i]-a[stk[top]],a[stk[top]]-a[stk[top-1]])>0)top--;
    		stk[++top]=i;
    	}
    	for(int i=1;i<top;i++){
    		con[++cntcon]=a[stk[i]];
    	}
    	top=2;
    	stk[1]=1,stk[2]=2;
    	for(int i=3;i<=n;i++){
    		while(top>1&&cross(a[i]-a[stk[top]],a[stk[top]]-a[stk[top-1]])<=0)top--;
    		stk[++top]=i;
    	}
    	for(int i=top;i>1;i--){
    		con[++cntcon]=a[stk[i]];
    	}
    }
    
    double getS(const vector<point> &p){
    	double ans=cross(p[0],p[p.size()-1]);
    	for(int i=1;i<int(p.size());i++)ans+=cross(p[i],p[i-1]);
    	return abs(ans)*0.5;
    }
    
    bool onright(point P,point S,point T){
    	return sgn(cross(T-S,P-S))<0;
    }
    

    Math

    Number theory

    Combinatorics

    Gauss

    #include<cstdio>
    #include<algorithm>
    using namespace std;
    const int maxn=103;
    const double eps=1e-8;
    int n;
    double a[maxn][maxn];
    int sgn(double x){return x<-eps?-1:x>eps;}
    bool gauss(){
        for(int i=1;i<=n;i++){
            int p=i;
            for(int j=i+1;j<=n;j++)if(abs(a[j][i])>abs(a[p][i]))p=j;
            if(p!=i)for(int j=1;j<=n+1;j++)swap(a[i][j],a[p][j]);
            if(sgn(a[i][i])==0){
                puts("No Solution");
                return 0;
            }
            for(int j=i+1;j<=n+1;j++)a[i][j]/=a[i][i];
            a[i][i]=1;
            for(int j=1;j<=n;j++){
                if(i!=j){
                    for(int k=i+1;k<=n+1;k++)a[j][k]-=a[j][i]*a[i][k];
                    a[j][i]=0;
                }
            }
        }
        return 1;
    }
    signed main(){
        scanf("%d",&n);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n+1;j++){
                scanf("%lf",&a[i][j]);
            }
        }
        if(!gauss())return 0;
        for(int i=1;i<=n;i++)printf("%.2lf
    ",a[i][n+1]);
        return 0;
    }
    

    Linear basis

    #include<bits/stdc++.h>
    using namespace std;
    const int maxlog=51;
    int n;
    long long p[maxlog];
    void insert(long long x){
    	for(int i=maxlog-1;i>=0;i--){
    		if((x>>i)&1){
    			if(p[i]==0){
    				p[i]=x;
    				break;
    			}
    			else x^=p[i];
    		}
    	}
    }
    long long query(){
    	long long ret=0;
    	for(int i=maxlog-1;i>=0;i--){
    		if((ret^p[i])>ret)ret^=p[i];
    	}
    	return ret;
    }
    int main(){
    	scanf("%d",&n);
    	for(int i=1;i<=n;i++){
    		long long x;
    		scanf("%lld",&x);
    		insert(x);
    	}
    	printf("%lld",query());
    	return 0;
    }
    

    BSGS

    unordered_map<int,int> mp;
    int bsgs(int y,int z){
    	if(z==1)return 0;
    	if(y%mod==0)return -1;
    	mp.clear();
    	int m=sqrt(mod)+1,ji=z;
    	for(int b=0;b<=m;b++){
    		mp[ji]=b;
    		ji=mul(ji,y);
    	}
    	int tmp=qpow(y,m);
    	ji=1;
    	for(int a=1;a<=m;a++){
    		ji=mul(ji,tmp);
    		if(mp.count(ji))return a*m-mp[ji];
    	}
    	return -1;
    }
    
  • 相关阅读:
    保险
    cron表达式的用法
    Hive 学习记录
    股票的五种估值方法
    AtCoder Beginner Contest 113 A
    ZOJ 4070 Function and Function
    银行业务队列简单模拟 (数据结构题目)
    算法3-7:银行排队
    算法3-5:n阶Hanoi塔问题
    算法3-1:八进制数
  • 原文地址:https://www.cnblogs.com/BlogOfchc1234567890/p/11663963.html
Copyright © 2011-2022 走看看