【题解】CF2B The least round way

The least round way

\(\text{Solution:}\)

第一种想法:设 \(dp[i][j][k]\) 表示格子 \((i,j)\) 尾数是 \(k\) 的最少零的个数,转移需要枚举尾数。

由于要输出路径,空间开不下而 GG .

第二种:考虑为什么会出现 \(0:\) 十进制下只有 \(2,5\) 相乘才可以。所以一个位置上最少的零的个数应该是 \(\min\left\{Num_2,Num_5\right\}\)

于是考虑对这两个东西进行 dp 。设 \(f[i][j]\) 是到达格子 \((i,j)\) 的最少 \(2\) 的个数, \(g[i][j]\) 对应 \(5\) 的个数。

那么对这两个东西进行转移即可。同时记录一下路径。

如果格子中有 \(0\) 的话,则答案必然不高于 \(1\) 特判一下即可。

#include<bits/stdc++.h>
using namespace std;
const int dyx=(1<<30);
int n,a[1001][1001];
int f[1001][1001],g[1001][1001];
int numt[1001][1001],numf[1001][1001];
int FG,F[1001][1001];
pair<int,int> pret[1001][1001],pref[1001][1001];
pair<int,int> pos0;
inline int Min(int x,int y) {
	return x<y?x:y;
}
void print(pair<int,int> path,int t){
	int fir=path.first;
	int sec=path.second;
//	printf("(%d %d)\n",fir,sec);
	if(fir==1&&sec==1)return;
	if(t==1)print(pret[fir][sec],t);
	else print(pref[fir][sec],t);
	if(t==1){
		if(pret[fir][sec].first<fir)printf("D");
		else printf("R");
	}
	else{
		if(pref[fir][sec].first<fir)printf("D");
		else printf("R");
	}
}
int main() {
	freopen("CF2B.txt","r",stdin);
	//f[i][j] g[i][j]
	scanf("%d",&n);
	for(int i=1; i<=n; ++i)
		for(int j=1; j<=n; ++j) {
			f[i][j]=g[i][j]=dyx;
			scanf("%d",&a[i][j]);
			if(a[i][j]==0)FG=1,pos0=make_pair(i,j),F[i][j]=1;
			int cnt=0;
			int x=a[i][j];
			while(x%2==0&&x)cnt++,x/=2;
			numt[i][j]=cnt;cnt=0;
			while(x%5==0&&x)cnt++,x/=5;
			numf[i][j]=cnt;cnt=0;
		}
//	for(int i=1;i<=n;++i)
//		for(int j=1;j<=n;++j)
//			printf("(%d %d)%c",numt[i][j],numf[i][j],j==n?'\n':' ');
	for(int i=0;i<=n;++i)
		for(int j=0;j<=n;++j)
			f[i][j]=g[i][j]=dyx;
	f[1][1]=numt[1][1];
	g[1][1]=numf[1][1];
	for(int i=1; i<=n; ++i) {
		for(int j=1; j<=n; ++j) {
			if(i==1&&j==1)continue;
			if(F[i][j])continue;
			if(f[i-1][j]<f[i][j-1])pret[i][j]=make_pair(i-1,j);
			else pret[i][j]=make_pair(i,j-1);
			f[i][j]=Min(f[i-1][j],f[i][j-1])+numt[i][j];
			if(g[i-1][j]<g[i][j-1])pref[i][j]=make_pair(i-1,j);
			else pref[i][j]=make_pair(i,j-1);
			g[i][j]=Min(g[i-1][j],g[i][j-1])+numf[i][j];
		}
	}
//	for(int i=1;i<=n;++i)
//		for(int j=1;j<=n;++j)
//			printf("(%d %d)%c",f[i][j],g[i][j],j==n?'\n':' ');
//	puts("PathF:");
	int ans=dyx;
//	for(int i=1;i<=n;++i){
//		for(int j=1;j<=n;++j){
//			printf("(%d %d) ",pret[i][j].first,pret[i][j].second);
//		}
//		puts("");
//	}
//	puts("PathG:");
//	for(int i=1;i<=n;++i){
//		for(int j=1;j<=n;++j){
//			printf("(%d %d) ",pref[i][j].first,pref[i][j].second);
//		}
//		puts("");
//	}
	ans=Min(ans,Min(f[n][n],g[n][n]));
	if(!FG||(FG&&ans<=1)){
		printf("%d\n",ans);
		if(f[n][n]<g[n][n]){
//			if(pret[n][n].first<n)printf("R");
//			else printf("D");
			print(make_pair(n,n),1);
		}
		else {
//			if(pref[n][n].first<n)printf("R");
//			else printf("D");
			print(make_pair(n,n),0);
		}
		puts("");
	}
	else {
		puts("1");
		for(int i=1;i<pos0.first;++i)printf("D");
		for(int i=1;i<pos0.second;++i)printf("R");
		for(int i=1;i<=n-pos0.first;++i)printf("D");
		for(int i=1;i<=n-pos0.second;++i)printf("R");
		puts("");
	}
	return 0;
}
posted @ 2021-07-07 14:49  Refined_heart  阅读(33)  评论(0编辑  收藏  举报