AgOH

Codeforces Round #824 (Div. 2) 题解

字数统计: 341阅读时长: 1 min
2022/10/04 44

A. Working Week

直接输出 n/32n/3-2 即可。

Code
#include <iostream>
#include <chrono>
using namespace std;
//==========================================
#include <vector>
typedef long long ll;
signed main(signed argc, char const *argv[])
{
#if defined(LOCAL) || defined(DEBUG)
freopen("in.in", "r", stdin);
#ifndef DEBUG
freopen("out.out", "w", stdout);
auto c1 = chrono::high_resolution_clock::now();
#endif
#endif
ios::sync_with_stdio(false);
cin.tie(nullptr);
//======================================
int t;
cin>>t;
while(t--)
{
int n;
cin>>n;
cout<<n/3-2<<endl;
}
//======================================
#ifdef LOCAL
auto c2 = chrono::high_resolution_clock::now();
cerr<<"Time Used:"<<chrono::duration_cast<chrono::milliseconds>(c2-c1).count()<<"ms"<<endl;
#endif
return 0;
}

B. Tea with Tangerines

找到最小一片橘子皮,设其大小为 xx,则其他橘子皮(设大小为 yy)对答案的贡献为 y2x11\lceil \cfrac{y}{2x-1} \rceil - 1

Code
#include <iostream>
#include <chrono>
using namespace std;
//==========================================
#include <vector>
#include <algorithm>
signed main(signed argc, char const *argv[])
{
#if defined(LOCAL) || defined(DEBUG)
freopen("in.in", "r", stdin);
#ifndef DEBUG
freopen("out.out", "w", stdout);
auto c1 = chrono::high_resolution_clock::now();
#endif
#endif
ios::sync_with_stdio(false);
cin.tie(nullptr);
//======================================
int t;
cin>>t;
while(t--)
{
int n;
cin>>n;
vector<int> v(n);
for(int i=0;i<n;i++) cin>>v[i];
int cnt = 0;
for(size_t i=1;i<v.size();i++) cnt += (v[i]-1)/(v.front()*2-1);
cout<<cnt<<endl;
}
//======================================
#ifdef LOCAL
auto c2 = chrono::high_resolution_clock::now();
cerr<<"Time Used:"<<chrono::duration_cast<chrono::milliseconds>(c2-c1).count()<<"ms"<<endl;
#endif
return 0;
}

C. Phase Shift

遍历字符串中的每个字母 c\texttt{c},若已知其前方字母则直接输出,否则贪心地选择尽量小、没有后项、且与 c\texttt{c} 连接后不会成一个大小小于 2626 的环的字母作为其前方字母。使用两个哈希表维护一个字母的前后项,判环可以使用并查集维护。

Code
#include <iostream>
#include <chrono>
using namespace std;
//==========================================
#include <string>
#include <vector>
#include <numeric>
#include <functional>
#include <unordered_map>
signed main(signed argc, char const *argv[])
{
#if defined(LOCAL) || defined(DEBUG)
freopen("in.in", "r", stdin);
#ifndef DEBUG
freopen("out.out", "w", stdout);
auto c1 = chrono::high_resolution_clock::now();
#endif
#endif
ios::sync_with_stdio(false);
cin.tie(nullptr);
//======================================
int t;
cin>>t;
while(t--)
{
int n;
cin>>n;
string s;
cin>>s;
unordered_map<char,char> frt, nxt;
vector<int> fa(150);
iota(fa.begin(), fa.end(), 0);
function<int(int)> find = [&](int x) { return fa[x]==x?x:fa[x]=find(fa[x]); };
int cnt = 26;
for(auto c : s)
{
if(frt[c]) cout<<frt[c];
else
{
for(char p='a';p<='z';p++)
{
if(nxt[p]) continue;
if(cnt>1&&find(p)==find(c)) continue;
fa[find(p)] = find(c);
nxt[p] = c;
frt[c] = p;
cout<<p;
cnt--;
break;
}
}
}
cout<<endl;
}
//======================================
#ifdef LOCAL
auto c2 = chrono::high_resolution_clock::now();
cerr<<"Time Used:"<<chrono::duration_cast<chrono::milliseconds>(c2-c1).count()<<"ms"<<endl;
#endif
return 0;
}

D. Meta-set

一个 Meta-set 中必定包括两个好牌组,且两个好牌组只能有一张重复的牌,我们称这张重复的牌为“中心牌”,则若某张牌出现在了 xx 个好牌组中,以其为中心牌的 Meta-set 即有 x(x1)x(x-1) 个。

考虑如何获取 xx 值,我们只需要遍历每个牌对,并求出这两张牌的补集,并将补集的 xx 值加一即可。这个过程可以使用哈希表来维护。

Code
#include <iostream>
#include <chrono>
using namespace std;
//==========================================
#include <vector>
#include <unordered_map>
typedef long long ll;
ll hs(const vector<int>& v)
{
ll sum=0, mul=1;
for(int i=v.size()-1;i>=0;i--,mul*=3)
sum += v[i]*mul;
return sum;
}
unordered_map<ll,ll> mp;
signed main(signed argc, char const *argv[])
{
#if defined(LOCAL) || defined(DEBUG)
freopen("in.in", "r", stdin);
#ifndef DEBUG
freopen("out.out", "w", stdout);
auto c1 = chrono::high_resolution_clock::now();
#endif
#endif
ios::sync_with_stdio(false);
cin.tie(nullptr);
//======================================
int n,k;
cin>>n>>k;
vector<vector<int>> cards(n, vector<int>(k));
for(int i=0;i<n;i++)
for(int j=0;j<k;j++)
cin>>cards[i][j];
for(int i=1;i<n;i++)
{
for(int j=0;j<i;j++)
{
vector<int> v(k);
for(int l=0;l<k;l++)
{
if(cards[i][l]==cards[j][l]) v[l] = cards[i][l];
else v[l] = 3-cards[i][l]-cards[j][l];
}
mp[hs(v)]++;
}
}
ll sum = 0;
for(auto& v : cards)
{
auto it = mp.find(hs(v));
if(it==mp.end()) continue;
sum += it->second*(it->second-1)/2;
}
cout<<sum<<endl;
//======================================
#ifdef LOCAL
auto c2 = chrono::high_resolution_clock::now();
cerr<<"Time Used:"<<chrono::duration_cast<chrono::milliseconds>(c2-c1).count()<<"ms"<<endl;
#endif
return 0;
}
CATALOG
  1. 1. A. Working Week
  2. 2. B. Tea with Tangerines
  3. 3. C. Phase Shift
  4. 4. D. Meta-set