Codeforces 1086 简要题解
传送门
这场比赛原地爆炸了啊!!!
只做了两道。
A题
传送门 手贱没关freopenfreopenfreopen于是wawawa了一次,死活调不出错。
题意:给出网格图上三个点坐标,让你求出让三个点连通的最少网格数并且输出任意一种连接的方案。
思路:可以直接上分类讨论。
不过可以人脑减去一些讨论。
我们设最初的坐标为(x1,y1),(x2,y2),(x3,y3)(x_1,y_1),(x_2,y_2),(x_3,y_3)(x1,y1),(x2,y2),(x3,y3)
然后把x,yx,yx,y分别排序变成x1′,x2′,x3′x_1',x_2',x_3'x1′,x2′,x3′和y1′,y2′,y3′y_1',y_2',y_3'y1′,y2′,y3′
显然横向走的长度是x3−x1+1x_3-x_1+1x3−x1+1,我们不妨直接把(x1′,y2′)−>(x3′,y2′)(x_1',y_2')->(x_3',y_2')(x1′,y2′)−>(x3′,y2′)全部染上。
然后就只用考虑纵向的了。
于是我们找到yyy最小的点p(px,py)p(p_x,p_y)p(px,py),把(px,py)−>(px,y1−1)(p_x,p_y)->(p_x,y_1-1)(px,py)−>(px,y1−1)染上,然后类似地去找到yyy最大的点q(qx,qy)q(q_x,q_y)q(qx,qy),把(qx,qy)−>(qx,y1−1)(q_x,q_y)->(q_x,y_1-1)(qx,qy)−>(qx,y1−1)染上就可以满足题意了。
代码:
#include<bits/stdc++.h>
#define ri register int
using namespace std;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
int x[3],y[3],X[3],Y[3];
struct Pot{int x,y;}p[3];
inline bool cmp(const Pot&a,const Pot&b){return a.y==b.y?a.x<b.x:a.y<b.y;}
int main(){
for(ri i=0;i<3;++i)p[i].x=x[i]=read(),p[i].y=y[i]=read();
sort(x,x+3),sort(y,y+3),sort(p,p+3,cmp);
int cnt=0;
vector<pair<int,int> >ans;
for(ri i=x[0];i<=x[2];++i)ans.push_back(make_pair(i,y[1]));
for(ri i=y[0];i<y[1];++i)ans.push_back(make_pair(p[0].x,i));
for(ri i=y[1]+1;i<=y[2];++i)ans.push_back(make_pair(p[2].x,i));
cout<<ans.size()<<'\n';
for(ri i=0;i<ans.size();++i)cout<<ans[i].first<<' '<<ans[i].second<<'\n';
return 0;
}
B题
传送门 简单贪心
题意:给出一棵带权树的边长总和SSS和树的边,让你任意给每条边分配非负边权使得树的直径最小。
思路:我们考虑只赋值给连向叶子的边权值,且每条边的权值都是s叶子数\frac s{叶子数}叶子数s,而其它边都赋值为0.
此时的答案为2s叶子数\frac{2s}{叶子数}叶子数2s,不难证明改变任意一条当前边的权值都无法使答案更优,因此答案就是2s叶子数\frac{2s}{叶子数}叶子数2s
代码:
#include<bits/stdc++.h>
#define ri register int
using namespace std;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
const int N=1e5+5;
int n,s,cnt=0,du[N];
int main(){
n=read(),s=read()<<1;
for(ri i=1;i<n;++i)++du[read()],++du[read()];
for(ri i=1;i<=n;++i)if(du[i]==1)++cnt;
printf("%.10lf",(double)s/cnt);
return 0;
}
C题
传送门 贪心好题
题意:给出三个字符串S,A,BS,A,BS,A,B和字符集大小kkk,问是否存在一种字符集的双射关系使得SSS映射成的新字符串S′S'S′满足A≤S′≤BA\le S'\le BA≤S′≤B,允许输出任意一种方案。
思路:我们根据AAA字符串逐位贪心确定SSS,考虑当前SSS的字符sis_isi和AAA当前的字符aia_iai
- 这个字符之前映射过了,那么直接判断:如果映射值si′<ais_i'<a_isi′<ai显然不满足条件;如果si′=ais_i'=a_isi′=ai就递归到下一位处理;如果si′>ais_i'>a_isi′>ai就直接贪心构造之后的位跟BBB比大小即可。
- 这个字符之前没有映射过,那么先看能否映射成aia_iai,如果可以就递归到下一位;否则看能否映射成一个比aia_iai大的,如果可以就贪心构造之后的位跟BBB比较大小。
代码:
#include<bits/stdc++.h>
#define ri register int
using namespace std;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
const int N=1e6+5;
int n,Tran[26],tran[26],a[N],b[N],s[N],ans[N],vis[26],Vis[26],k;
char S[N];
inline void print(){
puts("YES");
for(ri i=0;i<=k;++i)if(Tran[i]==-1)for(ri j=0;j<=k;++j)if(!Vis[j]){Vis[Tran[i]=j]=1;break;}
for(ri i=0;i<=k;++i)printf("%c",(char)(Tran[i]+'a'));
puts("");
}
inline bool update(int st){
for(ri i=0;i<=k;++i)Vis[i]=vis[i],Tran[i]=tran[i];
for(ri i=st;i<=n;++i){
if(Tran[s[i]]==-1)for(ri j=0;j<=k;++j)if(!Vis[j]){Vis[Tran[s[i]]=j]=1;break;}
ans[i]=Tran[s[i]];
}
for(ri i=1;i<=n;++i){
if(ans[i]==b[i])continue;
if(ans[i]<b[i])return print(),true;
return false;
}
return print(),true;
}
inline bool dfs(int pos){
if(pos==n+1)return update(pos);
if(~tran[s[pos]]){
ans[pos]=tran[s[pos]];
if(tran[s[pos]]<a[pos])return false;
if(tran[s[pos]]==a[pos])return dfs(pos+1);
return update(pos+1);
}
bool t;
if(!vis[a[pos]]){
vis[ans[pos]=tran[s[pos]]=a[pos]]=1,t=dfs(pos+1);
if(t)return true;
tran[s[pos]]=-1,ans[pos]=vis[a[pos]]=0;
}
for(ri i=a[pos]+1;i<=k;++i)
if(!vis[i]){
vis[ans[pos]=tran[s[pos]]=i]=1,t=update(pos+1);
if(t)return true;
return tran[s[pos]]=-1,vis[i]=0;
}
return false;
}
int main(){
for(ri tt=read();tt;--tt){
k=read()-1;
scanf("%s",S+1),n=strlen(S+1);
for(ri i=1;i<=n;++i)s[i]=S[i]-'a';
scanf("%s",S+1);
for(ri i=1;i<=n;++i)a[i]=S[i]-'a';
scanf("%s",S+1);
for(ri i=1;i<=n;++i)b[i]=S[i]-'a';
for(ri i=0;i<26;++i)tran[i]=-1,vis[i]=0;
if(!dfs(1))puts("NO");
}
return 0;
}
D题
传送门 思维好题
题意:nnn个人排成一排来猜拳,每个人规定每次出的是石头,剪刀或者布(这个规定可以修改)。允许操作n−1n-1n−1次,每次操作可以选当前剩下的两个相邻的人进行比赛,输了的进行淘汰(如果出的相同你可以自己定输赢),每次修改后问有多少人可能赢得最后胜利。
思路 :分情况讨论废话
- 所有人都出一样的:答案为nnn
- 只出了两种,答案为出较大的人数。
- 三种都有出的,发现对于一个人有没有可能赢,只跟左边,右边能否可以消成都不比自己大的有关,因此我们用setsetset维护一下每种出法最靠左和最靠右的位置然后容斥一下即可。
代码:
#include<bits/stdc++.h>
#define ri register int
using namespace std;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
const int N=2e5+5;
int ans=0,n,q,a[N],bit[N][3];
char s[N];
map<char,int>S;
set<int>pos[3];
inline int lowbit(int x){return x&-x;}
inline void update(int x,int id,int v){for(ri i=x;i<=n;i+=lowbit(i))bit[i][id]+=v;}
inline int query(int x,int id){int ret=0;for(ri i=x;i;i-=lowbit(i))ret+=bit[i][id];return ret;}
set<int>::iterator it;
inline void calc(){
ans=0;
for(ri i=0;i<3;++i){
int x=(i+1)%3,y=(i+2)%3,lx,ly,rx,ry;
ans+=pos[i].size();
if(!pos[i].size()||!pos[x].size())continue;
if(!pos[y].size()){ans-=pos[i].size();continue;}
lx=*pos[x].begin(),rx=*pos[x].rbegin(),ly=*pos[y].begin(),ry=*pos[y].rbegin();
if(ly>lx)ans-=query(ly,i)-query(lx-1,i);
if(ry<rx)ans-=query(rx,i)-query(ry-1,i);
}
}
int main(){
n=read(),q=read(),S['R']=0,S['P']=1,S['S']=2,scanf("%s",s+1);
for(ri i=1;i<=n;++i)a[i]=S[s[i]],pos[a[i]].insert(i),update(i,a[i],1);
calc(),cout<<ans<<'\n';
while(q--){
int p=read();
char t[2];
scanf("%s",t);
pos[a[p]].erase(p),update(p,a[p],-1);
a[p]=S[t[0]];
pos[a[p]].insert(p),update(p,a[p],1);
calc(),cout<<ans<<'\n';
}
return 0;
}
E题
传送门 计数好题
题意简述:定义一个n∗nn*nn∗n矩阵是好的当它满足如下条件:
- 这个矩阵每一行都是111~nnn的排列
- 对于这个矩阵任意相邻两行都不存在一个位置使得这两行的这个位置上的数相同。
现在给你一个好的矩阵问它在所有n∗nn*nn∗n的好的矩阵当中的字典序排名。
n≤2000n\le2000n≤2000
思路:
显然需要从上往下从左往右一个一个算,我们考虑令ansi,jans_{i,j}ansi,j表示按照从上往下从左往右一个一个填充使得到第(i,j)(i,j)(i,j)个格子之前的所有格子都已经跟给出的矩阵相同,而(i,j)(i,j)(i,j)这个格子填的数强制跟原矩阵不同,使得构造出的新矩阵字典序小于原矩阵的方案数。
这个定义看起来比较复杂但理解起来比较简单,我们来看这个图(我画图用得不好请原谅):
我们会发现,我们刚刚定义的状态说的就是红色部分跟原矩阵一样,第(i,j)(i,j)(i,j)号格子上面填的数要比原矩阵的数小的意思。
那么为啥(i,j)(i,j)(i,j)后面的格子分了两种颜色,绿色和蓝色呢?
假设我们已经知道了黄色和蓝色那一坨的总填法数,那么我们可以将第iii行看成一个数列,第i+1i+1i+1行的填法就是nnn个数排列的错排数,而对于第i+1i+1i+1行的每一种情况,第i+2i+2i+2行的填法又是nnn个数排列的错排数…于是后面n−in-in−i行的填法数就是n个数错排数n−in个数错排数^{n-i}n个数错排数n−i
于是我们只需要求出来黄蓝色部分的填法数啦!
考虑到黄色部分其实很好算填法数,我们只用维护一个以权值为下标的树状数组就可以动态统计黄色部分的选法数,关键在于蓝色部分,我们发现好像蓝色部分的选法数跟上面一行有关,仔细分析之后可以发现就是让你求nnn个数有mmm个数强制错排的方案数。
这个东西是可以用dpdpdp解决的。
设fi,jf_{i,j}fi,j表示iii个数jjj个数强制错排的方案数。
于是就相当于iii个数j−1j-1j−1个数强制错排的方案数扣去不合法的方案数,什么时候不合法?有一个本该错排并没有错排,因此这时候剩下i−1i-1i−1个数有j−1j-1j−1个数强制错排,故推导出了fi,j=fi,j−1−fi−1,j−1f_{i,j}=f_{i,j-1}-f_{i-1,j-1}fi,j=fi,j−1−fi−1,j−1
这个时候差不多已经做完了。
只需要对于第(i,j)(i,j)(i,j)号格子讨论它选出来填上去的值出现过几次就行了。
代码:
#include<bits/stdc++.h>
#define ri register int
using namespace std;
typedef long long ll;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
const int mod=998244353,N=2005;
inline int add(int a,int b){return a+b>=mod?a+b-mod:a+b;}
inline int dec(int a,int b){return a>=b?a-b:a-b+mod;}
inline int mul(int a,int b){return (ll)a*b%mod;}
int n,fac[N],mult[N],ans=0,a[N][N],f[N][N],vis[N];
struct Bit{
int bit[N];
Bit(){memset(bit,0,sizeof(bit));}
inline int lowbit(int x){return x&-x;}
inline void update(int x,int v){for(ri i=x;i<=n;i+=lowbit(i))bit[i]+=v;}
inline int query(int x){int ret=0;for(ri i=x;i;i-=lowbit(i))ret+=bit[i];return ret;}
inline void clear(){fill(bit+1,bit+n+1,0);}
}A,B;
int main(){
fac[0]=fac[1]=1,n=read(),mult[0]=1,f[1][0]=1;
for(ri i=2;i<=n;++i){
fac[i]=f[i][0]=mul(fac[i-1],i);
for(ri j=1;j<=i;++j)f[i][j]=dec(f[i][j-1],f[i-1][j-1]);
}
for(ri i=1;i<=n;++i)for(ri j=1;j<=n;++j)a[i][j]=read();
for(ri i=1;i<=n;++i)mult[i]=mul(mult[i-1],f[n][n]);
A.clear();
for(ri i=1;i<=n;++i)ans=add(ans,mul(fac[n-i],a[1][i]-1-A.query(a[1][i]-1))),A.update(a[1][i],1);
ans=mul(ans,mult[n-1]);
for(ri i=2,sum;i<=n;++i){
sum=0,A.clear(),B.clear(),fill(vis+1,vis+n+1,0);
for(ri x,y,t,j=n;j;--j){
if(++vis[a[i][j]]==2)B.update(a[i][j],1);
if(++vis[a[i-1][j]]==2)B.update(a[i-1][j],1);
A.update(a[i][j],1),x=B.query(a[i][j]-1),y=A.query(a[i][j]-1)-x,t=B.query(n);
if(a[i-1][j]<a[i][j]&&vis[a[i-1][j]]==2)--x;
if(vis[a[i-1][j]]==2)--t;
sum=add(sum,mul(x,f[n-j][t-1])),sum=add(sum,mul(y,f[n-j][t]));
}
ans=add(ans,mul(sum,mult[n-i]));
}
cout<<ans;
return 0;
}
Codeforces 1086 简要题解的更多相关文章
- Codeforces 863 简要题解
文章目录 A题 B题 C题 D题 E题 F题 G题 传送门 简要题解?因为最后一题太毒不想写了所以其实是部分题解... A题 传送门 题意简述:给你一个数,问你能不能通过加前导000使其成为一个回文数 ...
- Codeforces 381 简要题解
做的太糟糕了...第一题看成两人都取最优策略,写了个n^2的dp,还好pre-test良心(感觉TC和CF的pretest还是很靠谱的),让我反复过不去,仔细看题原来是取两边最大的啊!!!前30分钟就 ...
- Codeforces 1120 简要题解
文章目录 A题 B题 C题 D题 E题 F题 传送门 A题 传送门 题意简述:给你一个mmm个数的数列,现在规定把一个数列的1,2,...,k1,2,...,k1,2,...,k分成第一组,把k+1, ...
- Codeforces 1098 简要题解
文章目录 前言 A题 B题 C题 D题 E题 传送门 前言 没错因为蒟蒻太菜了这场的最后一道题也咕掉了,只有AAA至EEE的题解233 A题 传送门 题意简述:给出一棵带点权的树,根节点深度为111, ...
- Codeforces 1110 简要题解
文章目录 A题 B题 C题 D题 E题 F题 G题 传送门 众所周知ldxoildxoildxoi这种菜鸡选手是不会写HHH题的,因此该篇博客只有AAA题至GGG题的题解,实在抱歉. A题 传送门 题 ...
- Codeforces 380 简要题解
ABC见上一篇. 感觉这场比赛很有数学气息. D: 显然必须要贴着之前的人坐下. 首先考虑没有限制的方案数.就是2n - 1(我们把1固定,其他的都只有两种方案,放完后长度为n) 我们发现对于一个限制 ...
- Codeforces 845 简要题解
文章目录 A题 B题 C题 D题 E题 F题 G题 传送门 A题 传送门 题意:2n2n2n个人下棋,分为两个阵营,每个阵营nnn个人,每个人有一个积分,积分高的能赢积分低的,问如果你可以随意选人,然 ...
- Codeforces 1065 简要题解
文章目录 A题 B题 C题 D题 E题 F题 G题 传送门 GGG题略难,膜了一波zhouyuyang{\color{red} zhouyuyang}zhouyuyang巨佬的代码. 其余都挺清真的. ...
- Codeforces 888 简要题解
文章目录 A题 B题 C题 D题 E题 F题 G题 传送门 A题 传送门 题意简述:给一个数列,问有多少个峰值点(同时比两边都大/小的点) 思路:按照题意模拟. 代码: #include<bit ...
随机推荐
- Python+Selenium学习--浏览器设置
场景 设置浏览器窗口的大小有下面两个比较常见的用途: 在统一的浏览器大小下运行用例,可以比较容易的跟一些基于图像比对的工具进行结合,提升测试的灵活性及普遍适用性.比如可以跟sikuli结合,使用sik ...
- php中bootstrap框架.popover弹出框,鼠标移动到上面自动显示,离开自动消失
<div rel="name"></div> <script> $(function(){//显示弹出框 $("[rel=name]& ...
- Win10 安装 Anaconda3 用 Anaconda3 安装TensorFlow 1.2 (只支持python3.5)
Win10 安装 Anaconda3 1.安装Anaconda3 选择相应的Anaconda进行安装,下载地址点击这里,下载对应系统版本的Anaconda,官网现在的版本是Anaconda 4.3.1 ...
- HDU 1754 I Hate It(线段树区间查询,单点更新)
描述 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感.不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老 ...
- python中类变量和成员变量、局部变量总结
class Member(): num= #类变量,可以直接用类调用,或用实例对象调用 def __init__(self,x,y): self.x=x #实例变量(成员变量),需要它是在类的构造函数 ...
- iOS 开发实用工具
史蒂芬的博客 (各种软件) http://www.sdifen.com/ 产品原型设计工具 -- 1.墨刀 2.Axure RP 检测接口工具 ---- 1.Charles 2. postman607 ...
- Debug的使用
1.什么是Debug? Debug是DOS.Windows都提供的实模式(8086方式)程序的调试工具.使用它,可以查看CPU各种寄存器的内容.内存的情况和在机器码级跟踪程序的运行. 我用的是Mac机 ...
- C#执行javascript代码,执行复杂的javascript代码新方式
1. 使用nuget 包"Jurassic", 注意,如果 nuget上的包 用起来出现错误,请自行下载 github代码,自行编译最新代码成dll,再引用. 官方的nuget包 ...
- if 循环的深入理解 哈希表的一种应用
哈希表的值作为一个颜色容器,值默认为标识1, 表示未曾用过,若用过标识为0: 1: 程序第一步 遍历哈希表,查找标识为1 未曾用过的颜色 我用了这个: string colorno_us ...
- JAVA动手动脑及课后作业
1.查看其输出结果.如何解释这样的输出结果?从中你能总结出什么? 运行结果 true true false 原因 1)在Java中,内容相同的字串常量(“Hello”)只保存一份以节约内存,所以s0, ...