🔵
【ABC348】AtCoder Beginner Contest 348【C++】
コンテスト名
トヨタ自動車プログラミングコンテスト2024#4(AtCoder Beginner Contest 348)
コンテストURL
開催日
2024/04/06 21:00-22:40
A: Penalty Kick
解法
- 問題文通りに実装する
ABC348A.cpp
#include <iostream>
using namespace std;
int main(){
int n;
cin >> n;
for(int i=1; i<=n; i++){
if(i%3==0){
cout << "x";
}else{
cout << "o";
}
}
cout << endl;
return 0;
}
B: Farthest Point
解法
- 全探索
-
pow()
などで小数を用いると誤差が出るのでユークリッド距離の二乗でそのまま計算 - 最大値が更新されたら点の番号も更新する
ABC348B.cpp
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
int main(){
int n;
cin >> n;
vector<int> X(n), Y(n);
for(int i=0; i<n; i++){
cin >> X[i] >> Y[i];
}
vector<int> ans(n);
for(int i=0; i<n; i++){
int maxv = 0;
int num = -1;
for(int j=0; j<n; j++){
int dist = (X[i]-X[j])*(X[i]-X[j]) + (Y[i]-Y[j])*(Y[i]-Y[j]);
if(dist>maxv){
maxv = dist;
num = j+1;
}
}
ans[i] = num;
}
for(int i=0; i<n; i++){
cout << ans[i] << '\n';
}
return 0;
}
C: Colorful Beans
解法
- 各色の「おいしさ」の最小値を
unordered_map<int, int>
で管理する - 「おいしさ」の最小値が最大の色の「おいしさ」を出力する
ABC348C.cpp
#include <iostream>
#include <algorithm>
#include <unordered_map>
using namespace std;
int main(){
int n;
cin >> n;
unordered_map<int, int> M;
int a, c;
for(int i=0; i<n; i++){
cin >> a >> c;
if(!M.count(c)){
M[c] = a;
}else if(a<M[c]){
M[c] = a;
}
}
int maxv = 0;
for(auto p : M){
maxv = max(maxv, p.second);
}
cout << maxv << endl;
return 0;
}
D: Medicines on Grid
解法
- 幅優先探索 (BFS)
- 各マスのエネルギー残量の最大値を
vector<vector<int>>
で記録しながら幅優先探索 - 各マスで記録していたエネルギー残量よりも大きなエネルギー残量が可能であることが判明したら更新する
-
vector<vector<int>>
を更新して当該マスをqueue<pair<int, int>>
にプッシュする
-
- 幅優先探索終了後の
vector<vector<int>>
には各マスについてありうる最大のエネルギー残量が記録されている - 「ゴール地点」のエネルギー残量が 0 以上なら移動可能
- (「薬」を使う回数の最小値を求める問題の場合)
- 「薬」があるマスを頂点としたグラフを考えて「薬
」から「薬i 」まで移動できるかを幅優先探索で記録する(辺を張る)j - 「薬」があるマスを頂点としたグラフにおいて「スタート地点」を始点として「ゴール地点」までの距離を幅優先探索で求める
- 「薬」があるマスを頂点としたグラフを考えて「薬
ABC348D.cpp
#include <iostream>
#include <vector>
#include <algorithm>
#include <queue>
using namespace std;
int main(){
int h, w;
cin >> h >> w;
vector<vector<int>> G(h, vector<int>(w));
char a;
int sx, sy, tx, ty;
for(int i=0; i<h; i++){
for(int j=0; j<w; j++){
cin >> a;
if(a=='#'){
G[i][j] = -1;
}else if(a=='S'){
sx = i;
sy = j;
}else if(a=='T'){
tx = i;
ty = j;
}
}
}
int n;
cin >> n;
int r, c, e;
for(int i=0; i<n; i++){
cin >> r >> c >> e;
r--;
c--;
G[r][c] = e;
}
vector<int> dx = {1, 0, -1, 0}, dy = {0, 1, 0, -1};
queue<pair<int, int>> Q;
vector<vector<int>> energy(h, vector<int>(w, -1));
Q.emplace(sx, sy);
energy[sx][sy] = G[sx][sy];
while(!Q.empty()){
auto [x, y] = Q.front();
Q.pop();
for(int i=0; i<4; i++){
int nx = x + dx[i];
int ny = y + dy[i];
if(nx<0 || nx>h-1 || ny<0 || ny>w-1){
continue;
}
if(G[nx][ny]==-1 || energy[x][y]<1){
continue;
}
if(energy[nx][ny]<G[nx][ny] || energy[nx][ny]<energy[x][y]-1){
energy[nx][ny] = max(G[nx][ny], energy[x][y] - 1);
Q.emplace(nx, ny);
}
}
}
if(energy[tx][ty]>=0){
cout << "Yes" << endl;
}else{
cout << "No" << endl;
}
return 0;
}
E: Minimize Sum of Distances
解法
- 頂点
の重みがi の木の重心を考えるC_i -
は頂点f(x) が重心であるときに最小になるx
-
- 頂点
が重心であるとき頂点v に連なる各部分木について重みの総和はそれぞれ当該部分木以外の重みの総和(全体 - 各部分木)以下であるv
ABC348E.cpp
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
int main(){
int n;
cin >> n;
vector<vector<int>> G(n);
int a, b;
for(int i=0; i<n-1; i++){
cin >> a >> b;
a--;
b--;
G[a].push_back(b);
G[b].push_back(a);
}
vector<int> C(n);
long long int sum = 0;
for(int i=0; i<n; i++){
cin >> C[i];
sum += C[i];
}
int center;
auto dfs = [&](auto dfs, int v, int p=-1) -> long long int {
long long int res = C[v];
long long int maxv = 0;
for(int i=0; i<G[v].size(); i++){
if(G[v][i]==p){
continue;
}
long long int child = dfs(dfs, G[v][i], v);
maxv = max(maxv, child);
res += child;
}
maxv = max(maxv, sum-res);
if(maxv<=sum-maxv){
center = v;
}
return res;
};
dfs(dfs, 0);
long long int ans = 0;
auto dfs2 = [&](auto dfs2, int v, int dist=0, int p=-1) -> void {
ans += (long long int)dist*C[v];
for(int i=0; i<G[v].size(); i++){
if(G[v][i]==p){
continue;
}
dfs2(dfs2, G[v][i], dist+1, v);
}
};
dfs2(dfs2, center);
cout << ans << endl;
return 0;
}
Discussion