AtCoder Beginner Contest 148 题解

前言

包含题目:

ABC148 A ABC148 B ABC148 C ABC148 D ABC148 E ABC148 F

这次比赛坑好多啊(虽然水了),也可能是我的实现方法不对qaq。

A - Round One

题意

你有三个数:\(1, 2, 3\),现在其中两个数不对,输出对的那个数。

做法

建立一个集合,删去坏的,输出剩下的一个。

用三个if语句判断即可。

程序

考场做法:

#include<bits/stdc++.h>
using namespace std; set<int> t={1,2,3};
int a,b; int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0); cin>>a>>b;
t.erase(a);
t.erase(b);
cout<<*t.begin()<<endl; return 0;
}

真实做法:

#include<bits/stdc++.h>
using namespace std; int a,b; int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
//三行头文件,可以优化cin、cout速度(但仅能聊以慰籍) cin>>a>>b;
if(a>b)swap(a,b);//swap用作交换两个同一类型的变量的值,此处确保a<b
if(a==1&&b==2)cout<<3;
if(a==1&&b==3)cout<<2;
if(a==2&&b==3)cout<<1; return 0;
}

B - Strings with the Same Length

题意

给你长度\(N\)的字符串\(S\)和\(T\),让你把它们合并起来,方法如下:

拿\(S\)的开头字符,删去,添加到答案串的末尾。拿\(T\)的开头字符,删去,添加到答案串的末尾。

重复以上操作\(N\)次,此时\(S\)和\(T\)应该是空串。

做法

暴力模拟即可,不过不用真的删去字符,从前往后跑就可以了。

程序

#include<bits/stdc++.h>
using namespace std; int n;
string s,t,ans; int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0); cin>>n>>s>>t;
for(int i=0;i<n;i++){//从前往后跑s和t的字符
ans+=s[i];
ans+=t[i];//向ans的末尾添加字符
}
cout<<ans<<endl; return 0;
}

C - Snack

题意

给你\(A\)和\(B\),让你求一个数,使得它能被\(A\)和\(B\)整除吗,同时最小(也就是最小公倍数)。

做法

使用辗转相除法先求出最大公因数,对于\(A\)和\(B\),它们的递归的最大公因数求法\(f(A,B)\)如下:

(保证它们都是正整数

  • \(B\)为\(0\),此时返回\(A\)。
  • 返回\(f(B,A\mod B)\)。

\(A \times B \div f(A,B)\)就是最小公倍数。

程序

#include<bits/stdc++.h>
using namespace std; int gcd(int a,int b){//用前文的方法求最大公因数
if(b==0)return a;
else return gcd(b,a%b);
} int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0); int a,b;
cin>>a>>b;
cout<<((long long)(a/gcd(a,b)))*b<<endl;//要用long long,不然会爆int return 0;
}

D - Brick Break

题意

给你\(N\)个砖块,每个砖块写有数字\(a_i\),问你删去一些砖块(最多删去\(N-1\)块)后,剩下的\(K\)块砖块,是一个形如\(1,2,3,\dots,K\)的序列时,\(K\)最大是多少,如果不能有这样的序列,那么就输出-1

做法

其实就是求一个形如\(1,2,3,\dots,K\)的\(a_i\)的子序列(子序列的定义是元素可以不连续的),可以\(O(N)\)完成。

不考虑-1,只求出\(K\)最大是多少,那么如果\(K=0\),就输出-1,否则输出\(K\)。

\(K\)的求法:

记录当前的序列的最后一个的值\(c\)(也就是序列的长度)(初始为\(0\)),从前往后跑\(a_i\)中的元素,如果是\(c+1\),那么就把\(c\)加一。这样最后它就是\(K\)了。

程序

#include<bits/stdc++.h>
using namespace std; int n;
int a[200005]; int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0); cin>>n;
int c=1;//此处c的值是当前序列的最后一个的值加一,当时写的时候太急,失误了
for(int i=0;i<n;i++){
cin>>a[i];
if(a[i]==c){//如同做法中描述的,是序列长度加一时,就增加序列长度
c++;
}
}
cout<<(n-c+1==n?-1:n-c+1)<<endl;//序列长度为c-1,那么n-c+1就表示删去的东西的个数 return 0;
}

E - Double Factorial

题意

给你一个数\(n\)。

定义一个函数\(f(n)\):

  • \(f(n)=1\) (if \(n<2\))
  • \(f(n)=nf(n−2)\) (if \(n≥2\))

输出函数值的末尾的0的个数。

做法

很容易就可以看出,后缀零的个数只和函数值\(2\)和\(5\)因子的个数有关。

由于\(f(n)\)只要求了\(f(n-2)\),所以奇偶性肯定不会变的,\(n\)为奇数时不可能包含\(2\)作为因数,特判一下。

我们再来看偶数,那么我们可以这么计算\(2\)的因子的个数:

