比赛链接:

https://codeforces.com/contest/1156

A. Inscribed Figures

题意:

给出$n(2\leq n\leq 100)$个数,只含有1,2,3,分别代表圆,高与底相等的三角形,正方形

$a_{i+1}$在$a_{i}$的里面,$a_{i+1}$的面积尽可能的大

求不同的交点个数

分析:

注意正方形里面一个圆,再里面一个三角形的时候,有一个交点重合

ac代码:

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=1e5+10;
const int maxm=200000*2+10;
const int mod=1e9+7;
int num[maxn];
ll ans;
int main()
{
int n;
scanf("%d",&n);
for(int i=1;i<=n;i++)
scanf("%d",&num[i]);
for(int i=2;i<=n;i++)
{
if(num[i]==1)
{
if(num[i-1]==2)ans+=3;
if(num[i-1]==3)ans+=4; }
if(num[i]==2)
{
if(num[i-1]==1)
{
if(i-2>=1&&num[i-2]==3)
ans+=2;
else ans+=3;
}
if(num[i-1]==3)ans=1e18;
}
if(num[i]==3)
{
if(num[i-1]==1)ans+=4;
if(num[i-1]==2)ans=1e18;
}
}
if(ans>=1e17)
cout<<"Infinite"<<endl;
else
{
cout<<"Finite"<<endl;
cout<<ans<<endl;
}
return 0;
}

  

B. Ugly Pairs

题意:

给出一个字符串$s(1\leq |s|\leq 100)$,重排它们,让每对相邻的字符在$ascii$表中不相邻

分析:

将$a,c,e,g...$放一个集合,将$b,d,f,h...$放另一个集合,看它们是否能连线在一起

ac代码:

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=100+10;
const int maxm=200000*2+10;
const int mod=1e9+7;
int num[30];
char word[maxn];
int main()
{
int T;
scanf("%d",&T);
while(T--)
{
getchar();
scanf("%s",word);
int gkd=0;
for(int i=0;word[i];i++)
if((int)word[i]%2!=(int)word[0]%2)gkd=1;
if(gkd==0)
{
printf("%s\n",word);
continue;
}
memset(num,0,sizeof(num));
for(int i=0; word[i]; i++)
num[word[i]-'a']++;
int a,b,fla=0;
for( int i=0; i<26; i++)
for(int j=0; j<26; j++)
if(i%2==0&&j%2==1&&abs(i-j)!=1&&num[j]&&num[i])
{
a=i,b=j;
fla=1;
}
if(fla==0)
{
printf("No answer\n");
continue;
}
num[a]--;
num[b]--;
for(int i=0; i<26; i+=2)
for(int j=0; j<num[i]; j++)
printf("%c",(char)(i+'a'));
printf("%c%c",(char)(a+'a'),(char)(b+'a'));
for(int i=1; i<26; i+=2)
for(int j=0; j<num[i]; j++)
printf("%c",(char)(i+'a'));
printf("\n");
}
return 0;
}

  

C. Match Points

题意:

给出n个数,组成最多的数对,满足$|a_i-a_j|\geq d$

分析:

这题一看就是贪心,但是,我的策略是,用最小的去匹配最小能匹配的,wa

例如:

4 2

1 3 4 7

最优解是$(1,4)(3,7)$

正确匹配方式是,用前面一部分匹配后面一部分,对答案二分即可

ac代码:

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=2e5+10;
const int maxm=200000*2+10;
const int mod=1e9+7;
int n,d,num[maxn];
bool check(int x)
{
int a=x,b=n;
for(;a>=1;a--,b--)
if(num[b]-num[a]<d)return 0;
return 1;
}
int main()
{
scanf("%d %d",&n,&d);
for(int i=1;i<=n;i++)
scanf("%d",&num[i]);
sort(num+1,num+1+n);
int st=0,en=n/2;
while(st!=en)
{
int md=(st+en)/2;
if(check(md+1))st=md+1;
else en=md;
}
printf("%d\n",st);
return 0;
}

