Toyota Programming Contest 2023#4(AtCoder Beginner Contest 311)(A-G)

这篇具有很好参考价值的文章主要介绍了Toyota Programming Contest 2023#4(AtCoder Beginner Contest 311)(A-G)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Contest Duration: 2023-07-22(Sat) 20:00 - 2023-07-22(Sat) 21:40 (local time) (100 minutes)

头文件和宏

#include<iostream>
#include<string>
#include<vector>
using namespace std;
#define int long long
#define fer(i,a,b) for(int i=a;i<b;i++)
#define cf int T;cin>>T;while(T--)
#define pb push_back
#define IOS ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);

以下只附主代码

A First ABC

从头开始,找出ABC三个字母都至少出现一次的最小长度

signed main(){
	IOS;
	int n;cin>>n;
	string s;cin>>s;
	bool fa=0,fb=0,fc=0;
	int cnt=0;
	fer(i,0,n){
		if(s[i]=='A')fa=1;
		else if(s[i]=='B')fb=1;
		else if(s[i]=='C')fc=1;
		if(fa&&fb&&fc){
			cnt=i;break;
		}
	}
	cout<<cnt+1;
	return 0;
}

B Vacation Together

找出n人均有空闲的最大天数

signed main(){
	IOS;
	int n,d;cin>>n>>d;
	bool f[d]={0};
	string s;
	fer(i,0,n){
		cin>>s;
		fer(j,0,d){
			if(s[j]=='x')f[j]=1;
		}
	}
	int cnt=0,mx=0;
	fer(i,0,d){
		if(f[i]==0){
			cnt++;
			mx=max(mx,cnt);
		}else cnt=0;
	}
	cout<<mx;
	return 0;
}

C Find it!

有向图,N个结点,N条边,一个结点只会指向另外一个结点,产生一条边。题目要求输出任意环。
这种方式形成的有向图一定有环,最小的环是M=2,最大的环是M=N,所以只要顺着找N+1次,必然会出现环,res里最后一个结点必然在环里。即使重复在一个环里循环也无妨,只需从后向前截取这个环作为结果。

const int N=2e5+5,mod=1e9+7;
int a[N];
signed main(){
	IOS;
	int n;cin>>n;
	fer(i,1,n+1)cin>>a[i];
	vector<int>res;res.clear();
	int j=1;
	res.pb(j);
	fer(i,0,n+1){
		res.pb(a[j]);
		j=a[j];
	}
	int k;
	for(int i=n-1;i>=0;i--){
		if(res[i]==res[n]){
			k=i;break;
		}
	}
	cout<<n-k<<endl;
	for(int i=k;i<n;i++){
		cout<<res[i]<<" ";
	}
	return 0;
}

D Grid Ice Floor

滑冰,如果玩过类似的4399小游戏会感到熟悉。
在冰面上只能朝指定方向滑行,直到遇到障碍物停下,中途不能改变方向。问最多能滑过多少块冰。问题在于:如何判定停止递归?不能只是单纯遇到走过的点就停止。比如下图中的红线是可以走的,但如果“遇到走过的点就停止”,那么就不会递归这条红线的情况,这是错误的。
Toyota Programming Contest 2023#4(AtCoder Beginner Contest 311)(A-G),acm,算法,c++,acm
因此我设置了两个布尔数组,f代表走过的点,用来计数;f2代表该点是否作为过停留的结点向四个方向递归,如果f2==1,那么我们不必重复去递归