\(i\)从\(1\)开始递增,查看包含\(2^i\)作为因子的数的个数(也就是\(\lfloor { \frac{n}{2^i} } \rfloor\),此处\(\lfloor a \rfloor\)指不超过\(a\)的最大整数,也就是下取整),把函数值中的\(2\)的因子个数加上这个个数,你可能会问了,它包含的因数不应该是\(\lfloor { \frac{n}{2^i} } \rfloor \times i\)吗?其实,里面的\(\lfloor { \frac{n}{2^i} } \rfloor \times ( i - 1 )\),已经在\(\lfloor { \frac{n}{2^{i-1}} } \rfloor\)中加过了,所以就是\(\lfloor { \frac{n}{2^i} } \rfloor\)就可以了。

对于\(5\)的因数,我们可以大致使用同样的方法求,但是每次对于答案的贡献是\(\lfloor { \frac{n}{2 \times 5^i} } \rfloor\),因为必须是偶数,所以需要再乘一个\(2\)以确保只计算了偶数。具体原因如下:

我们举个例子,列出\(1,2,3,\dots,25\)的\(5\)的因子的个数:

\[0,0,0,0,1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,2
\]

总和是\(6\)。

那么,只计算偶数的话,就只有这样的序列:

\[N,0,N,0,N,0,N,0,N,1,N,0,N,0,N,0,N,0,N,1,N,0,N,0,N
\]

(\(N\)代表奇数,舍去了)总和是\(2\)。

现在可以理解为什么要乘以\(2\)了吧。

程序

#include<bits/stdc++.h>
using namespace std; typedef long long ll; ll n,c2,c5; int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0); cin>>n;
if(n&1){
cout<<0<<endl;
return 0;
}
for(ll i=2;i<=n;i*=2){
c2+=n/i;
}
for(ll i=10;i<=n;i*=5){
c5+=n/i;
}
cout<<min(c2,c5)<<endl; return 0;
}

F - Playing Tag on Tree

题意

有一棵\(N\)个结点的树,高桥和青木在上面的\(u\)和\(v\)结点上玩抓人游戏。每个回合如下:

  1. 如果高桥和青木在同一个结点上,游戏结束。高桥选一个和当前结点相邻的结点,走过去。
  2. 如果高桥和青木在同一个结点上,游戏结束。青木选一个和当前结点相邻的结点,走过去。
  3. 开始下一个回合

轮到某人走时,不能停留在同一个结点。

高桥希望回合数尽可能多,青木希望回合数尽可能少,两人都采用最优策略

输出游戏结束时青木走的步数

做法

首先,青木的最优策略很好确定,就是走通向的当前高桥的位置的简单路径就可以了。同时,不论高桥的位置怎样变化,青木走的路径都会沿着通向最终会抓到高桥的结点的这条道路,证明很难,我就不证明了。感性地理解,就是由于高桥和青木不会在途中相遇,所以青木走向的那个(以当前青木所在结点为根的)子树肯定包含高桥和游戏结束时的高桥的终点。

高桥的呢,有点麻烦,所以我们就搜索一下吧qwq(暴力、大好き)。

首先,我们预处理出青木到每个结点的距离,之后搜索高桥的路径(深搜、广搜皆可)。对于当前结点是否可以走,只需要满足高桥到这里的距离严格小于青木到这里的距离即可。特殊地,由于只能移动,不能停留在同一个结点,所以到了叶子结点时,高桥会在相邻结点和这个叶子结点之间重复移动,最终抓到的位置一定是在相邻结点上(一共两种情况,自己手推一下就好了)。

程序

#include<bits/stdc++.h>
using namespace std; int n,ta,ao,ans;
vector<int> g[100005];
int dis[100005]; void dfs(int x,int p,int d){
if(d>=dis[x]){//如果被抓到了就退出
ans=max(ans,dis[x]);
return;
}
for(int i=0;i<g[x].size();i++){
int &y=g[x][i];
if(y!=p){
dfs(y,x,d+1);
}
}
if(g[x].size()==1){//是叶子结点,就特判一下之前做法中提到的情况
ans=max(ans,dis[x]-1);
}
} int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0); cin>>n>>ta>>ao;
for(int i=1;i<n;i++){
int a,b;
cin>>a>>b;
g[a].push_back(b);
g[b].push_back(a);
}
memset(dis,-1,sizeof(dis));
dis[ao]=0;
{//预处理青木到每个结点的距离dis
queue<int> q;
q.push(ao);
while(!q.empty()){
int x=q.front();q.pop();
for(int i=0;i<g[x].size();i++){
int &y=g[x][i];
if(dis[y]==-1){
dis[y]=dis[x]+1;
q.push(y);
}
}
}
}
dfs(ta,-1,0);
cout<<ans<<endl; return 0;
}

结束语

感谢观看,感觉自己写得很high,完全没有考虑到别人的观感qaq。所以,有什么意见就评论吧<3<3<3,我一定会改的qaq。