D. 0-1-Tree

题意:

给出一颗节点数为$n$的树,每条边有一个属性0或者1

求出点对$(a,b)$的数量,满足$a$到$b$的最短路线上,经过1之后不经过0

分析:

这样的点对有三种情况

1.$(a,b)$的路径只包含0

2.$(a,b)$的路径只包含1

3.$(a,b)$的路径先全是0,再全是1

对于题目的样例有$ans=5\times 5-5+3\times 3-3+4\times 2=34$

用两个并查集记录0和1的集合和大小。

ac代码:

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=200000+10;
const int maxm=200000*2+10;
const int mod=1e9+7;
int boss[maxn][2];
ll siz[maxn][2],ans;
int ma[maxn][2],n;
int fin0(int x)
{
if(boss[x][0]==x)
return x;
return boss[x][0]=fin0(boss[x][0]);
}
int fin1(int x)
{
if(boss[x][1]==x)
return x;
return boss[x][1]=fin1(boss[x][1]);
}
void uni0(int a,int b)
{
boss[fin0(a)][0]=fin0(b);
}
void uni1(int a,int b)
{
boss[fin1(a)][1]=fin1(b);
}
int main()
{
scanf("%d",&n);
for(int i=1; i<=n; i++)
boss[i][0]=boss[i][1]=i;
for(int i=1; i<=n-1; i++)
{
int a,b,c;
scanf("%d %d %d",&a,&b,&c);
ma[a][c]=b;
ma[b][c]=a;
if(c==1)
uni1(a,b);
else
uni0(a,b);
}
for(int i=1; i<=n; i++)
{
siz[fin0(i)][0]++;
siz[fin1(i)][1]++;
//cout<<fin1(i)<<endl;
}
for(int i=1; i<=n; i++)
{
if(boss[i][1]==i)
{
//cout<<siz[i][1]<<endl;
ans+=siz[i][1]*siz[i][1]-siz[i][1];
} if(boss[i][0]==i)
{
//cout<<siz[i][0]<<endl;
ans+=siz[i][0]*siz[i][0]-siz[i][0];
} if(ma[i][0]&&ma[i][1])
{
ans+=(siz[fin0(i)][0]-1)*(siz[fin1(i)][1]-1);
//cout<<i<<" "<<(siz[fin0(i)][0]-1)*(siz[fin1(i)][1]-1)<<endl;
}
//cout<<"ans="<<ans<<endl; }
printf("%lld\n",ans);
return 0;
}

E. Special Segments of Permutation

题意:

给出$n(3 \le n \le 2 \cdot 10^5)$个数,满足$1 \le p_i \le n$,并且每个数只出现一次

计算区间$(l,r)$的数量,满足$p_l + p_r = \max \limits_{i = l}^{r} p_i$

分析:

用单调栈求出以坐标$i$为最大值的最大区间

然后遍历区间左边或者右边的数,看是否能在另一边找到可以匹配的数

遍历左边或者右边,取决于左右区间的大小,不然一定超时

最初我用递归分治去做,ac了,但是,题目的数据很少,我自己造的数据,会栈溢出

ac代码:

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=2e5+10;
const int maxm=200000*2+10;
const int mod=1e9+7;
struct Node
{
int l,r,x;
}node[maxn];
int sta[maxn],top=-1,n,pos[maxn];
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d",&node[i].x);
pos[node[i].x]=i;
node[i].l=i;
}
for(int i=1;i<=n;i++)
{
while(top!=-1&&node[i].x>node[sta[top]].x)
{
node[sta[top]].r=i-1;
node[i].l=node[sta[top]].l;
top--;
}
sta[++top]=i;
}
while(top!=-1)
node[sta[top]].r=n,top--;
int ans=0;
for(int i=1;i<=n;i++)
{
if(i-node[i].l<node[i].r-i)
{
for(int j=node[i].l;j<i;j++)
{
int x=pos[node[i].x-node[j].x];
if(x>i&&x<=node[i].r)ans++;
}
}
else
{
for(int j=i+1;j<=node[i].r;j++)
{
int x=pos[node[i].x-node[j].x];
if(x>=node[i].l&&x<i)ans++;
}
}
}
printf("%d\n",ans);
return 0;
}