int a[202][202];bool f[202][202],f2[202][202];
int cnt=0;
void solve(int x,int y,int op){//1上 2下 3左 4右 

	if(f[x][y]==0){
		cnt++;f[x][y]=1;
	}
	if(op==0&&f2[x][y]==0){
		f2[x][y]=1;
		if(a[x-1][y])solve(x-1,y,1);
		if(a[x+1][y])solve(x+1,y,2);
		if(a[x][y-1])solve(x,y-1,3);
		if(a[x][y+1])solve(x,y+1,4);
		return;
	}else if(op==1){
		if(a[x-1][y])solve(x-1,y,1);
		else solve(x,y,0);
	}else if(op==2){
		if(a[x+1][y])solve(x+1,y,2);
		else solve(x,y,0);
	}else if(op==3){
		if(a[x][y-1])solve(x,y-1,3);
		else solve(x,y,0);
	}else if(op==4){
		if(a[x][y+1])solve(x,y+1,4);
		else solve(x,y,0);
	}
}
signed main(){
	IOS;
	int n,m;cin>>n>>m;
	string s;
	fer(i,0,n){
		cin>>s;
		fer(j,0,m){
			if(s[j]=='.')a[i+1][j+1]=1;
		}
	}
	solve(2,2,0);
	cout<<cnt<<endl;
//	fer(i,1,n+1){
//		fer(j,1,m+1)cout<<f[i][j]<<" ";
//		cout<<endl;
//	}
	return 0;
}

下面是非递归的版本,用的是队列,这种方式开销较小

vector<int> dx = {1, 0, -1, 0};
vector<int> dy = {0, 1, 0, -1};
int main(){
  int N, M;
  cin >> N >> M;
  vector<string> S(N);
  for (int i = 0; i < N; i++){
    cin >> S[i];
  }
  vector<vector<bool>> used(N, vector<bool>(M, false));
  used[1][1] = true;
  vector<vector<bool>> used2(N, vector<bool>(M, false));
  used2[1][1] = true;
  queue<pair<int, int>> Q;
  Q.push(make_pair(1, 1));
  while (!Q.empty()){
    int x = Q.front().first;
    int y = Q.front().second;
    Q.pop();
    for (int i = 0; i < 4; i++){
      int x2 = x, y2 = y;
      while (S[x2 + dx[i]][y2 + dy[i]] == '.'){
        x2 += dx[i];
        y2 += dy[i];
        used2[x2][y2] = true;
      }
      if (!used[x2][y2]){
        used[x2][y2] = true;
        Q.push(make_pair(x2, y2));
      }
    }
  }
  int ans = 0;
  for (int i = 0; i < N; i++){
    for (int j = 0; j < M; j++){
      if (used2[i][j]){
        ans++;
      }
    }
  }
  
  cout << ans << endl;
  for (int i = 0; i < N; i++){
    for (int j = 0; j < M; j++){
      cout<<used2[i][j]<<" ";
      }
      cout<<endl;
    }
    return 0;
}

参考:Submission #43834581

E Defect-free Squares

问无洞正方形有多少个。动态规划
dp初始化为最大值3000+

  • 如果该点是洞,该点dp[i][j]=0;
  • 如果不是:
    • 如果是最底下一行和最右面一列,dp[i][i]=1
    • 如果不是:dp[i][i]=min(dp ,dp[i+1][j]+1,dp[i][j+1]+1,dp[i+1][j+1]+1),也就是取(自身、右面+1、下面+1、右下+1)中的最小值。

如果右面或者下面是洞,都影响自身构建无洞正方形

bool f[3001][3001];
int dp[3001][3001];
signed main(){
	IOS;
	int h,w,n;cin>>h>>w>>n;
	int a,b;
	fer(i,1,n+1){
		cin>>a>>b;
		f[a][b]=1;
	}
	fer(i,1,h+1){
		fer(j,1,w+1)dp[i][j]=3005;
	}

	int sum=0;
	for(int i=h;i>=1;i--){
		for(int j=w;j>=1;j--){
			if(i==h||j==w)dp[i][j]=1;
			if(i<h)dp[i][j]=min(dp[i][j],dp[i+1][j]+1);
			if(j<w)dp[i][j]=min(dp[i][j],dp[i][j+1]+1);
			if(i<h&&j<w)dp[i][j]=min(dp[i][j],dp[i+1][j+1]+1);
			if(f[i][j])dp[i][j]=0;
			sum+=dp[i][j];	
		}
	} 
	cout<<sum<<endl;
	return 0;
}

