Codeforces Round #532 (Div. 2) 题解

Codeforces Round #532 (Div. 2)

题目总链接:https://codeforces.com/contest/1100

A. Roman and Browser

题意:

给出由-1和1组成的n个数,现在任意选定一个起点,从起点开始向左向右k个k个地拿走。最后问abs(cnt(-1)-cnt(1))最大是多少。

 

题解:

由于n和k最多只有100,所以枚举起点直接暴力就好了。

 

代码如下:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5+5;
int n,k;
int a[N];
int main(){
    cin>>n>>k;
    int ans=0;
    for(int i=1;i<=n;i++) cin>>a[i];
    int sum1=0,sum2=0;
    for(int i=1;i<=n;i++){if(a[i]==1) sum1++;else sum2++;}
    for(int b=1;b<=k;b++){
        int now1=sum1,now2=sum2;
        for(int i=b;i<=n;i+=k){
            if(a[i]==1) now1--;
            else now2--;
        }
        ans=max(ans,abs(now1-now2));
    }
    cout<<ans;
    return 0;
}
View Code

 

B. Build a Contest

题意:

给出m个数,每个数不超过n,然后问从1到哪些位置,能包含1-n所有的数各一个。注意的是,每个数只能被用一次,当它被一个位置用了之后,其余的位置就不能用它了。

最后输出的时候,如果目前位置满足条件,则输出1;否则输出-1。

 

题解:

比赛的时候我些了个玄学复杂度的代码,最后被卡了...

但可以o(m)来做,具体看代码吧...

 

代码如下:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5+5;
int n,m;
int cnt[N],a[N];
int main(){
    int num = 0;
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++){
        scanf("%d",&a[i]);
        cnt[a[i]]++;
        if(cnt[a[i]]==1) num++;
        if(num==n){
            printf("1");
            for(int j=1;j<=n;j++){
                cnt[j]--;
                if(!cnt[j]) num--;
            }
        }else printf("0");
    }
    return 0;
}
View Code

 

C. NN and the Optical Illusion

题意:

给出内圆半径以及外接圆的个数,外接圆都与旁边的外接圆相切,求外接圆的半径。

 

题解:

高中知识正弦定理推一下就出来了。

 

代码如下:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
double n,r;
int main(){
    cin>>n>>r;
    double pi;
    double now = (double)360/n;
    now/=2;
    double Sin = sin(now*3.1415926/180);
    double tmp = 1-Sin;
    double ans = r*Sin/tmp;
    printf("%.8f",ans);
    return 0;
}
View Code

 

D. Dasha and Chess

题意:

这是一个交互题。题目给出了一个999*999的格子,现在有1个白棋以及666个黑棋。白棋每次只能走向周围的八个空格子,黑棋每次可以任意走。

现在你来执白子先走,如果你要赢,只要你走到一个位置满足白棋和黑棋同一列或者同一行就行了。

现在你每次输出白棋走的位置,然后会给回应:哪个黑子走到哪个位置。

如果最后白棋赢了,应该立即结束程序。

 

题解:

这题本来白子只能走2000步的,但是其实白棋是一定可以赢的下面就说说为什么。

我们先假设白棋走到了棋盘的中间,即(500,500)处。那么现在整个棋盘都被分为了四个部分,黑棋想要赢,最好就是平均分布。

因为我们假设白棋往左上角走,其实它是可以辐射三个部分的,白棋只需要走499步即可到(1,1)点,即只需要499步可以将三个部分辐射完。

如果哪个部分比较多,那黑棋可能就不能在499步之内将三个部分的黑棋都移到右下角去。

但是这题中666/4 = 166...2,也就是说,至少有一个部分,它的黑棋数是多余平均数的,这也就说明,我们可以选择白棋往四周走。

恰好166*3+2 > 499,那么我们只需要选择黑棋最少的那个反方向走就行了。

我一开始就是这里被坑了...我是直接看哪个方向黑棋最多就往哪走的...

 

