zoukankan      html  css  js  c++  java
  • eJOI2017~2019 做题记录

    ISIJ 作业,所以来做了。

    打星号的是我能力范围内想出来的。([eJOI2018] 元素周期表很久以前见过,不清楚现在见到能不能做出来,所以不打星号)

    [eJOI2017] 魔法(*)

    每种字符出现次数相同,也就是每种字符的 (s_{r,c}-s_{l-1,c}) 相同。

    注意到,若 (forall i,s_{l-1,i}-s_{l-1,0}=s_{r,i}-s_{r,0}),那么 ([l,r]) 满足要求,反之也成立。

    那么就能随便做了。

    时间复杂度看实现,我写的垃圾 (O(n|Sigma|log n))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=100010,mod=1000000007;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,cnt,a[maxn],vis[128],pre[maxn][52],ans;
    map<vector<int>,int> mp;
    char s[maxn];
    int main(){
    	n=read();
    	scanf("%s",s+1);
    	FOR(i,1,n){
    		if(!vis[s[i]]) vis[s[i]]=++cnt;
    		a[i]=vis[s[i]];
    	}
    	FOR(i,1,n) a[i]--; 
    	FOR(i,1,n){
    		FOR(j,0,cnt-1) pre[i][j]=pre[i-1][j];
    		pre[i][a[i]]++;
    	}
    	FOR(i,0,n){
    		vector<int> v;
    		FOR(j,0,cnt-1) v.PB(pre[i][j]-pre[i][0]);
    		ans=(ans+mp[v])%mod;
    		mp[v]++;
    	}
    	printf("%d
    ",ans);
    }
    

    [eJOI2017] 六(*)

    被 AGC 虐爆后,我养成了自闭就搜的习惯。

    然后在这题用上了,就离谱(

    首先我们只用考虑每次加的数包含哪些质因子。设这个质因子集合为 (S)。那么要求要求任意时刻,(S) 的出现次数不超过 (2)

    那么设 (f_T)(T) 的第 (S) 位是 (0/1/2),表示质因子集合为 (S) 的数已经有了多少个。

    状态数 (O(3^{2^k})),没救了。

    但是这个状态是真的不满,直接搜搜记记。

    实现优秀点就能过了。

    顺便聊聊我从 2s+ 变成 0.2s 的过程:

    1. vector 改成两个 ll 压位。卡进 2s。
    2. 三进制改成四进制(要用 ull)。效果拔群。本地已经能 0.8s,然而洛谷太老爷了。
    3. 修改限制方式。本来我的写法是枚举加哪种数,然后判断限制,就有很多无用枚举。现在变成加数时就更新别的数的状态(能不能加)。这本来能过了,洛谷上 0.7s+,但后来时限改了。
    4. (tot_s)(s) 内的倍数的个数)用 (O(sqrt{n}k))。我也不知道当时怎么想的,改成 (O(sqrt{n}+2^kk)) 就洛谷上 0.3s+ 了,目前最优解。
    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef unsigned long long ull;
    typedef pair<ll,ll> PII;
    const int maxn=100010,mod=1000000007;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int k,tot[64],cnt[6];
    ll n,p[6];
    vector<int> v[64];
    map<PII,int> mp;
    inline int tbit(ull x,int p){
    	return (x>>(p<<1))&3;
    }
    int dfs(ull x,ull y){
    	if(mp.count(MP(x,y))) return mp[MP(x,y)];
    	int ans=1;
    	FOR(i,1,(1<<k)-1) if((i<32?tbit(x,i):tbit(y,i-32))<=1){
    		ll tmpx=x,tmpy=y;
    		bool flag=true;
    		FOR(jj,0,(int)v[i].size()-1){
    			int j=v[i][jj];
    			if(j<32){
    				if(tbit(x,j)<2) x+=1ull<<(j<<1);
    			}
    			else{
    				if(tbit(y,j-32)<2) y+=1ull<<((j-32)<<1);
    			}
    		}
    		if(flag) ans=(ans+1ll*dfs(x,y)*tot[i])%mod;
    		x=tmpx;y=tmpy;
    	}
    	return mp[MP(x,y)]=ans;
    }
    int main(){
    	n=read();
    	ll tmp=n;
    	for(int i=2;1ll*i*i<=n;i++) if(n%i==0){
    		p[k]=i;
    		while(n%i==0) n/=i,cnt[k]++;
    		k++;
    	}
    	if(n>1) p[k]=n,cnt[k++]=1;
    	n=tmp;
    	FOR(i,0,(1<<k)-1){
    		tot[i]=1;
    		FOR(j,0,k-1) if((i>>j)&1) tot[i]=1ll*tot[i]*cnt[j]%mod;
    	}
    	FOR(i,1,(1<<k)-1) FOR(j,1,(1<<k)-1) if(i&j) v[i].PB(j);
    	printf("%d
    ",(dfs(0,0)-1+mod)%mod);
    }
    

    [eJOI2017] 粒子(*)

    每次操作时二分,判断此时跑到最右的 (x) 粒子是否在跑到最左的 (y) 粒子右边。模拟 (k) 次即可。

    由于我相信 (O(nklog n)) 不能过,我们要接着优化。

    其实就是求个半平面交的事。每次重构半平面交,然后把两边的上半部分分段加起来,在上面判断。

    由于直线就那么几条,所以不用每次排序。

    时间复杂度 (O(nlog n+nk))

    然后当我发现评测记录里一大堆 0.9s+ 的时候心态就崩了。

    第二天我过来补代码,补着补着发现时限甚至变成了 2s,然后提交记录前几个 1.8s+。看起来就是暴力常数太大然后申请开大时限。

    然后开始无 能 狂 怒。是真的无能,由于当时是那一天,发不了犇犇和帖子,连博客都不能发。

    拿了个最优解不亏

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=50050,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    struct line{
    	int k,id1,id2;
    	ll b;
    	bool operator<(const line &l)const{
    		if(k!=l.k) return k<l.k;
    		return b>l.b;
    	}
    }lx[maxn],ly[maxn],hx[maxn],hy[maxn],h[maxn*2];
    int n,l,k,tx[maxn],vx[maxn],ty[maxn],vy[maxn],tpx,tpy,tot;
    inline double interx(line l1,line l2){
    	return 1.0*(l1.b-l2.b)/(l2.k-l1.k);
    }
    line merge(line l1,line l2){
    	return (line){l1.k+l2.k,l1.id1,l2.id1,l1.b+l2.b};
    } 
    int main(){
    	n=read();l=read();k=read();
    	FOR(i,1,n) tx[i]=read(),vx[i]=read(),lx[i]=(line){vx[i],i,0,-1ll*vx[i]*tx[i]};
    	FOR(i,1,n) ty[i]=read(),vy[i]=read(),ly[i]=(line){vy[i],i,0,-1ll*vy[i]*ty[i]};
    	sort(lx+1,lx+n+1);sort(ly+1,ly+n+1);
    	while(k--){
    		tpx=0;
    		FOR(i,1,n){
    			if(tpx && lx[i].k==hx[tpx].k) continue;
    			while(tpx>=2 && interx(lx[i],hx[tpx])<=interx(lx[i],hx[tpx-1])) tpx--;
    			hx[++tpx]=lx[i];
    		}
    		tpy=0;
    		FOR(i,1,n){
    			if(tpy && ly[i].k==hy[tpy].k) continue;
    			while(tpy>=2 && interx(ly[i],hy[tpy])<=interx(ly[i],hy[tpy-1])) tpy--;
    			hy[++tpy]=ly[i];
    		}
    		tot=0;
    		int j=1;
    		FOR(i,1,tpx){
    			double lft=i==1?-1e18:interx(hx[i],hx[i-1]),rig=i==tpx?1e18:interx(hx[i],hx[i+1]);
    			while(j<=tpy){
    				double L=j==1?-1e18:interx(hy[j],hy[j-1]),R=j==tpy?1e18:interx(hy[j],hy[j+1]);
    				if(L>rig) break;
    				if(R>=lft) h[++tot]=merge(hx[i],hy[j]);
    				if(R>rig) break;
    				j++;
    			}
    		}
    		int ans1=0,ans2=0;
    		FOR(i,1,tot){
    			double lft=i==1?-1e18:interx(h[i],h[i-1]),rig=i==tot?1e18:interx(h[i],h[i+1]);
    			double x=1.0*(l-h[i].b)/h[i].k;
    			if(x>=lft && x<=rig){
    				ans1=h[i].id1;
    				ans2=h[i].id2;
    				break;
    			}
    		}
    		printf("%d %d
    ",ans1,ans2);
    		FOR(i,1,n-1) if(lx[i].id1==ans1) swap(lx[i],lx[i+1]);
    		FOR(i,1,n-1) if(ly[i].id1==ans2) swap(ly[i],ly[i+1]);
    		n--;
    	}
    }
    

    [eJOI2017] 骆驼(*)

    居然前几天模拟赛做过……

    至少我写的很重工业。也幸好是模拟赛做过不然根本静不下心来写这个(

    天哪要是我在现场怕不是要被这个题耗费一生然后白给了

    这个范围提醒我们 (5 imes 5) 分成一块块。

    然后瞎做。我用了类似下图的写法。

    时间复杂度 (O(n^2))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=1010,mod=998244353;
    const int a1[5][5]={
    	{1,15,25,22,14}, 
    	{8,5,19,11,6},
    	{17,23,2,16,24},
    	{20,12,7,21,13},
    	{9,4,18,10,3}
    };
    const int a2[5][5]={
    	{1,11,8,20,12},
    	{16,5,23,15,6},
    	{9,19,2,10,25},
    	{22,14,7,21,13},
    	{17,4,24,18,3}
    };
    const int a3[5][5]={
    	{1,6,21,18,7},
    	{13,16,24,10,15},
    	{22,19,2,5,20},
    	{25,9,14,17,8},
    	{12,4,23,11,3}
    };	// go out at 21
    const int a4[5][5]={
    	{25,8,11,22,7},
    	{18,3,14,17,2},
    	{10,21,6,9,12},
    	{24,16,1,23,15},
    	{19,4,13,20,5}
    };
    const int a5[5][5]={
    	{24,13,16,23,1},
    	{4,21,10,5,18},
    	{15,7,2,14,8},
    	{25,12,17,22,11},
    	{3,20,9,6,19}
    };
    const int a6[5][5]={
    	{9,17,1,10,18},
    	{14,23,20,15,24},
    	{4,11,8,5,2},
    	{21,16,25,22,19},
    	{13,6,3,12,7}
    };
    const int a7[5][5]={
    	{9,2,21,10,1},
    	{19,5,13,16,6},
    	{22,25,8,3,24},
    	{12,15,20,11,14},
    	{18,4,23,17,7}
    };
    const int a8[5][5]={
    	{19,6,12,20,5},
    	{24,16,9,23,2},
    	{13,21,4,14,11},
    	{18,7,1,17,8},
    	{25,15,10,22,3}
    };
    const int a9[5][5]={
    	{25,10,7,22,11},
    	{18,15,4,1,16},
    	{8,21,12,9,6},
    	{24,2,17,23,3},
    	{19,14,5,20,13}
    };
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,ans[maxn][maxn],cnt;
    int main(){ 
    	n=read();
    	if(n%10==0){
    		FOR(i,0,4) FOR(j,0,4) ans[i][j]=a3[i][j]<=21?a3[i][j]:n*n-(25-a3[i][j]);
    		cnt=1;
    		FOR(i,0,n/5-1) if(i%2==0){
    			FOR(j,1,n/5-1) if(j!=n/5-1){
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a1[k][l];
    				cnt++;
    			}
    			else{
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a2[k][l];
    				cnt++;
    			}
    		}
    		else{
    			ROF(j,n/5-1,1) if(j!=1){
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a1[k][4-l];
    				cnt++;
    			}
    			else if(i!=n/5-1){
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a2[k][4-l];
    				cnt++;
    			}
    			else{
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a5[k][l];
    				cnt++;
    			} 
    		}
    		ROF(i,n/5-1,1) FOR(j,0,0){
    			FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a4[k][l];
    			cnt++;
    		}
    	}
    	else{
    		FOR(i,0,4) FOR(j,0,4) ans[i][j]=a3[i][j]<=21?a3[i][j]:n*n-(25-a3[i][j]);
    		cnt=1;
    		FOR(i,0,n/5-3) if(i%2==0){
    			FOR(j,1,n/5-1) if(j!=n/5-1){
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a1[k][l];
    				cnt++;
    			}
    			else if(i!=n/5-3){
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a2[k][l];
    				cnt++;
    			}
    			else{
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a5[k][4-l];
    				cnt++;
    			}
    		}
    		else{
    			ROF(j,n/5-1,1) if(j!=1){
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a1[k][4-l];
    				cnt++;
    			}
    			else{
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a2[k][4-l];
    				cnt++;
    			}
    		}
    		ROF(j,n/5-1,1) if((n/5-1-j)%2==0){
    			FOR(i,n/5-2,n/5-1) if(i==n/5-2){
    				if(j==n/5-1){
    					FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a6[k][l]; 
    				}
    				else{
    					FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a7[4-l][4-k];
    				}
    				cnt++;
    			}
    			else{
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a7[k][l];
    				cnt++;
    			}
    		}
    		else{
    			ROF(i,n/5-1,n/5-2) if(i==n/5-2){
    				if(j==1){
    					FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a8[k][l];
    				}
    				else{
    					FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a7[4-k][l];
    				}
    				cnt++;
    			}
    			else{
    				FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a7[4-l][k];
    				cnt++;
    			}
    		}
    		ROF(i,n/5-1,1) FOR(j,0,0) if(i!=n/5-1){
    			FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a4[k][l];
    			cnt++;
    		}
    		else{
    			FOR(k,0,4) FOR(l,0,4) ans[i*5+k][j*5+l]=cnt*25-4+a9[k][l];
    			cnt++;
    		}
    	}
    	FOR(i,0,n-1){
    		FOR(j,0,n-1) printf("%d ",ans[i][j]);
    		puts("");
    	}
    }
    

    [eJOI2017] 经验(*)

    首先发现,对于每一条链,两个端点的 (w) 一定是一个链上最小值一个链上最大值。如果不是,那么找出最小值和最大值的位置,把外面的砍掉,不会变劣。

    把题目变一下:选出一个链划分,若每条链的端点是 (u,v),求 (|w_u-w_v|) 的和的最大值。

    再变一下:选出一个链划分,若每条链的端点是 (u,v),那么你可以选定这条链的价值是 (w_u-w_v) 还是 (w_v-w_u),求链的价值的和的最大值。

    那么设 (f_{u,0/1}) 表示 (u) 的子树内,(0) 表示 (u) 所在的链是浅的权值减去深的权值,(1) 反过来,的最大答案。

    时间复杂度 (O(n))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=100010,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,el,head[maxn],to[maxn*2],nxt[maxn*2],w[maxn];
    ll f[maxn][2];
    inline void add(int u,int v){
    	to[++el]=v;nxt[el]=head[u];head[u]=el;
    }
    void dfs(int u,int F){
    	ll tot=0,mx0=-1e18,mx1=-1e18;
    	for(int i=head[u];i;i=nxt[i]){
    		int v=to[i];
    		if(v==F) continue;
    		dfs(v,u);
    		ll x=max(f[v][0]+w[v],f[v][1]-w[v]);
    		tot+=x;
    		mx0=max(mx0,f[v][0]-x);
    		mx1=max(mx1,f[v][1]-x);
    	}
    	f[u][0]=max(tot-w[u],tot+mx0);
    	f[u][1]=max(tot+w[u],tot+mx1);
    }
    int main(){
    	n=read();
    	FOR(i,1,n) w[i]=read();
    	FOR(i,1,n-1){
    		int u=read(),v=read();
    		add(u,v);add(v,u);
    	}
    	dfs(1,0);
    	printf("%lld
    ",max(f[1][0]+w[1],f[1][1]-w[1]));
    }
    

    [eJOI2017] 游戏

    这题居然没想出来,大概可以退役了 /kk

    显然每次取的都是最大的数。

    问题变成每次加入一个数,然后把最大值删掉。

    用一个指针记录最大值的位置。

    注意到如果加入的数比目前的最大值大,那么它会立刻被删掉,直接处理。

    否则,就暴力移指针。

    指针只会往小了移,所以时间复杂度 (O(nk))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=100010,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,q,a[maxn],cnt[maxn];
    int main(){
    	n=read();q=read();
    	FOR(i,1,n) a[i]=read();
    	while(q--){
    		int k=read(),cur=0;
    		ll ans=0;
    		FOR(i,1,n) cnt[i]=0;
    		FOR(i,1,k-1) cnt[a[i]]++,cur=max(cur,a[i]);
    		FOR(i,1,n){
    			int x=cur;
    			if(k+i-1<=n){
    				if(a[i+k-1]>cur) x=a[i+k-1];
    				else cnt[a[i+k-1]]++,cnt[cur]--;
    			}
    			else cnt[cur]--;
    			while(cur && !cnt[cur]) cur--;
    			if(i%2==0) ans-=x;
    			else ans+=x;
    		}
    		printf("%lld
    ",ans);
    	}
    }
    

    [eJOI2018] 山(*)

    两座被选定的山肯定不能相邻。

    如果选定了保留哪些山,那么只有这些山两旁的山会被降低。

    那么随便做了。

    (f_{i,j,0/1/2}) 表示前 (i) 座山,选了 (j) 座留下,(0) 表示 (i)(i-1) 都没留下,(1) 表示 (i-1) 留下了,(2) 表示 (i) 留下了。

    时间复杂度 (O(n^2))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=5050,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,a[maxn],f[maxn][maxn/2][3];
    int main(){
    	n=read();
    	FOR(i,1,n) a[i]=read();
    	FOR(i,0,n) FOR(j,0,(n+1)/2) f[i][j][0]=f[i][j][1]=f[i][j][2]=1e9;
    	f[0][0][0]=0;
    	FOR(i,1,n) FOR(j,0,(i+1)/2){
    		f[i][j][0]=min(f[i-1][j][0],f[i-1][j][1]);
    		f[i][j][1]=f[i-1][j][2];
    		if(j){
    			f[i][j][2]=f[i-1][j-1][0]+max(0,a[i-1]-a[i]+1)+max(0,a[i+1]-a[i]+1);
    			if(i>=2) f[i][j][2]=min(f[i-1][j-1][1]+max(0,min(a[i-1],a[i-2]-1)-a[i]+1)+max(0,a[i+1]-a[i]+1),f[i][j][2]);
    		}
    	}
    	FOR(i,1,(n+1)/2) printf("%d ",min(min(f[n][i][0],f[n][i][1]),f[n][i][2]));
    }
    

    [eJOI2018] 护照

    [eJOI2018] AB 串

    声明:这题我没过。是因为洛谷计分系统的问题导致把我判成了 AC。

    [eJOI2018] 元素周期表

    还是很妙的。

    把每行每列都抽象成点,一个点 ((x,y)) 就将第 (x) 行和第 (y) 行连边。

    ((x1,y1),(x1,y2),(x2,y1)) 都存在,那么第 (x2) 行和第 (y2) 行一定联通,此时也看做 ((x2,y2)) 存在。

    所以会变成一堆联通块。答案就是联通块个数减一。

    时间复杂度 (O(n+m+qmathrm{DSU}(n+m)))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=200020,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,m,q,fa[maxn*2],cnt;
    int getfa(int x){
    	return x==fa[x]?x:fa[x]=getfa(fa[x]); 
    }
    int main(){
    	n=read();m=read();q=read();
    	cnt=n+m;
    	FOR(i,1,n+m) fa[i]=i;
    	while(q--){
    		int x=read(),y=read()+n;
    		x=getfa(x);y=getfa(y);
    		if(x!=y) fa[x]=y,cnt--;
    	}
    	printf("%d
    ",cnt-1);
    }
    

    [eJOI2018] 互素树

    [eJOI2018] 循环排序

    [eJOI2019] 异或橙子(*)

    考虑询问 ([l,r]) 时,(i) 被异或了多少次。显然是 ((i-l+1)(r-i+1))

    (l,r) 不同奇偶时,这个永远是偶数,答案就是 (0)

    (l,r) 同奇偶时,当且仅当 (i)(l) 同奇偶时是奇数。两个树状数组维护即可。

    时间复杂度 (O((n+q)log n))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=200020,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,q,a[maxn],b[2][maxn];
    inline void update(int *b,int p,int v){
    	for(int i=p;i<=n;i+=i&-i) b[i]^=v;
    }
    inline int query(int *b,int p){
    	int s=0;
    	for(int i=p;i;i-=i&-i) s^=b[i];
    	return s;
    }
    inline int query(int *b,int l,int r){
    	return query(b,r)^query(b,l-1);
    }
    int main(){
    	n=read();q=read();
    	FOR(i,1,n) update(b[i%2],i,a[i]=read());
    	while(q--){
    		int op=read(),x=read(),y=read();
    		if(op==1){
    			update(b[x%2],x,a[x]^y);
    			a[x]=y;
    		}
    		else{
    			if((y-x)%2==1) puts("0");
    			else printf("%d
    ",query(b[x%2],x,y));
    		} 
    	}
    }
    

    [eJOI2019] 挂架(*)

    看看样例解释找规律,然后就是普及组模拟了。

    具体一点,每次特判 (kle 2),然后后面会分成长度为 (2^1,2^2,2^3dots) 的段,每段会加上 (2^{n-2},2^{n-3},2^{n-4},dots)

    找到 (k) 所在那段,加上,减去左端点,接着做。

    时间复杂度看实现,反正瞎写都能过。(给自己的垃圾 (O(nlog k)) 找借口)

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=1000100,mod=1000000007;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,pw[maxn],ans;
    ll k;
    int main(){
    	n=read();k=read();
    	pw[0]=1;
    	FOR(i,1,n) pw[i]=2*pw[i-1]%mod;
    	while(k){
    		if(k==1){ans=(ans+1)%mod;break;}
    		if(k==2){ans=(ans+1+pw[n-1])%mod;break;}
    		ll tmp=2;
    		int cur=n-1;
    		while(tmp<k) tmp*=2,cur--;
    		ans=(ans+pw[cur])%mod;
    		k-=tmp/2;
    	}
    	printf("%d
    ",ans);
    }
    

    [eJOI2019] T 形覆盖(*)

    部分分启发意义很大。

    第一个包,两两独立,很好做。

    第二个包,对于相邻的显然只有一种方法,否则还是独立的。

    第三个包,对于只有一个顶点相邻的,虽然有两种方法,但是覆盖到的格子完全一样。那么我们对每个八连通块分开考虑,八连通块之间独立。

    只剩下 (x_i=x_j,|y_i-y_j|=2) 或者 (|x_i-x_j|=2,y_i=y_j) 了,其它的上面讨论过了。

    我们考虑先把其它的都确定了,然后最后剩下一堆无法确定的。把满足上面那条式子的两点连个边,然后考虑每个联通块。

    如果一个联通块是树,(c) 个点,那么除掉 (c) 个中心,还剩 (3c+1) 个格子。所以选择最小的一个格子不选即可。手玩发现肯定能还原放的方法。

    如果一个联通块是基环树,那么还剩 (3c) 个格子,都得选。

    如果都不是,那么剩余不到 (3c) 个格子,一定无解。

    注意当有中心在边界上或者与已经被覆盖的格子相邻时,还有一点点区别,不过问题不大。

    时间复杂度 (O(nm))

    由于菜,写的又臭又长。

    写的时候可以这样写:每覆盖一个格子,它周围的 T 中心都处理一下(此时对于这个 T 中心合法方案不超过一种),递归下去。这样细节会少一点。

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=1000100,mod=998244353,d1[4][2]={{0,1},{0,-1},{-1,0},{1,0}},d2[4][2]={{1,1},{-1,-1},{-1,1},{1,-1}},d3[4][2]={{0,2},{0,-2},{2,0},{-2,0}};
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int n,m,k,ans,cnt,mn;
    int *a[maxn],*tp[maxn],*vis[maxn];
    bool *bl[maxn];
    inline void nsol(){puts("No");exit(0);}
    void work2(int,int);
    void work(int x,int y,int d){
    	if(tp[x][y]!=-1 && tp[x][y]!=d) nsol();
    	if(tp[x][y]==d) return;
    	tp[x][y]=d;
    	FOR(i,0,3) if(i!=d){
    		int tx=x+d1[i][0],ty=y+d1[i][1];
    		if(tx<1 || tx>n || ty<1 || ty>m || vis[tx][ty]!=-1) nsol();
    		ans+=a[tx][ty];
    		vis[tx][ty]=i^1;
    		work2(tx,ty);
    	}
    }
    void work2(int x,int y){
    	FOR(i,0,3) if(i!=vis[x][y]){
    		int tx=x+d1[i][0],ty=y+d1[i][1];
    		if(bl[tx][ty]) work(tx,ty,i^1);
    	}
    }
    void dfs(int x,int y,int lx,int ly){
    	cnt+=2;
    	tp[x][y]=-2;
    	FOR(i,0,3){
    		int tx=x+d3[i][0],ty=y+d3[i][1];
    		if(tx<1 || tx>n || ty<1 || ty>m || tp[tx][ty]>=0 || !bl[tx][ty]) continue;
    		cnt--;
    		if(tp[tx][ty]==-1) dfs(tx,ty,x,y);
    	}
    	FOR(i,0,3){
    		int tx=x+d1[i][0],ty=y+d1[i][1];
    		if(tx<1 || tx>n || ty<1 || ty>m || vis[tx][ty]!=-1){
    			if(vis[tx][ty]!=-2) cnt-=2;
    			continue;
    		}
    		vis[tx][ty]=-2;
    		ans+=a[tx][ty];
    		mn=min(mn,a[tx][ty]);
    	}
    }
    int main(){
    	n=read();m=read();
    	FOR(i,0,n+1){
    		a[i]=new int[m+2];
    		bl[i]=new bool[m+2];
    		vis[i]=new int[m+2];
    		tp[i]=new int[m+2];
    		FOR(j,0,m+1){
    			a[i][j]=0;
    			bl[i][j]=false;
    			vis[i][j]=-1;
    			tp[i][j]=-1;
    		}
    	}
    	FOR(i,1,n) FOR(j,1,m) a[i][j]=read();
    	k=read();
    	while(k--){
    		int x=read()+1,y=read()+1;
    		ans+=a[x][y];
    		bl[x][y]=true;
    		vis[x][y]=-2;
    		FOR(i,0,3){
    			int tx=x+d1[i][0],ty=y+d1[i][1];
    			if(bl[tx][ty] && tp[x][y]==-1) work(x,y,i),work(tx,ty,i^1);
    		}
    	}
    	FOR(x,1,n) FOR(y,1,m) if(bl[x][y] && tp[x][y]==-1){
    		FOR(i,0,3){
    			int tx=x+d2[i][0],ty=y+d2[i][1];
    			if(bl[tx][ty] && tp[x][y]==-1){work(x,y,i);break;}
    		}
    	}
    	FOR(x,1,n) FOR(y,1,m) if(bl[x][y] && tp[x][y]==-1){
    		mn=1e9;cnt=0;
    		dfs(x,y,-1,-1);
    		if(cnt<0) nsol();
    		if(cnt==2) ans-=mn;
    	}
    	printf("%d
    ",ans);
    	FOR(i,0,n+1){
    		delete[] a[i];
    		delete[] bl[i];
    		delete[] vis[i];
    		delete[] tp[i];
    	}
    }
    

    [eJOI2019] 塔(*)

    先特判 (n=1)

    如果第 (i) 次操作让 (l_i=1,r_i=i),那么最后会得到 (2^{i-1})

    所以我们找到一个最小的 (m) 满足 (2^{m-1}ge n),明显至少要 (m) 次操作。

    (m) 次操作也是可以做到的。选一些 (l_i) 加一,发现会让最后一个数减掉 (2^{max(1,m-i-1)})

    要减哪些很好弄,正确性也显然。

    时间复杂度 (O(Tlog n))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=100010,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    int T,m,l[maxn],r[maxn];
    ll n,tmp[maxn];
    void solve(){
    	n=read();
    	if(n==1) return void(puts("0"));
    	m=0;
    	while(true){
    		l[++m]=1;
    		r[m]=m;
    		if(m==1) tmp[m]=1;
    		else tmp[m]=tmp[m-1]*2;
    		if(tmp[m]>=n) break;
    	}
    	n=tmp[m]-n;
    	reverse(tmp,tmp+m+1);
    	FOR(i,1,m-1) if(n>=tmp[i]) n-=tmp[i],l[i]++;
    	printf("%d
    ",m);
    	FOR(i,1,m) printf("%d %d
    ",l[i],r[i]);
    }
    int main(){
    	T=read();
    	while(T--) solve();
    }
    

    [eJOI2019] 矩形染色(*)

    先考虑另一个问题:每次可以染一行或一列。

    容易发现,要么所有行都选,要么所有列都选。因为如果有一行不选,那么为了把那行都染上色,只能所有列都选。有一列不选同理。

    到这题我们也用类似的想法。

    首先黑白染色,那么黑白部分独立。这样接下来看着会稍微舒服一些。下面以包括左上角的那部分为例。

    我们稍微转个 45 度,然后会变成个形如这样的东西。

    我们从左往右枚举哪些列没有被选。先判掉所有列都选了的情况。

    (f_i) 表示考虑了前 (i) 列,第 (i)没有被选的最小代价。

    转移枚举上一个没选的列 (j),由于这个图很特殊,只需要知道 (j),就能知道少选第 (i) 列就要多选哪些行。只不过情况有点多。

    发现可以线段树优化,时间复杂度 (O((n+m)log(n+m)))

    代码很恶心。虽然可能纯粹是因为我菜而写得很丑。

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=400040,mod=998244353;
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    struct segtree{
    	int n;
    	ll mn[maxn*4];
    	inline void pushup(int o){
    		mn[o]=min(mn[o<<1],mn[o<<1|1]);
    	}
    	void build(int o,int l,int r){
    		if(l==r) return void(mn[o]=1e18);
    		int mid=(l+r)>>1;
    		build(lson);build(rson);
    		pushup(o);
    	}
    	inline void build(int n){
    		build(1,1,n);
    		this->n=n;
    	}
    	void update(int o,int l,int r,int p,ll v){
    		if(l==r) return void(mn[o]=v);
    		int mid=(l+r)>>1;
    		if(mid>=p) update(lson,p,v);
    		if(mid<p) update(rson,p,v);
    		pushup(o);
    	}
    	inline void update(int p,ll v){
    		update(1,1,n,p,v);
    	}
    	ll query(int o,int l,int r,int ql,int qr){
    		if(l>=ql && r<=qr) return mn[o];
    		int mid=(l+r)>>1;
    		if(mid<ql) return query(rson,ql,qr);
    		if(mid>=qr) return query(lson,ql,qr);
    		return min(query(lson,ql,qr),query(rson,ql,qr));
    	}
    	inline ll query(int l,int r){
    		l=max(l,1);r=min(r,n);
    		if(l>r) return 1e18;
    		return query(1,1,n,l,r);
    	}
    }t1,t2,t3;
    int n,m,a[maxn],b[maxn];
    ll f[maxn],g[maxn],apre[maxn],bpre[maxn];
    ll work(int beg){
    	t1.build(n+m-1);t2.build(n+m-1);t3.build(n+m-1);
    	ll ans=0;
    	for(int i=beg;i<=n+m-1;i+=2){
    		ans+=b[i];
    		if(i<=m){
    			int l=m-i+1,r=m+i-1;
    			ll c=apre[r]-apre[max(0,l-2)];
    			f[i]=c+(i==1?0:bpre[i-2]);
    			f[i]=min(f[i],t1.query(1,i)+c+(i==1?0:bpre[i-2]));
    			t1.update(i,f[i]-c-bpre[i]);
    			t2.update(i,f[i]-apre[r]-bpre[i]);
    			t3.update(i,f[i]-bpre[i]);
    		}
    		else if(i<=n){
    			int l=i-m+1,r=m+i-1;
    			ll c=apre[r]-apre[max(0,l-2)];
    			f[i]=c+bpre[i-2];
    			f[i]=min(f[i],t3.query(1,l-m)+c+bpre[i-2]);
    			f[i]=min(f[i],t1.query(l-m+1,2*m-i)+c+bpre[i-2]);
    			f[i]=min(f[i],t2.query(2*m-i+1,i)+apre[r]+bpre[i-2]);
    			t2.update(i,f[i]-apre[r]-bpre[i]);
    			t3.update(i,f[i]-bpre[i]);
    		}
    		else{
    			int l=i-m+1,r=m+2*n-i-1;
    			ll c=apre[r]-apre[max(0,l-2)];
    			f[i]=c+bpre[i-2];
    			f[i]=min(f[i],t3.query(1,l-m)+c+bpre[i-2]);
    			f[i]=min(f[i],t1.query(l-m+1,m-l+1)+c+bpre[i-2]);
    			f[i]=min(f[i],t2.query(m-l+2,2*n-i)+apre[r]+bpre[i-2]);
    			f[i]=min(f[i],t3.query(2*n-i+1,i)+bpre[i-2]);
    			t3.update(i,f[i]-bpre[i]);
    		}
    	}
    	int upr=n+m-1;
    	if(upr%2!=beg%2) upr--;
    	for(int i=beg;i<=n+m-1;i+=2) ans=min(ans,f[i]+bpre[upr]-bpre[i]);
    	return ans;
    }
    int main(){
    	n=read();m=read();
    	FOR(i,1,n+m-1) a[i]=read();
    	FOR(i,1,n+m-1) b[i]=read();
    	if(n<m){
    		swap(n,m);
    		reverse(a+1,a+n+m);
    	}
    	FOR(i,1,n+m-1){
    		if(i==1) apre[i]=a[i],bpre[i]=b[i];
    		else apre[i]=apre[i-2]+a[i],bpre[i]=bpre[i-2]+b[i];
    	}
    	printf("%lld
    ",work(1)+work(2));
    }
    

    [eJOI2019] 箭头国探险(*)

    这个中二的题目名是我自己弄的

    我们把对 ((i,j)) 的转向变成走到 ((i,j)) 上再转。

    (f_{i,j}) 表示走到 ((i,j)) 的最小代价。

    用最短路转移。

    时间复杂度 (O(nmlog nm)),写四个队列也可以做到 (O(nm)) 带个不小的常数。

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    typedef pair<int,int> PII;
    const int maxn=505,MAXN=maxn*maxn,MAXM=MAXN*4,mod=998244353,d[4][2]={{0,1},{1,0},{0,-1},{-1,0}};
    #define MP make_pair
    #define PB push_back
    #define lson o<<1,l,mid
    #define rson o<<1|1,mid+1,r
    #define FOR(i,a,b) for(int i=(a);i<=(b);i++)
    #define ROF(i,a,b) for(int i=(a);i>=(b);i--)
    #define MEM(x,v) memset(x,v,sizeof(x))
    inline ll read(){
    	char ch=getchar();ll x=0,f=0;
    	while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
    	while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
    	return f?-x:x;
    }
    struct node{
    	int u,d;
    	bool operator<(const node &nd)const{return d>nd.d;}
    };
    int n,m,tp[128],el,head[MAXN],to[MAXM],nxt[MAXM],w[MAXM],dis[MAXN];
    char mp[maxn][maxn];
    priority_queue<node> pq;
    inline int id(int x,int y){
    	return 1ll*(x-1)*m+y;
    }
    inline void add(int u,int v,int w_){
    	to[++el]=v;nxt[el]=head[u];head[u]=el;w[el]=w_;
    }
    int main(){
    	tp['E']=0;tp['S']=1;tp['W']=2;tp['N']=3; 
    	n=read();m=read();
    	FOR(i,1,n) scanf("%s",mp[i]+1);
    	FOR(i,1,n) FOR(j,1,m) if(mp[i][j]!='X') FOR(k,0,3){
    		int x=i+d[k][0],y=j+d[k][1];
    		if(x<1 || x>n || y<1 || y>m) continue;
    		add(id(i,j),id(x,y),(k-tp[mp[i][j]]+4)%4);
    	}
    	FOR(i,1,n*m) dis[i]=1e9;
    	dis[1]=0;
    	pq.push((node){1,0});
    	while(!pq.empty()){
    		int u=pq.top().u,d=pq.top().d;pq.pop();
    		if(d>dis[u]) continue;
    		for(int i=head[u];i;i=nxt[i]){
    			int v=to[i];
    			if(dis[v]>d+w[i]) pq.push((node){v,dis[v]=d+w[i]});
    		}
    	}
    	printf("%d
    ",dis[id(n,m)]==1e9?-1:dis[id(n,m)]);
    }
    

    Bonus: [eJOI2018] 互素树 加强版

  • 相关阅读:
    编写JS代码的“use strict”严格模式及代码压缩知识
    开发网站要从用户的角度出发!
    你好,世界
    JavaScript的几种函数的结构形式
    JavaScript功能检测技术和函数构造
    android打造万能的适配器
    C语言第二次博客作业分支结构
    C语言第三次博客作业单层循环结构
    C语言第一次博客作业——输入输出格式
    C语言第四次博客作业嵌套循环
  • 原文地址:https://www.cnblogs.com/1000Suns/p/12996785.html
Copyright © 2011-2022 走看看