参考:Submission #43837221

F Yet Another Grid Task

给一个网格,有黑块有白块。小明要把白块涂成黑块使得网格变beautiful,beautiful的定义如下:一个黑块的下面一块和右下角的一块也是黑块。问有多少种涂法。
动态规划,dp初始化为1,按列dp,对每列找到黑块出现的最早一次位置,在这位置之后的块的可能次数均为0(因为黑块之下必须是黑块,已经被定好了),在这位置之前的块,可能次数是自身的可能次数+下方块的可能次数,然后dp整体下移

const int N=2e5+1,mod=998244353;
signed main(){
	IOS;
	int n,m;cin>>n>>m;
	vector<string>s(n);
	fer(i,0,n)cin>>s[i];
	vector<int> dp(n+1,1);
	fer(i,0,m){
		int x=0;
		while(x<n&&s[x][i]=='.')x++;
		for(int j=x+1;j<=n;j++){
			dp[j]=0;
		}
		for(int j=n-1;j>=0;j--){
			dp[j]+=dp[j+1];
			dp[j]%=mod;
		}
		for(int j=n;j>0;j--){
			dp[j]=dp[j-1];
			dp[j]%=mod;
		}
	} 
	cout<<dp[0];
	return 0;
}

参考:Submission #43851107

G One More Grid Task

在网格内拉一个矩形,求(矩形内和x矩形内最小值)的最大值
s存放该列数字的和,t存放该列数字的最小值,待想

signed main(){
	IOS;
	int n,m;cin>>n>>m;
	vector<vector<int> > a(n,vector<int>(m));
	fer(i,0,n){
		fer(j,0,m)cin>>a[i][j];
	}
	int ans=0;
	fer(u,0,n){
		vector<int>s(m),t(m,1e9);
		fer(d,u,n){
			fer(i,0,m){
				s[i]+=a[d][i];
				t[i]=min(a[d][i],t[i]);
			}
			vector<int> l(m,-1),r(m,m),stk;
			fer(i,0,m){
				while(!stk.empty()&&t[i]<t[stk.back()]){
					r[stk.back()]=i;
					stk.pop_back();
				}
				if(!stk.empty())l[i]=stk.back();
				stk.pb(i);
			}
			vector<int> pre(m+1);
			fer(i,0,m)pre[i+1]=pre[i]+s[i];
			fer(i,0,m)ans=max(ans,t[i]*(pre[r[i]]-pre[l[i]+1]));
		}
	}
	cout<<ans;
	return 0;
}

参考:Submission #43853592文章来源地址https://www.toymoban.com/news/detail-602846.html

