zoukankan      html  css  js  c++  java
  • 省选测试15

    总结

    这个分基本上和爆零没什么区别

    只会打无脑暴力,高分暴力也不会

    (T1) 写了一个随机化贪心拿了 (20)(T2) 水了一个 (prufer) 的分

    (T2) 的时候居然认为边双是有向图的概念,暴搜也没有打出来

    明天有一天整理的时间,要好好调整状态

    毕竟放假之前最多也只有三次考试了

    A. U.N.OWEN就是她吗?

    分析

    首先要了解什么是霍尔定理

    (M(U)) 为与 (U) 中的点相连的点集,一个二分图 (U,V(|U|<=|V|))存在完美匹配,满足对于任意点集 (x in U)都有 |(M(X)|>=|X|)

    因为要去枚举每一个子集,所以复杂度是 (3^n)

    但是可以证明在题目条件下,只需要对每个左边的连续区间验证霍尔定理

    考虑反证法,假如我们左边选了很多段连续区间,只考虑两段的情况,其余情况是类似的

    先把所有没有和左边点相连的右边点去掉,那么对于每个左边的连续区间,与其相邻的右边点也是一个连续区间

    那么我们分两种情况讨论:

    (1):左边两段在右边的对应点是两段区间
    如果左边两段分别验证霍尔定理都合法,那么左边这两段的并也一定合法

    (2):左边两段在右边的对应点是一段区间

    那么我们考虑把左边两段中间的点也选上,根据条件右边的点不会变多,这样的限制显然更紧

    所以没有必要选择左边不连续的多段验证霍尔定理

    只需要枚举 (n^2) 个区间就可以了

    先把所有询问按照左端点从小到大排序

    因为区间没有包含关系,那么排完序之后左右端点一定都是单调递增的,那么就可以用前缀和处理

    (a[i]) 为石子个数的前缀和 ,(b[i]) 为每一次操作中实际选择的石子的前缀和

    注意这里是实际选择的,而不是题目中要求选择的

    因为实际选择的一定要满足霍尔定理,因为你已经选了这些了

    但是题目中要求的你不一定全选

    对于没有被询问包含的石子,要把它们的个数置成 (0),否则会影响结果

    那么根据霍尔定理对于任意的 (i leq j) ,必定存在 (b_j-b_{i-1} leq a_{r[j]}-a_{l[i]-1})

    (b_j-a_{r[j]} leq b_{i-1}-a_{l[i]-1})

    此时左边和右边分别只有 (j)(i) ,可以分开考虑

    (s_i=b_i-a_{r[i]},t_i=b_{i-1}-a_{l[i]-1})

    即对于任意的 (i leq j),都有 (s_j leq t_i)

    将询问按照时间排序处理

    设当前询问在按照左端点排序时的排名为 (p)

    当前的答案就是区间 ([1,p])(t) 的最小值减去区间 ([p,m])(s) 的最大值

    还要和想要的石子个数取一个 (min)

    考虑当前答案会对 (s)(t) 数组造成什么影响

    对于 (s),要在区间 ([p,m]) 加上 (ans)

    对于 (t),要在区间 ([p+1,m]) 加上 (ans)

    对于左右端点都在 ([p+1,m])的区间,因为都加上了 (ans),所以减完之后关系不会改变

    对于左右端点都在 ([1,p-1])的区间,当前的答案不会影响到它们,所以也不用考虑

    我们要考虑的只是左端点在 ([1,p]) ,右端点在 ([p,m]) 的区间

    我们加上答案后不能违背霍尔定理,所以答案不能超过区间 ([1,p])(t) 的最小值减去区间 ([p,m])(s) 的最大值

    不要忘了更新 (s)(t) 数组

    这个东西拿线段树维护一下就行了

    代码

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<iostream>
    #include<cmath>
    #define rg register
    inline int read(){
    	rg int x=0,fh=1;
    	rg char ch=getchar();
    	while(ch<'0' || ch>'9'){
    		if(ch=='-') fh=-1;
    		ch=getchar();
    	}
    	while(ch>='0' && ch<='9'){
    		x=(x<<1)+(x<<3)+(ch^48);
    		ch=getchar();
    	}
    	return x*fh;
    }
    const int maxn=1e6+5;
    int n,m,sum[maxn],a[maxn];
    struct jie{
    	int l,r,k,id,rk;
    }b[maxn];
    bool cmp(rg jie aa,rg jie bb){
    	return aa.l<bb.l;
    }
    bool cmp2(rg jie aa,rg jie bb){
    	return aa.id<bb.id;
    }
    struct trr{
    	int l,r,lazt,lazs,maxs,mint,siz;
    }tr[maxn];
    void push_up(rg int da){
    	tr[da].maxs=std::max(tr[da<<1].maxs,tr[da<<1|1].maxs);
    	tr[da].mint=std::min(tr[da<<1].mint,tr[da<<1|1].mint);
    }
    void updats(rg int da,rg int val){
    	tr[da].lazs+=val;
    	tr[da].maxs+=val;
    }
    void updatt(rg int da,rg int val){
    	tr[da].lazt+=val;
    	tr[da].mint+=val;
    }
    void push_down(rg int da){
    	if(tr[da].lazs){
    		updats(da<<1,tr[da].lazs);
    		updats(da<<1|1,tr[da].lazs);
    		tr[da].lazs=0;
    	}
    	if(tr[da].lazt){
    		updatt(da<<1,tr[da].lazt);
    		updatt(da<<1|1,tr[da].lazt);
    		tr[da].lazt=0;
    	}
    }
    void build(rg int da,rg int l,rg int r){
    	tr[da].l=l,tr[da].r=r,tr[da].siz=r-l+1;
    	if(tr[da].l==tr[da].r){
    		tr[da].maxs=-sum[b[l].r];
    		tr[da].mint=-sum[b[l].l-1];
    		return;
    	}
    	rg int mids=(l+r)>>1;
    	build(da<<1,l,mids);
    	build(da<<1|1,mids+1,r);
    	push_up(da);
    }
    void xgt(rg int da,rg int l,rg int r,rg int val){
    	if(tr[da].l>=l && tr[da].r<=r){
    		updatt(da,val);
    		return;
    	}
    	push_down(da);
    	rg int mids=(tr[da].l+tr[da].r)>>1;
    	if(l<=mids) xgt(da<<1,l,r,val);
    	if(r>mids) xgt(da<<1|1,l,r,val);
    	push_up(da);
    }
    void xgs(rg int da,rg int l,rg int r,rg int val){
    	if(tr[da].l>=l && tr[da].r<=r){
    		updats(da,val);
    		return;
    	}
    	push_down(da);
    	rg int mids=(tr[da].l+tr[da].r)>>1;
    	if(l<=mids) xgs(da<<1,l,r,val);
    	if(r>mids) xgs(da<<1|1,l,r,val);
    	push_up(da);
    }
    int cxt(rg int da,rg int l,rg int r){
    	if(tr[da].l>=l && tr[da].r<=r) return tr[da].mint;
    	push_down(da);
    	rg int mids=(tr[da].l+tr[da].r)>>1,nans=0x3f3f3f3f;
    	if(l<=mids) nans=std::min(nans,cxt(da<<1,l,r));
    	if(r>mids) nans=std::min(nans,cxt(da<<1|1,l,r));
    	return nans;
    }
    int cxs(rg int da,rg int l,rg int r){
    	if(tr[da].l>=l && tr[da].r<=r) return tr[da].maxs;
    	push_down(da);
    	rg int mids=(tr[da].l+tr[da].r)>>1,nans=-0x3f3f3f3f;
    	if(l<=mids) nans=std::max(nans,cxs(da<<1,l,r));
    	if(r>mids) nans=std::max(nans,cxs(da<<1|1,l,r));
    	return nans;
    }
    int cf[maxn];
    int main(){
    	n=read();
    	for(rg int i=1;i<=n;i++) a[i]=read();
    	m=read();
    	for(rg int i=1;i<=m;i++){
    		b[i].l=read(),b[i].r=read(),b[i].k=read(),b[i].id=i;
    		cf[b[i].l]++,cf[b[i].r+1]--;
    	}
    	std::sort(b+1,b+1+m,cmp);
    	for(rg int i=1;i<=m;i++) b[i].rk=i;
    	for(rg int i=1;i<=n+1;i++) cf[i]+=cf[i-1];
    	for(rg int i=1;i<=n;i++) if(cf[i]==0) a[i]=0;
    	for(rg int i=1;i<=n;i++) sum[i]=sum[i-1]+a[i];
    	build(1,1,m);
    	std::sort(b+1,b+1+m,cmp2);
    	rg int tmp=0,wz;
    	for(rg int i=1;i<=m;i++){
    		wz=b[i].rk;
    		tmp=std::min(b[i].k,cxt(1,1,wz)-cxs(1,wz,m));
    		xgs(1,wz,m,tmp);
    		printf("%d
    ",tmp);
    		if(wz+1<=m) xgt(1,wz+1,m,tmp);
    	}
    	return 0;
    }
    

    B. 哈德曼的妖怪少女

    咕咕咕

    C. 平安时代的外星人

    分析

    首先,我们可以证明,答案的环会包含所有左上角到关键点左上角的最短路

    如图,绿色线是最短路,如果我们按照蓝色的区域去划分的话

    不如把黄色部分也划进来,这样答案不会更劣

    接着,我们把每个点按顺序拆成 (4) 个点 (0,1,2,3) ,每个点向顺时针的下一个点连边权为 (0) 的边

    两个跨过一条边的点要连边权为权值的边

    那么任何一个可以自交的环可以看做从左上角的 (1) 到左上角的 (3) 的一条路径

    图中红色部分代表最短路,由于我们答案路径要包含最短路所以((A 0 , A 3 ), (A 2 , A 3 ), (C 0 , C 1 ), (C 1 , C 2 ), (D 0 , D 3 ), (D 2 , D 3 ))之间不能连边

    当然这还不够,我们还需要保证我们的路径能够把所有的关键点包起来

    这就相当于是我们的路径不能经过关键点里面的 (4) 个点,在连边时去掉这些点即可

    就是图中红色虚线的部分

    注意左上角的 ((0,1),(0,3))不能连边

    代码

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<iostream>
    #include<cmath>
    #include<queue>
    #define rg register
    inline int read(){
    	rg int x=0,fh=1;
    	rg char ch=getchar();
    	while(ch<'0' || ch>'9'){
    		if(ch=='-') fh=-1;
    		ch=getchar();
    	}
    	while(ch>='0' && ch<='9'){
    		x=(x<<1)+(x<<3)+(ch^48);
    		ch=getchar();
    	}
    	return x*fh;
    }
    const int maxn=3e6+5,maxm=405;
    int val1[maxm][maxm],val2[maxm][maxm],a[maxm][maxm],rk[maxm][maxm],n,m,cnt,rkx[maxn],rky[maxn],jud[maxn];
    int getid(rg int i,rg int j,rg int id){
    	return id*(n+1)*(m+1)+(i-1)*(m+1)+j;
    }
    int h[maxn],tot=1;
    struct asd{
    	int from,to,nxt,val;
    }b[maxn];
    void ad(rg int aa,rg int bb,rg int cc){
    	b[tot].from=aa;
    	b[tot].to=bb;
    	b[tot].nxt=h[aa];
    	b[tot].val=cc;
    	h[aa]=tot++;
    }
    struct jie{
    	int num;
    	long long jl;
    	jie(){}
    	jie(rg int aa,rg long long bb){
    		num=aa,jl=bb;
    	}
    	bool operator < (const jie& A)const{
    		return jl>A.jl;
    	}
    };
    std::priority_queue<jie> q;
    int vis[maxn],pre[maxn];
    long long dis[maxn];
    void dij(rg int qd){
    	memset(dis,0x3f,sizeof(dis));
    	memset(pre,0,sizeof(pre));
    	memset(vis,0,sizeof(vis));
    	dis[qd]=0;
    	q.push(jie(qd,dis[qd]));
    	while(!q.empty()){
    		rg int now=q.top().num;
    		q.pop();
    		if(vis[now]) continue;
    		vis[now]=1;
    		for(rg int i=h[now];i!=-1;i=b[i].nxt){
    			rg int u=b[i].to;
    			if(dis[u]>dis[now]+b[i].val){
    				dis[u]=dis[now]+b[i].val;
    				pre[u]=i;
    				q.push(jie(u,dis[u]));
    			}
    		}
    	}
    }
    bool ban[maxn][4];
    void updat(rg int now){
    	while(now){
    		jud[pre[now]]=1;
    		now=b[pre[now]].from;
    	}
    }
    void gettag(rg int aa,rg int bb){
    	if(rkx[aa]==rkx[bb]){
    		if(rky[aa]>rky[bb]) std::swap(aa,bb);
    		ban[aa][1]=ban[bb][3]=1;
    	} else {
    		if(rkx[aa]>rkx[bb]) std::swap(aa,bb);
    		ban[aa][2]=ban[bb][0]=1;
    	}
    }
    int main(){
    	memset(h,-1,sizeof(h));
    	n=read(),m=read();
    	for(rg int i=1;i<=n;i++){
    		for(rg int j=1;j<=m;j++){
    			a[i][j]=read();
    		}
    	}
    	for(rg int i=1;i<=n+1;i++){
    		for(rg int j=1;j<=m+1;j++){
    			rk[i][j]=++cnt;
    			rkx[cnt]=i,rky[cnt]=j;
    		}
    	}
    	for(rg int i=1;i<=n;i++){
    		for(rg int j=1;j<=m+1;j++){
    			val1[i][j]=read();
    			ad(rk[i][j],rk[i+1][j],val1[i][j]),ad(rk[i+1][j],rk[i][j],val1[i][j]);
    		}
    	}
    	for(rg int i=1;i<=n+1;i++){
    		for(rg int j=1;j<=m;j++){
    			val2[i][j]=read();
    			ad(rk[i][j],rk[i][j+1],val2[i][j]),ad(rk[i][j+1],rk[i][j],val2[i][j]);
    		}
    	}
    	dij(rk[1][1]);
    	for(rg int i=1;i<=n;i++){
    		for(rg int j=1;j<=m;j++){
    			if(a[i][j]){
    				updat(rk[i][j]);
    				ban[rk[i][j]][1]=ban[rk[i][j]][2]=1;
    				ban[rk[i][j+1]][2]=ban[rk[i][j+1]][3]=1;
    				ban[rk[i+1][j]][0]=ban[rk[i+1][j]][1]=1;
    				ban[rk[i+1][j+1]][0]=ban[rk[i+1][j+1]][3]=1;
    			}
    		}
    	}
    	for(rg int i=1;i<tot;i++){
    		if(jud[i]) gettag(b[i].from,b[i].to);
    	}
    	tot=1;
    	memset(h,-1,sizeof(h));
    	ban[rk[1][1]][0]=ban[rk[1][1]][3]=1;
    	for(rg int i=1;i<=n+1;i++){
    		for(rg int j=1;j<=m+1;j++){
    			if(!ban[rk[i][j]][0]) ad(getid(i,j,0),getid(i,j,1),0),ad(getid(i,j,1),getid(i,j,0),0);
    			if(!ban[rk[i][j]][1]) ad(getid(i,j,1),getid(i,j,2),0),ad(getid(i,j,2),getid(i,j,1),0);
    			if(!ban[rk[i][j]][2]) ad(getid(i,j,2),getid(i,j,3),0),ad(getid(i,j,3),getid(i,j,2),0);
    			if(!ban[rk[i][j]][3]) ad(getid(i,j,3),getid(i,j,0),0),ad(getid(i,j,0),getid(i,j,3),0);
    		}
    	}
    	for(rg int i=1;i<=n;i++){
    		for(rg int j=1;j<=m+1;j++){
    			ad(getid(i,j,3),getid(i+1,j,0),val1[i][j]),ad(getid(i+1,j,0),getid(i,j,3),val1[i][j]);
    			ad(getid(i,j,2),getid(i+1,j,1),val1[i][j]),ad(getid(i+1,j,1),getid(i,j,2),val1[i][j]);
    		}
    	}
    	for(rg int i=1;i<=n+1;i++){
    		for(rg int j=1;j<=m;j++){
    			ad(getid(i,j,1),getid(i,j+1,0),val2[i][j]),ad(getid(i,j+1,0),getid(i,j,1),val2[i][j]);
    			ad(getid(i,j,2),getid(i,j+1,3),val2[i][j]),ad(getid(i,j+1,3),getid(i,j,2),val2[i][j]);
    		}
    	}
    	dij(getid(1,1,1));
    	printf("%lld
    ",dis[getid(1,1,3)]);
    	return 0;
    }
    
  • 相关阅读:
    IOS 改变UISearchBar的背景色
    电话正则表达式
    理解Callable 和 Spring DeferredResult(翻译)
    Java设计模式——观察者模式(事件监听)
    SpringMVC注解@initbinder解决类型转换问题
    内部系统间调用client包的封装方法
    Java中Comparable和Comparator实现对象比较
    linux上FTP服务器搭建
    百度地图——判断用户是否在配送范围内解决方案
    nodejs之express的使用
  • 原文地址:https://www.cnblogs.com/liuchanglc/p/14358828.html
Copyright © 2011-2022 走看看