代码如下:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1005;
struct node{
    int x,y;
}b[N];
int ax,ay,End;
int d[5];
int vis[N][N];
void query(int x,int y){
    printf("%d %d\n",x,y);
    fflush(stdout);
    int k,nowx,nowy;
    scanf("%d%d%d",&k,&nowx,&nowy);
    if(k==-1){
        End=1;
        exit(0);
    }
    vis[b[k].x][b[k].y]=0;
    b[k].x=nowx;b[k].y=nowy;
    vis[nowx][nowy]=1;
}
void go_mid(){
    while(1){
        if(ax==500 && ay==500) break ;
        if(ax<500) ax++;
        else if(ax>500)ax--;
        else if(ay<500)ay++;
        else if(ay>500)ay--;
        query(ax,ay);
        if(End) return ;
    }
}
int main(){
    scanf("%d%d",&ax,&ay);
    for(int i=1;i<=666;i++){
        scanf("%d%d",&b[i].x,&b[i].y);
        vis[b[i].x][b[i].y]=1;
    }
    go_mid();
    if(End) return 0;
    for(int i=1;i<=666;i++){
        int x=b[i].x,y=b[i].y;
        if(x<500){
            if(y<500) d[1]++;
            else d[0]++;
        }else{
            if(y>500) d[3]++;
            else d[2]++;
        }
    }
    int max_d,mx=100000;
    for(int i=0;i<4;i++){
        if(d[i]<mx){
            mx=d[i];
            max_d=i;
        }
    }
    int dx,dy;
    if(max_d==0) dx=1,dy=-1;
    else if(max_d==1) dx=1,dy=1;
    else if(max_d==2) dx=-1,dy=1;
    else if(max_d==3) dx=-1,dy=-1;
    while(1){
        int curx=ax+dx;
        int cury=ay+dy;
        if(vis[curx][cury]){
            ax+=dx;
            query(ax,ay);
            return 0;
        }
        ax=curx;ay=cury;
        query(ax,ay);
        if(End) return 0;
    }
    return 0;
}
View Code

 

E. Andrew and Taxi

题意:

现在给出一个有向无环图,每条边都有相应的边权。

现在你要翻转一些边,使得这个图中不存在环,要求翻转的边的边权最大值最小。

 

题解:

看到最大值最小,其实应该比较容易想到二分的。

我们就二分答案x,如果边权小于等于x,我们就将先无视这些边;否则我们边加边到新图中。

然后在新图上跑拓扑排序,如若有环,那此时就不合法;如若没有环,就合法。最后根据拓扑序判断就好了。

简略证明一下这样做的正确性:

我们假设无向边连接u,v两个点,而我们跑出来是不存在环的。假设现在无环,那么自然合法;

如果存在环了,如果u的拓扑序小于v的拓扑序,那么必然是另外一条路径产生环,设其为u->v->...->w->u,我们来分析w->u这条边。

首先可以知道w->u这条边不可能在新图中,所以之前也是一条无向边,如果我们连w->u,说明w的拓扑序要小于u,又因为v到w,所以v的拓扑序小于w,又根据我们的假设,u的拓扑序小于v,这就存在矛盾了。

以上,就是简略的证明,主要还是结合拓扑序的性质。

 

代码如下:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5+5;
int n,m,tot;
int head[N],in[N],bfn[N],tmp[N];
struct Edge{
    int u,v,w,id;
    bool operator < (const Edge&A)const{
        return w<A.w;
    }
}edge[N];
struct EDGE{
    int u,v,next;
}e[N];
void adde(int u,int v){
    e[++tot].v=v;e[tot].u=u;e[tot].next=head[u];head[u]=tot;
}
bool check(int x){
    memset(head,-1,sizeof(head));tot=0;
    memset(in,0,sizeof(in));memset(bfn,0,sizeof(tmp));
    for(int i=1;i<=m;i++){
        if(edge[i].w>x){
            adde(edge[i].u,edge[i].v);
            in[edge[i].v]++;
        }
    }
    queue <int> q;
    int cnt = 0;
    for(int i=1;i<=n;i++) if(!in[i]) q.push(i),bfn[i]=++cnt;
    while(!q.empty()){
        int u=q.front();q.pop();
        for(int i=head[u];i!=-1;i=e[i].next){
            int v=e[i].v;
            in[v]--;
            if(!in[v]) q.push(v),bfn[v]=++cnt;
        }
    }
    if(cnt<n) return false;
    return true;
}
int main(){
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++){
        int u,v,w;
        scanf("%d%d%d",&u,&v,&w);
        edge[i]={u,v,w,i};
    }
    sort(edge+1,edge+1+m);
    int l = 0 , r = 1e9+1,mid;
    int ans=1;
    while(l<r){
        mid=(l+r)>>1;
        if(check(mid)){
            r=mid;
            ans=mid;
        }else l=mid+1;
    }
    check(ans);
    tot=0;
    printf("%d",ans);
    for(int i=1;i<=m;i++){
        int u=edge[i].u,v=edge[i].v,w=edge[i].w;
        if(bfn[u]>bfn[v] && w<=ans){
            tmp[i]=1;
            tot++;
        }
    }
    printf(" %d\n",tot);
    for(int i=1;i<=m;i++){
        if(tmp[i]) printf("%d ",edge[i].id);
    }
    return 0;
}
View Code

 

posted @ 2019-01-15 17:27  heyuhhh  阅读(241)  评论(0编辑  收藏  举报