到了这里,关于Toyota Programming Contest 2023#4(AtCoder Beginner Contest 311)(A-G)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • AtCoder Beginner Contest 313

    貌似这次很难,还好去吃烧烤了 发现原来G就是个类欧几里德算法,参考abc283 ex,更了个G 给定 (n) 个数 (a_i) ,问第一个数要成为唯一的最大的数,应该加多少。 找到后面的最大的数 (m) ,答案就是 (max(0, m + 1 - a_0)) 。 神奇的代码 给定 (m) 条关于 (n) 个数的大小关系,

    2024年02月14日
    浏览(27)
  • AtCoder Beginner Contest 314

    怎么好多陌生单词 审核怎么这么逆天,半小时还没审完 给定 (pi) 的值以及数 (n) ,要求保留 (n) 位小数输出,不四舍五入。 字符串形式储存然后截取末尾即可。 神奇的代码 (n) 个人玩轮盘赌游戏,简单说就是一个转盘有 (37) 个数字以及一个箭头,箭头会等概率停在某

    2024年02月13日
    浏览(31)
  • AtCoder Beginner Contest 322

    给定一个字符串,找到最先出现 ABC 的位置。 直接查找判断即可。 神奇的代码 给定字符串 s 和 t ,问 s 是不是 t 的前缀和后缀。 根据前后缀定义判断即可。这里试了下 python 神奇的代码 (n) 天,有 (m) 天会放烟花。 问每一天,距离未来最近的放烟花的天数。 两个双指针一

    2024年02月08日
    浏览(28)
  • AtCoder Beginner Contest 333

    人生第一次掉rating 各种降智操作 水题 逆天操作 WA了3发 第三次交的时候以为过了,等到切完E发现怎么B还没过( 人家题把上限都告诉你了 然后我 (O(12^3)) 的枚举不写,写半个小时四进制枚举( 以 (1) 为根, (ans=n - max(size[y]hspace{0.4cm} |hspace{0.4cm} y in H[1])) (H[x]) 为 (x

    2024年02月04日
    浏览(48)
  • AtCoder Beginner Contest 299

    给定一个包含 |*. 的字符串,其中 | 两个, * 一个,问 * 是否在两个 | 之间。 找到两个 | 的下标 (l, r) 以及 * 的下标 (mid) ,看看是否满足 (l mid r) 即可。 神奇的代码 给定 (n) 个人的卡片,颜色为 (c_i) ,数字为 (r_i) 。 如果其中有颜色为 (T) 的牌,则该颜色中数字最大

    2023年04月23日
    浏览(11)
  • AtCoder Beginner Contest 343

    给定 (a,b) ,输出 (c) ,使得 (a+b neq c) 从 (0) 开始枚举 (c) 的取值即可。 神奇的代码 给定一个邻接矩阵,对于第 (i) 个点,输出与之有连边的点的编号,升序。 即输出第 (i) 行中值为 (1) 的列即可。 神奇的代码 给定一个数 (n) ,问不超过 (n) 的最大的数 (x) ,其

    2024年03月09日
    浏览(5)
  • AtCoder Beginner Contest 346

    给定 (n) 个数,依次输出相邻俩数的乘积。 按照题意模拟即可。 神奇的代码 给定一个由 wbwbwwbwbwbw 无限拼接的字符串。 给定 (w,b) ,问是否由一个子串,其有 (w) 个 w , (b) 个 b 考虑枚举子串的起点,其实只有 (len(wbwbwwbwbwbw)=12) 种情况。 枚举起点后,统计其后的 (w+b

    2024年03月24日
    浏览(9)
  • AtCoder Beginner Contest 348

    给定 (n) ,输出 (ooxooxoox...) ,长度为 (n) 。 按题意模拟即可。 神奇的代码 给定 (n) 个点,对每个点,求出与其距离最远的点的下标。 (n) 只有 (100) ,对于每个点,花 (O(n)) 遍历每个点最大化距离,时间复杂度为 (O(n^2)) 。 神奇的代码 (n) 个豌豆,每个豌豆有美味值

    2024年04月08日
    浏览(8)
  • AtCoder Beginner Contest 347

    给定 (n) 个数 (a_i) 以及 (k) ,输出是 (k) 的倍数的 (a_i) 整除以 (k) 的值。 按照题意判断取模和求整除即可。 神奇的代码 给定字符串 (s) ,问子串种类数。 (|s|) 只有 (100) ,直接 (O(n^2)) 枚举子串丢进 (set) 里,答案就是 (set) 的大小。 长度大的话得后缀自动机。

    2024年04月08日
    浏览(6)
  • AtCoder Beginner Contest 344

    给定一个字符串,包含两个 | ,将 | 和两个 | 之间的字符消去。 按照题意模拟即可。 Python 比较简洁。 神奇的代码 给定 (n) 个数,倒序输出。 储存这 (n) 个数,然后倒着输出即可。 神奇的代码 给定三个数组 (a,b,c) ,回答 (q) 次询问。 每次询问,给定 (x) ,问能否从三

    2024年03月10日
    浏览(22)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包