AtCoder Beginner Contest 148 题解的更多相关文章

  1. AtCoder Beginner Contest 154 题解

    人生第一场 AtCoder,纪念一下 话说年后的 AtCoder 比赛怎么这么少啊(大雾 AtCoder Beginner Contest 154 题解 A - Remaining Balls We ...

  2. AtCoder Beginner Contest 153 题解

    目录 AtCoder Beginner Contest 153 题解 A - Serval vs Monster 题意 做法 程序 B - Common Raccoon vs Monster 题意 做 ...

  3. AtCoder Beginner Contest 177 题解

    AtCoder Beginner Contest 177 题解 目录 AtCoder Beginner Contest 177 题解 A - Don't be late B - Substring C ...

  4. AtCoder Beginner Contest 184 题解

    AtCoder Beginner Contest 184 题解 目录 AtCoder Beginner Contest 184 题解 A - Determinant B - Quizzes C - S ...

  5. AtCoder Beginner Contest 173 题解

    AtCoder Beginner Contest 173 题解 目录 AtCoder Beginner Contest 173 题解 A - Payment B - Judge Status Summ ...

  6. AtCoder Beginner Contest 172 题解

    AtCoder Beginner Contest 172 题解 目录 AtCoder Beginner Contest 172 题解 A - Calc B - Minor Change C - Tsu ...

  7. AtCoder Beginner Contest 169 题解

    AtCoder Beginner Contest 169 题解 这场比赛比较简单,证明我没有咕咕咕的时候到了! A - Multiplication 1 没什么好说的,直接读入两个数输出乘积就好了. ...

  8. AtCoder Beginner Contest 151 题解报告

    总的来说,这次的题目比较水,然而菜菜的我并没有把所有题目都做完,话不多说,直接来干货: A:Next Alphabet 题目链接:https://atcoder.jp/contests/abc151/ ...

  9. AtCoder Beginner Contest 115 题解

    题目链接:https://abc115.contest.atcoder.jp/ A Christmas Eve Eve Eve 题目: Time limit : 2sec / Memory limit ...

随机推荐

  1. 七、Hadoop3.3.1 HA 高可用集群QJM (基于Zookeeper,NameNode高可用+Yarn高可用)

    目录 前文 Hadoop3.3.1 HA 高可用集群的搭建 QJM 的 NameNode HA Hadoop HA模式搭建(高可用) 1.集群规划 2.Zookeeper集群搭建: 3.修改Hadoo ...

  2. Web Api 宿主的搭建

    首先我们要清楚一个概念,宿主.宿主是什么意思?先从了解一下Hosting开始吧! 有关Hosting的基础知识 Hosting是一个非常重要,但又很难翻译成中文的概念.翻译成:寄宿,大概能勉强地传达它 ...

  3. 如何用three.js搭建处理3D园区、3D楼层、3D机房管线(机房升级版)-第九课(二)

    接着上一篇文章,<如何用webgl(three.js)搭建处理3D园区.3D楼层.3D机房管线问题(机房升级版)-第九课(一)> 继续讲解关于三维数据中心管线可视化的解决方案. 上一篇我们 ...

  4. 【豆科基因组】普通豆/菜豆/四季豆Common bean (Phaseolus vulgaris L.) 基因组

    目录 研究一:G19833组装,2014NG 研究二:BAT 93组装,2016 genome biology 菜豆属(Phaseolus L.)为同源二倍体作物,包含有80 多个物种,多数为野生种, ...

  5. 重测序(RADseq)做群体遗传分析套路

    实验材料 构建的群体,或自然群体,如各地方品种. RAD文库构建 提取DNA后,构建文库,简要步骤如下: ① 限制性内切酶TaqI酶切: ② 连接P1接头: ③ DNA随机打断片断化: ④ 目的片段回 ...

  6. 毕业设计之dns搭建:

    [apps@dns_sever ~]$ sudo yum install -y bind [apps@dns_sever ~]$ sudo vim /etc/named.conf // // name ...

  7. windows下typora图片自动上传到图床(附带腾讯云教程)

    目录 1. 背景 2. 在Typora中使用 3. 安装 3.1 安装教程环境说明 3.2 安装步骤 3.3 配置图床 3.3.1 插件配置 3.3.2 腾讯云参数获取 3.3.2.1 新建子用户,获 ...

  8. 安装octave详解

    1. 一些可以替换的库(可跳过) 默认的库安装libblas.dll.OpenBLAS-v2.6.0-0-54e7b37_dynamicarch_nt4(自动检测CPU类型) 在目录下<your ...

  9. 解决CentOS7 docker容器映射端口只监听ipv6的问题

    问题现象 docker容器起来以后,查看9100端口监听情况,如下图: $ ss -lntp State Recv-Q Send-Q Local Address:Port Peer Address:P ...

  10. 巩固javaweb第十五天

    巩固内容: 单选按钮: 在注册功能中,用户选择学历使用的是单选按钮,并且是多个单选按钮,每个选项对 应一个单选按钮,用户只能选择其中一个,这多个单选按钮的格式相同.如果用户要输入 的信息只有少数几种可 ...