AtCoder Beginner Contest 267 解题报告

A - Saturday

题意 :输入字符串 代表 周一至周五的某一天,输出这一天离周六还有多少天
分析 :映射一下,直接输入输出

ac代码:

#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <queue>
#include <map>
#include <vector>
#include <stack>
#include <set>
#include <sstream>
#include <fstream> 
#include <cmath>
#include <iomanip>
#include <bitset>
#include <unordered_map>
#include <unordered_set>
#include <random>
//#pragma GCC optimize(3)
#define x first
#define y second
#define ios ios::sync_with_stdio(false),cin.tie(0);
#define endl '\n'
#define pb push_back
#define all(x) x.begin(),x.end()
#define all1(x) x.begin()+1,x.end()

using namespace std;
 
typedef unsigned long long uLL;
typedef long long LL;
typedef pair<int,int> PII;
 
const int N = 2000010,M = 10010,INF = 0x3f3f3f3f,mod = 1e9 + 7;
const double INFF = 0x7f7f7f7f7f7f7f7f,pi = acos(-1.0),eps = 1e-6;


int n,m,k,t;

int main()
{
	ios;
	map<string,int> mp;
	mp["Monday"] = 5;
	mp["Tuesday"] = 4;
	mp["Wednesday"] = 3;
	mp["Thursday"] = 2;
	mp["Friday"] = 1;
	string s;
	cin >> s;
	cout << mp[s] << endl;
	return 0;
}

B - Split?

题意 :有7列保龄球瓶,现在告诉你哪几个瓶倒了,问是否有以下情况发生:
1.编号为1的瓶倒了
2.有两列至少有一个瓶站着,且这两列中间有一列全倒了
分析 :模拟 + 暴力判断

image

ac代码

#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <queue>
#include <map>
#include <vector>
#include <stack>
#include <set>
#include <sstream>
#include <fstream> 
#include <cmath>
#include <iomanip>
#include <bitset>
#include <unordered_map>
#include <unordered_set>
#include <random>
//#pragma GCC optimize(3)
#define x first
#define y second
#define ios ios::sync_with_stdio(false),cin.tie(0);
#define endl '\n'
#define pb push_back
#define all(x) x.begin(),x.end()
#define all1(x) x.begin()+1,x.end()

using namespace std;
 
typedef unsigned long long uLL;
typedef long long LL;
typedef pair<int,int> PII;
 
const int N = 2000010,M = 10010,INF = 0x3f3f3f3f,mod = 1e9 + 7;
const double INFF = 0x7f7f7f7f7f7f7f7f,pi = acos(-1.0),eps = 1e-6;


int n,m,k,t;

int main()
{
	ios;
	int id[] = {0,4,3,5,2,4,6,1,3,5,7};
	string s;
	cin >> s;
	s = ' ' + s;
	if(s[1] == '1') cout << "No" << endl;
	else
	{
		int a[] = {0,1,1,2,2,2,1,1};
		for(int i = 1;i <= 10;i ++)
		{
			if(s[i] == '0')
			{
				a[id[i]] --;
			}
		}

		bool success = false;
		for(int i = 1;i <= 7;i ++)
			for(int j = i + 1;j <= 7;j ++)
			if(a[i] && a[j])
			{
				int tp = 0;
				for(int k = i + 1;k < j;k ++) if(!a[k]) tp ++;
				if(tp) success = true;
			}
		if(success) cout << "Yes" << endl;
		else cout << "No" << endl;

	}
	return 0;
}

C - Index × A(Continuous ver.)

题意 : 给定一个数组a,求a的一个长度为m的连续子序列b,使 \(\sum_1^m i * b_i\) 最大
分析 :前缀和 + 暴力枚举,求出\(i * a_i\)\(a_i\) 的前缀和s 和 a,当b以i为开头时,所求答案为\(s_{i + m - 1} - s_{i - 1} - (i - 1) * a_{i - 1}\)
ac代码

#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <queue>
#include <map>
#include <vector>
#include <stack>
#include <set>
#include <sstream>
#include <fstream> 
#include <cmath>
#include <iomanip>
#include <bitset>
#include <unordered_map>
#include <unordered_set>
#include <random>
//#pragma GCC optimize(3)
#define x first
#define y second
#define ios ios::sync_with_stdio(false),cin.tie(0);
#define endl '\n'
#define pb push_back
#define all(x) x.begin(),x.end()
#define all1(x) x.begin()+1,x.end()

using namespace std;
 
typedef unsigned long long uLL;
typedef long long LL;
typedef pair<int,int> PII;
 
const int N = 2000010,M = 10010,INF = 0x3f3f3f3f,mod = 1e9 + 7;
const double INFF = 0x7f7f7f7f7f7f7f7f,pi = acos(-1.0),eps = 1e-6;


int n,m,k,t;

int main()
{
	ios;
	cin >> n >> m;
	vector<LL> a(n + 1),b(n + 1);
	for(int i = 1;i <= n;i ++) cin >> a[i],b[i] = b[i - 1] + 1LL * a[i] * i,a[i] += a[i - 1];

	LL ans = - 1e18;
	for(int i = m;i <= n;i ++)
	{
		ans = max(ans,b[i] - b[i - m] - (a[i] - a[i - m]) * (i - m));
	}
	cout << ans << endl;
	return 0;
}