F. Card Bag

题意:

有$n(2 \le n \le 5000)$个卡片在背包里面,每个卡片有一个属性$a_i(1\leq a_i\leq n)$

每次取出一张卡片属性看作$x$,如果上一张有卡片,把上一张卡片属性看作$y$

如果$x=y$,胜利

如果$x<y$,失败

如果$x>y$,游戏继续

求胜利的期望概率

分析:

这题一看就是概率DP

定义$prob[x][y]$为抽出$x$张不重复的卡牌,以$y$作为最后一张的概率

定义$num[x]$为$x$卡牌的数量

那么:

$$prob[x][y]=\frac{num[y]}{n-x+1}\times \sum_{i=1}^{y-1}prob[x-1][i]$$

再枚举所有胜利的情况,抽$x$张牌,以$y$为最后两张的概率

$$ans=ans+prob[x-1][y]\times \frac{num[y]-1}{n-x+1}$$

ac代码:

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=5e3+10;
const int maxm=200000*2+10;
const int mod=998244353;
ll num[maxn];
ll ans,sum[maxn],prob[maxn][maxn],inv[maxn];
ll qpow(ll x,ll y)
{
ll res=1,k=x;
while(y)
{
if(y&1)res=res*k%mod;
k=k*k%mod;
y/=2;
}
return res;
}
int main()
{
int n;
scanf("%d",&n);
for(int i=1;i<=n;i++)
inv[i]=qpow(i,mod-2);
for(int i=1;i<=n;i++)
{
int x;
scanf("%d",&x);
num[x]++;
}
for(int i=1;i<=n;i++)//prob[x][y]为已经选择了x个数,末尾为y的概率
{
prob[1][i]=num[i]*inv[n]%mod;
for(int len=2;n-len+1>0;len++)
prob[len][i]=sum[len-1]*num[i]%mod*inv[n-len+1]%mod;
for(int len=1;len<=n;len++)
sum[len]=(sum[len]+prob[len][i])%mod;//sum[len]为prob[len][x]的和
}
for(int i=1;i<=n;i++)//枚举胜利的所有情况,选择len+1个数,并且以i作为结尾
if(num[i]>=2)
for(int len=1;len<n;len++)
ans=(ans+prob[len][i]*(num[i]-1)%mod*inv[n-len]%mod)%mod;
cout<<ans<<endl;
return 0;
}

  

总结:

A.没考虑到交点重合的情况,wa了很多发,我应该模拟所有情况的

B.比赛的时候没有思路

C.贪心的策略错了,导致wa了还不知道原因

贪心是最难的算法Orz

E的数据有点少,我可以自己hack我的代码,第一次遇到cf数据有漏洞

栈的深度最大是$5\times 10^{4}$左右,以后要注意

