2021杭电多校赛2021“MINIEYE杯”中国大学生算法设计超级联赛签到题5题
Posted 小哈里
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了2021杭电多校赛2021“MINIEYE杯”中国大学生算法设计超级联赛签到题5题相关的知识,希望对你有一定的参考价值。
Solved Pro.ID Title Ratio(Accepted / Submitted)
1001 Fall with Fake Problem 3.45%(3/87)
1002 Fall with Soldiers 28.57%(8/28)
1003 Fall with Trees 23.77%(618/2600) 结论,推公式
1004 Link with Balls 43.45%(179/412)
1005 Link with EQ 69.82%(303/434)
1006 Link with Grenade 43.55%(206/473)
1007 Link with Limit 16.21%(490/3022)建图,找环
1008 Smzzl with Greedy Snake 57.28%(527/920) 小模拟
1009 Smzzl with Safe Zone 5.48%(12/219)
1010 Smzzl with Tropical Taste 53.36%(898/1683) 签到,判断
1011 Yiwen with Formula 12.73%(76/597)
1012 Yiwen with Sqc 32.05%(483/1507)统计贡献,差分优化
1003 Fall with Trees
题意:
- 在二维平面坐标系中,给出根节点和左右子节点的坐标,按照规范画k层二叉树并求凸包面积。
- 规范为:子节点横坐标关于父节点对称,高度相等,左子树小于右子树。
思路:
- 2e5个询问,很容易想到是个结论题,所以推公式,需要注意输出的精度。
#include<bits/stdc++.h>
using namespace std;
int main(){
int T; scanf("%d", &T);
while(T--){
int k; scanf("%d", &k);
double px, py, lx, ly, rx, ry;
scanf("%lf%lf%lf%lf%lf%lf",&px,&py,&lx,&ly,&rx,&ry);
double x = (rx-lx)/2, y = py-ly;
double ans = x*y*(4*k-10+3.0/pow(2,k-2));
//cout<<fixed<<setprecision(3)<<ans<<"\\n";
printf("%.3f\\n", ans);
}
return 0;
}
1007 Link with Limit
题意:
- 给出f[1]到f[n], 定义f1[x]=f[x], fn[x]=f[f{n-1}[x]].
- 定义g(x)=sum{fi[x], 1<=i<=∞}, 判断x=[1,n]时g(x)是否满足都相等, x,n<=1e5
思路:
- 开始看错题了,以为g(x)是1-n的,就在想怎么优化预处理出所有的f数组,后来意识到∞才知道方向错了,需要往数学方面想,又绕了一段时间,意识到1e5的数据可能是图论或数据结构。
- 建图,点 x 向点 f(x) 连边,那么迭代的过程实质上是在图上行走的过程。原题实际上问的是每一个环的点权平均值是否相同,使用任意方法找出所有环并求出平均值即可。
- 扔一个好多年前不知道多久没用过的kosaraju板子跑出了所有强连通分量,注意统计的时候需要考虑,如样例“2 1 1”,2是连向1的自环,2虽然是联通分量但是它本身不算环,需要加一条特判一下有没有连向自己的边。同时在比较平均值是否相等的时候,为了防止除法精度的问题,可以采用比较当前与下一个分子分母错位相乘结果的方式。
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int maxn = 2e5+10;
//kosaraju
vector<int>G[maxn], rG[maxn];
vector<int>vs, cmp[maxn];
int vis[maxn], book[maxn], cnt;
void dfs(int u){
if(vis[u])return ;
vis[u] = 1;
for(int to: G[u])dfs(to);
vs.push_back(u);
}
void rdfs(int u){
book[u] = cnt;
cmp[cnt].push_back(u);
for(int to: rG[u])if(!book[to])rdfs(to);
}
int main(){
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int T; cin>>T;
while(T--){
int n; cin>>n;
vs.clear(); cnt = 0;
for(int i = 1; i <= n; i++)G[i].clear(),rG[i].clear(),cmp[i].clear(),vis[i]=book[i]=0;
for(int i = 1; i <= n; i++){
int fi; cin>>fi;
G[i].push_back(fi); rG[fi].push_back(i);
}
for(int i = 1; i <= n; i++)dfs(i);
for(int i = n-1; i >= 0; i--){
if(!book[vs[i]]){
++cnt;
rdfs(vs[i]);
}
}
int ok = 1;
for(int i = 1; i+1 <= cnt; i++){
double sum = 0, sum2=0;
if(cmp[i].size()==1 && G[cmp[i][0]][0]!=cmp[i][0]){//特判
continue;
}
for(int x:cmp[i]) sum += x;
for(int x:cmp[i+1])sum2+=x;
if(sum*cmp[i+1].size()!=sum2*cmp[i].size()){//比较
ok = 0; break;
}
}
if(ok==1)cout<<"YES\\n";
else cout<<"NO\\n";
}
return 0;
}
1008 Smzzl with Greedy Snake
题意:
- 给出n个食物及其坐标,必须按顺序吃掉全部,保证方案存在。
- 从(x,y)出发,沿着d方向,每次可以前进或顺时针逆时针转弯,用时都为1,求构造用时最少的方案。
思路:
- 因为要按顺序吃,所以可以贪心在当前吃完的地方直接直线+转弯+直线的方式吃掉下一个,直接模拟就好。
- 注意到如果当前和下一个要吃的食物是对角线时,当前方向如果与某个方向一致,那需要先走这个方向,实际写的时候可以直接两个都跑一遍选最短即可。
#include<bits/stdc++.h>
using namespace std;
string go(int x, int y, int &d, int sx, int sy){
string s1="";
if(sx<x && d!=3){
if(d==0)s1+='u';
if(d==2)s1+='c';
if(d==1)s1+="uu";
d=3;
}
if(sx>x && d!=1){
if(d==0)s1+='c';
if(d==2)s1+='u';
if(d==3)s1+="uu";
d=1;
}
s1 += string(abs(sx-x), 'f');
if(sy<y && d!=2){
if(d==0)s1+="cc";
if(d==1)s1+='c';
if(d==3)s1+='u';
d=2;
}
if(sy>y && d!=0){
if(d==1)s1+='u';
if(d==2)s1+="uu";
if(d==3)s1+='c';
d=0;
}
s1 += string(abs(sy-y), 'f');
return s1;
}
string go2(int x, int y, int &d, int sx, int sy){
string s1="";
if(sy<y && d!=2){
if(d==0)s1+="cc";
if(d==1)s1+='c';
if(d==3)s1+='u';
d=2;
}
if(sy>y && d!=0){
if(d==1)s1+='u';
if(d==2)s1+="uu";
if(d==3)s1+='c';
d=0;
}
s1 += string(abs(sy-y), 'f');
if(sx<x && d!=3){
if(d==0)s1+='u';
if(d==2)s1+='c';
if(d==1)s1+="uu";
d=3;
}
if(sx>x && d!=1){
if(d==0)s1+='c';
if(d==2)s1+='u';
if(d==3)s1+="uu";
d=1;
}
s1 += string(abs(sx-x), 'f');
return s1;
}
int main(){
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int T; cin>>T;
while(T--){
int x, y, d; cin>>x>>y>>d;
int n; cin>>n;
string ans = "";
for(int i = 1; i <= n; i++){
int sx, sy; cin>>sx>>sy;
int d1=d, d2=d;
string t1= go(x,y,d1,sx,sy);
string t2= go2(x,y,d2,sx,sy);
if(t1.size()<t2.size())ans+=t1,d=d1;
else ans+=t2,d=d2;
x = sx, y = sy;
}
cout<<ans<<"\\n";
}
return 0;
}
1010 Smzzl with Tropical Taste
题意:
- 一个游泳池,开始有V升冰红茶,店主每秒加入qV升,男孩喝每秒pV升,判断男孩能否喝到G升。(G取任意正整数)
思路:
- 只要加入的比喝的快就行,即q>=p。
#include<bits/stdc++.h>
using namespace std;
int main(){
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int T; cin>>T;
while(T--){
double p, q; cin>>p>>q;
if(q>=p)cout<<"N0 M0R3 BL4CK 1CE TEA!\\n";
else cout<<"ENJ0Y YOURS3LF!\\n";
}
return 0;
}
1012 Yiwen with Sqc
题意:
- 给出一个字符串,计算每个字母在每个区间出现次数的平方和。
思路:
- 可以发现不同字母之间没有影响,所以分别考虑每一种字母,比如考虑 ‘a’ , ‘a’ 在字符串中一共出现cnt次,第i次的位置为 p[i](p[0]=0, p[cnt+1]=n+1),记len[i] = p[i+1]-p[i],即两个相邻 ‘a’ 之间的距离。
- 首先固定左端点为1,移动右端点,可以发现答案=sum{k=1->cnt, len[k]*k^2},因为右端点在两个字母间移动时字母数量不变,对答案贡献也不变,可以直接累加。然后固定右端点,移动左端点,可以发现对于p[i]到p[i+1]之间的位置,答案为 sum{ k=i+1->cnt, len[k]*(k-i)^2}。
- 考虑维护两次差分对复杂度进行优化,最后O(n)可以通过。
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int mod = 998244353;
int calc(vector<int>vc){
int ans = 0, per = 0, dif = 0;
for(int i = 1; i < vc.size(); i++){
ans = (1ll*ans+1ll*per*(vc[i]-vc[i-1]))%mod;
dif = (1ll*dif+2ll*vc[i-1]+vc[i]-vc[i-1])%mod;
per = (per+dif)%mod;
}
return ans;
}
int main(){
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int T; cin>>T;
while(T--){
string s; cin>>s; s="1"+s;
int n = s.size()-1;
vector<int>vc[30];
for(int i = 0; i <= 25; i++)vc[i].push_back(0);
for(int i = 1; i <= n; i++)vc[s[i]-'a'].push_back(i);
for(int i = 0; i <= 25; i++)vc[i].push_back(n+1);
int ans = 0;
for(int i = 0; i <= 25; i++)
ans = (ans+calc(vc[i]))%mod;
cout<<ans<<"\\n";
}
return 0;
}
以上是关于2021杭电多校赛2021“MINIEYE杯”中国大学生算法设计超级联赛签到题5题的主要内容,如果未能解决你的问题,请参考以下文章
2021杭电多校赛2021“MINIEYE杯”中国大学生算法设计超级联赛签到题4题
2021杭电多校赛2021“MINIEYE杯”中国大学生算法设计超级联赛签到题5题
2021杭电多校赛2021“MINIEYE杯”中国大学生算法设计超级联赛签到题5题
2021杭电多校赛2021“MINIEYE杯”中国大学生算法设计超级联赛签到题2题