D - Index × A(Not Continuous ver.)

题意 : 与C题意相同 但 求的b是a的子序列
分析 : 考虑dp,从\(a_i\) 选与不选考虑(01背包)
\(f(i,j) 表示 : 前i个物品中选了j个所能得到的最大值\)
\(不选 : f(i,j) = f(i - 1,j)\)
\(选 : f(i,j) = f(i - 1,j - 1) + j * a_i\)
坑点 :因为\(a_i\)可能为负,f数组的赋初值1e18不够小
ac代码

#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <queue>
#include <map>
#include <vector>
#include <stack>
#include <set>
#include <sstream>
#include <fstream> 
#include <cmath>
#include <iomanip>
#include <bitset>
#include <unordered_map>
#include <unordered_set>
#include <random>
//#pragma GCC optimize(3)
#define x first
#define y second
#define ios ios::sync_with_stdio(false),cin.tie(0);
#define endl '\n'
#define pb push_back
#define all(x) x.begin(),x.end()
#define all1(x) x.begin()+1,x.end()
 
using namespace std;
 
typedef unsigned long long uLL;
typedef long long LL;
typedef pair<int,int> PII;
 
const int N = 2000010,M = 10010,INF = 0x3f3f3f3f,mod = 1e9 + 7;
const double INFF = 0x7f7f7f7f7f7f7f7f,pi = acos(-1.0),eps = 1e-6;
 
 
int n,m,k,t;
LL f[2010][2010];
int main()
{
	ios;
	cin >> n >> m;
	vector<int> a(n + 1);
	for(int i = 1;i <= n;i ++) cin >> a[i];
	memset(f,-0x3f,sizeof f);
	for(int i = 0;i <= n;i ++) f[i][0] = 0;
 	for(int i = 1;i <= n;i ++)
	{
		for(int j = 1;j <= min(i,m);j ++)
		{
			f[i][j] = f[i - 1][j];
			f[i][j] = max(f[i][j],f[i - 1][j - 1] + 1LL * j * a[i]);
		}
	}
	cout << f[n][m] << endl;
	return 0;
}

E - Erasing Vertices 2

题意 :给定一个无向图带有n个点m条边,每个点都有权值。每次要删除一条边,一次删除的代价是从这个点直接相邻的点的权值和,使n次删除操作后的最大值最小,求其最小值。
分析 : 最大值最小问题,经典二分思路,二分答案,check函数模拟删点看能否删完所有点
ac代码

#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <queue>
#include <map>
#include <vector>
#include <stack>
#include <set>
#include <sstream>
#include <fstream> 
#include <cmath>
#include <iomanip>
#include <bitset>
#include <unordered_map>
#include <unordered_set>
#include <random>
//#pragma GCC optimize(3,"Ofast","inline")
#define x first
#define y second
#define ios ios::sync_with_stdio(false),cin.tie(0);
#define endl '\n'
#define pb push_back
#define all(x) x.begin(),x.end()
#define all1(x) x.begin()+1,x.end()
 
using namespace std;
 
typedef unsigned long long uLL;
typedef long long LL;
typedef pair<int,int> PII;
 
const int N = 200010,M = 10010,INF = 0x3f3f3f3f,mod = 1e9 + 7;
const double DNF = 0x7f7f7f7f7f7f7f7f,pi = acos(-1.0),eps = 1e-6;
const long long LNF = 0x3f3f3f3f3f3f3f3f;
 
int n,m,k,t;
LL a[N];
int h[N],e[N << 1],ne[N << 1],idx;

void add(int a,int b)
{
	e[idx] = b,ne[idx] = h[a],h[a] = idx ++;
}

bool check(LL x)
{
	vector<LL> w(n + 1);
	for(int i = 1;i <= n;i ++)
		for(int j = h[i];~ j;j = ne[j]) 
			w[e[j]] += a[i];

	queue<int> q;
	for(int i = 1;i <= n;i ++) if(w[i] <= x) q.push(i);
	vector<bool> st(n + 1);
	while(q.size())
	{
		int u = q.front();
		q.pop();
		if(st[u]) continue;
		st[u] = true;

		for(int i = h[u];~ i;i = ne[i])
		{
			int j = e[i];
			if(st[j]) continue;
			w[j] -= a[u];
			if(w[j] <= x) q.push(j);
		}
	} 

	for(int i = 1;i <= n;i ++) if(!st[i]) return false;
	return true;
}

int main()
{
	ios;
	cin >> n >> m;
	for(int i = 1;i <= n;i ++) h[i] = -1;
	for(int i = 1;i <= n;i ++) cin >> a[i];

	while(m --)
	{
		int a,b;
		cin >> a >> b;
		add(a,b), add(b,a);
	}

	LL l = -1, r = 1e18;

	while(l < r)
	{
		LL mid = l + r >> 1;
		if(check(mid)) r = mid;
		else l = mid + 1;
	}

	cout << l << endl;
	return 0;
}