Educational Codeforces Round 64 (Rated for Div. 2) A,B,C,D,E,F的更多相关文章

  1. Educational Codeforces Round 64 (Rated for Div. 2)题解

    Educational Codeforces Round 64 (Rated for Div. 2)题解 题目链接 A. Inscribed Figures 水题,但是坑了很多人.需要注意以下就是正方 ...

  2. Educational Codeforces Round 64 (Rated for Div. 2) (线段树二分)

    题目:http://codeforces.com/contest/1156/problem/E 题意:给你1-n  n个数,然后求有多少个区间[l,r] 满足    a[l]+a[r]=max([l, ...

  3. Educational Codeforces Round 64 (Rated for Div. 2)D(并查集,图)

    #include<bits/stdc++.h>using namespace std;int f[2][200007],s[2][200007];//并查集,相邻点int find_(in ...

  4. Educational Codeforces Round 85 (Rated for Div. 2)

    \(Educational\ Codeforces\ Round\ 85\ (Rated\ for\ Div.2)\) \(A. Level Statistics\) 每天都可能会有人玩游戏,同时一部 ...

  5. Educational Codeforces Round 65 (Rated for Div. 2)题解

    Educational Codeforces Round 65 (Rated for Div. 2)题解 题目链接 A. Telephone Number 水题,代码如下: Code #include ...

  6. Educational Codeforces Round 60 (Rated for Div. 2) - C. Magic Ship

    Problem   Educational Codeforces Round 60 (Rated for Div. 2) - C. Magic Ship Time Limit: 2000 mSec P ...

  7. Educational Codeforces Round 60 (Rated for Div. 2) - D. Magic Gems(动态规划+矩阵快速幂)

    Problem   Educational Codeforces Round 60 (Rated for Div. 2) - D. Magic Gems Time Limit: 3000 mSec P ...

  8. Educational Codeforces Round 43 (Rated for Div. 2)

    Educational Codeforces Round 43 (Rated for Div. 2) https://codeforces.com/contest/976 A #include< ...

  9. Educational Codeforces Round 35 (Rated for Div. 2)

    Educational Codeforces Round 35 (Rated for Div. 2) https://codeforces.com/contest/911 A 模拟 #include& ...

随机推荐

  1. webpack打包时删除console.log,和debugger

    开发过程中我们不可避免的需要console.log调试,然而在上线时如果不删除这些console.log可能会造成内存泄漏,因为console.log出来的变量是不会被GC的,webpack给我们提供 ...

  2. ReLeQ:一种自动强化学习的神经网络深度量化方法

    ReLeQ:一种自动强化学习的神经网络深度量化方法     ReLeQ:一种自动强化学习的神经网络深度量化方法ReLeQ: An Automatic Reinforcement Learning Ap ...

  3. 基于bootstrap selectpicker ,实现select下拉框模糊查询功能

    1.html代码块 需要引入bootstrap的css js jquery bootstrap.css bootstrap-select.min.css jquery-1.11.3.min.js bo ...

  4. 08Dockerfile基本使用

    使用Dockerfile创建镜像 Dockerfile是一个文本格式的配置文件,用户可以使用Dockerfile赖快速创建自定义的镜像. Dockerfile由一行行命令组成,#开头为注释. 1:Do ...

  5. 轻松搭建CAS 5.x系列(3)-连接数据库,使用数据库表中的帐号做登录

    概要说明 前面的CAS SEVER中的登录帐号名是配置中写死的,实际情况中不太可能用这些方法.通常情况下,数据库的帐号名密码都是在数据库表中的,这样可以对登录帐号进行增删改的处理. 如果您对搭建固定帐 ...

  6. Matlab函数kmeans

    Matlab函数kmeans K-means聚类算法采用的是将N*P的矩阵X划分为K个类,使得类内对象之间的距离最大,而类之间的距离最小. 使用方法:Idx=Kmeans(X,K)[Idx,C]=Km ...

  7. Springboot整合MybatisPlus

    目录 1.pom文件 2.创建CodeGenerator.java 3.在application.yml中配置mybatis-plus 4.创建MybatisPlusConfig.java文件 其他 ...

  8. sql server 查看表中某一字段的排序规则

    SELECT o.name,o.object_id,c.name,c.column_id,c.collation_name   FROM sys.columns c      JOIN sys.obj ...

  9. .NET 反射应用

    object request = null; string requestObjClassName = "命名空间" + 类型.ToString(); Type type = Ty ...

  10. SIFT算法相关资料

    SIFT算法相关资料 一.SIFT算法的教程.源码及应用软件1.ubc:DAVID LOWE---SIFT算法的创始人,两篇巨经典经典的文章http://www.cs.ubc.ca/~lowe/ 2. ...