F - Exactly K Steps

题意 : 给定一棵树,有若干个询问,每次给定一个节点u和参数k,请随意输出一个和u距离为k的节点,如果不存在,输出-1。
分析 : 树有一个性质:任意一个点的最远点一定是直径的一个端点。所以我们求出树的直径的两个端点,找出离点u最远的点v,如果最远距离 < k ,则输出 -1 ,否则我们利用树上倍增,来在u与v的路径上找出答案点
ac代码

#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <queue>
#include <map>
#include <vector>
#include <stack>
#include <set>
#include <sstream>
#include <fstream> 
#include <cmath>
#include <iomanip>
#include <bitset>
#include <unordered_map>
#include <unordered_set>
#include <random>
//#pragma GCC optimize(3,"Ofast","inline")
#define x first
#define y second
#define ios ios::sync_with_stdio(false),cin.tie(0);
#define endl '\n'
#define pb push_back
#define all(x) x.begin(),x.end()
#define all1(x) x.begin()+1,x.end()
 
using namespace std;
 
typedef unsigned long long uLL;
typedef long long LL;
typedef pair<int,int> PII;
 
const int N = 200010,M = 10010,INF = 0x3f3f3f3f,mod = 1e9 + 7;
const double DNF = 0x7f7f7f7f7f7f7f7f,pi = acos(-1.0),eps = 1e-6;
const long long LNF = 0x3f3f3f3f3f3f3f3f;
 
int n,m,k,t;
int l,r;
int h[N],e[N << 1],ne[N << 1],idx;
int fa[N][18],depth[N];
int dist[N];
void add(int a,int b)
{
	e[idx] = b,ne[idx] = h[a],h[a] = idx ++;
}

void bfs()
{
	queue<int> q;
	for(int i = 1;i <= n;i ++) depth[i] = INF;
	depth[0] = 0,depth[1] = 1;
	q.push(1);

	while(q.size())
	{
		int u = q.front();
		q.pop();

		for(int i = h[u];~ i;i = ne[i])
		{
			int j = e[i];
			if(depth[j] > depth[u] + 1)
			{
				depth[j] = depth[u] + 1;
				fa[j][0] = u;
				q.push(j);

				for(int k = 1;k <= 17;k ++)
					fa[j][k] = fa[fa[j][k - 1]][k - 1];
			}
		}
	}
}

int lca(int a,int b)
{
	if(depth[a] < depth[b]) swap(a,b);

	for(int k = 17;~ k;k --)
		if(depth[fa[a][k]] >= depth[b])
			a = fa[a][k];

	if(a == b) return a;

	for(int k = 17;~ k;k --)
		if(fa[a][k] != fa[b][k])
		{
			a = fa[a][k];
			b = fa[b][k];
		}

	return fa[a][0];
}

void dfs(int u,int fa)
{
	for(int i = h[u];~ i;i = ne[i])
	{
		int j = e[i];
		if(j == fa) continue;
		if(dist[j] > dist[u] + 1) 
		{
			dist[j] = dist[u] + 1;
			dfs(j,u);
		}
	}
}

void find()
{
	for(int i = 1;i <= n;i ++) dist[i] = INF;
	dist[1] = 0;
	dfs(1,0);
	for(int i = 2;i <= n;i ++) if(dist[i] > dist[l]) l = i;
	for(int i = 1;i <= n;i ++) dist[i] = INF;
	dist[l] = 0;
	dfs(l,0);
	for(int i = 1;i <= n;i ++) if(dist[i] > dist[r]) r = i;
}

void solve(int u,int v,int anc,int k)
{
	if(depth[u] - depth[anc] >= k)
	{
		for(int i = 17;~ i;i --) if(k >> i & 1) u = fa[u][i];
	}
	else
	{
		k = depth[u] + depth[v] - 2 * depth[anc] - k;
		u = v;
		for(int i = 17;~ i;i --) if(k >> i & 1) u = fa[u][i];

	}
	
	cout << u << endl;
}

int main()
{
	ios;
	
	cin >> n;
	for(int i = 1;i <= n;i ++) h[i] = -1;
	for(int i = 1;i < n;i ++)
	{
		int a, b;
		cin >> a >> b;
		add(a,b), add(b,a);
	}
	
	find();

	bfs();
	cin >> m;
	while(m --)
	{
		int u, k, v;
		cin >> u >> k;
		int ancl = lca(u,l),ancr = lca(u,r);
		//cout << ancl<< " " << ancr << endl;
		if(depth[u] + depth[l] - 2 * depth[ancl] > depth[u] + depth[r] - 2 * depth[ancr]) v = l;
		else v = r;
		int anc;
		if(v == l) anc = ancl;
		else anc = ancr;
		if(depth[u] + depth[v] - 2 * depth[anc] < k) cout << -1 << endl;
		else solve(u,v,anc,k);
	}

	return 0;
}
posted @ 2022-09-04 17:09  notyour_young  阅读(129)  评论(0编辑  收藏  举报