题目链接:

https://codeforces.com/gym/101987

题意:

有长度为$n$的只包含$B,R$的字符串

有m种关系,每个关系说出三个位置的确切字符

这三个位置的字符最多有一个是错的

数据范围:

$1\leq n \leq 5000$

$1\leq m \leq 10000$

分析:

比如给出1B 2R 3B

那么如果1是R,2必须是R,3必须是B

如果2是B,1必须是R,3必须是B

如果3是R,1必须是B,2必须是R

对这些关系建立2—SAT模型

1B的编号为1,1R的编号为1+n

缩点,用tarjin算法把两两互达的点连接在一起,这个联通分量要么全被选择要么全部不选

拓扑排序,从小到大遍历,如果这个点没来过,就标记“选择”,相反点的代表则标记“不选择”,并且传递下去

如果还是没看懂:https://blog.csdn.net/jarjingx/article/details/8521690

AC代码:

#include<bits/stdc++.h>
#define ll long long
#define pic pair<int,char>
using namespace std;
const int maxn=1e4+7;
int low[maxn],tim[maxn],tot,vis[maxn],top,boss[maxn],sk[maxn],pick[maxn],n,dp[maxn];
vector<int>ve[maxn],ve2[maxn];
void tarjin(int x){
tim[x]=low[x]=++tot;//加入递归的时间和最低递归时间
sk[++top]=x;
vis[x]=1;
for(int i=0;i<ve[x].size();i++){
int v=ve[x][i];
if(tim[v]==0){
tarjin(v);
low[x]=min(low[x],low[v]);//对后面产生影响,那么自己也将属于这个联通图
}else if(vis[v])
low[x]=min(low[x],low[v]);//构成环了,当然和他联通呀
}
if(low[x]==tim[x]){
int v;
while(1){
v=sk[top--];//出栈
vis[v]=0;
boss[v]=x;//找到老大了
if(v==x)break;
}
}
}
void dfs(int x){//不选择的标记往后传递
if(pick[x]!=0)return ;
// cout<<x<<" "<<-1<<endl;
pick[x]=-1;
for(int i=0;i<ve2[x].size();i++)
dfs(ve2[x][i]);
}
void tp(){//拓扑排序,这个简单
queue<int>que;
for(int i=1;i<=2*n;i++)
if(boss[i]==i&&dp[i]==0)
que.push(i);
while(que.size()){
int v=que.front();
if(pick[v]==0){
// cout<<v<<" "<<1<<endl;
pick[v]=1;
int w=v;
if(w>n)w-=n;
else w+=n;
dfs(boss[w]);
}
que.pop();
for(int i=0;i<ve2[v].size();i++){
dp[ve2[v][i]]--;
if(dp[ve2[v][i]]==0)que.push(ve2[v][i]);
}
}
}
int main(){
int m;
scanf("%d %d",&n,&m);
for(int i=1;i<=m;i++){
pic zz[3];
for(int j=0;j<3;j++){
scanf("%d %c",&zz[j].first,&zz[j].second);
getchar();
}
for(int j=0;j<3;j++){
int v1=zz[j].first,v2=zz[j].first+n;
if(zz[j].second=='B')
swap(v1,v2);
for(int k=j+1;k<3;k++){
int w1=zz[k].first,w2=zz[k].first+n;
if(zz[k].second=='R')
swap(w1,w2);
if(w1!=v1)ve[w1].push_back(v1);//选择了v1就选择w1,这里反向建边
if(v2!=w2)ve[v2].push_back(w2);
}
}
}
for(int i=1;i<=2*n;i++)
if(tim[i]==0)tarjin(i); for(int i=1;i<=n;i++)
if(boss[i]==boss[i+n]){
printf("-1\n");
return 0;
}
for(int i=1;i<=2*n;i++){//缩点后重新建边
for(int j=0;j<ve[i].size();j++){
int a=boss[i],b=boss[ve[i][j]];
if(a!=b){
ve2[a].push_back(b);
dp[b]++;
}
}
} tp();
for(int i=1;i<=n;i++){
if(pick[boss[i]]==-1)
printf("R");
else printf("B");
}
printf("\n");
return 0;
}

  

codeforces gym #101987K -TV ShowGame(2-SAT)的更多相关文章

  1. codeforces gym #102082C Emergency Evacuation(贪心Orz)

    题目链接: https://codeforces.com/gym/102082 题意: 在一个客车里面有$r$排座位,每排座位有$2s$个座位,中间一条走廊 有$p$个人在车内,求出所有人走出客车的最 ...

  2. Codeforces Gym 100269 Dwarf Tower (最短路)

    题目连接: http://codeforces.com/gym/100269/attachments Description Little Vasya is playing a new game na ...

  3. codeforces Gym 100286J Javanese Cryptoanalysis (二染色)

    每一单词相邻两个字母,不能同时为元音或者辅音... 各种姿势都可以过:7个for,dp,黑白染色,dfs,并查集.... 最主要的思路就是相邻字母连边,把元音和辅音看成两个集合,那么有连边的两个字母一 ...

  4. CodeForces 450B Jzzhu and Sequences (矩阵优化)

    CodeForces 450B Jzzhu and Sequences (矩阵优化) Description Jzzhu has invented a kind of sequences, they ...

  5. Codeforces Gym 101480C - Cow Confinement(扫描线+线段树)

    题面传送门 题意: 有一个 \(10^6\times 10^6\) 的地图.其中 \(m\) 个位置上有花,\(f\) 个矩形外围用栅栏围了起来.保证 \(f\) 个矩形两两之间没有公共点. \(q\ ...

  6. Codeforces Gym100502H:Clock Pictures(KMP算法)

    http://codeforces.com/gym/100502/attachments 题意:有两个时钟上面有n个指针,给出的数字代表指针的角度.问能否在某一时刻使得两个时钟的指针重合. 思路:容易 ...

  7. Codeforces 758C:Unfair Poll(思维+模拟)

    http://codeforces.com/problemset/problem/758/C 题意:教室里有n列m排,老师上课点名从第一列第一排开始往后点,直到点到第一列第m排,就从第二列第一排开始点 ...

  8. Codeforces 639E - Bear and Paradox(二分+贪心)

    Codeforces 题目传送门 & 洛谷题目传送门 原来 jxd 作业里也有我会做的题 i 了 i 了 首先这种题目的套路就是先考虑对于一个固定的 \(c\),怎样求出得分最高的策略,而类似 ...

  9. CodeForces 342A Xenia and Divisors (水题)

    题意:给定 n 个数(小于等于7),让你把它分成 m 组,每组有三个数,且满足,a < b < c,并且 a 能整除 b,b 能整除 c. 析:对于这个题,因为题目说了是不大于7的,那么一 ...

随机推荐

  1. XML-RPC-1概述

    XML-RPC是一个远程过程调用(远端程序呼叫)(remote procedure call,RPC)的分布式计算协议,通过XML将调用函数封装,并使用HTTP协议作为传送机制.   中文名 XML- ...

  2. [转载]详解网络传输中的三张表,MAC地址表、ARP缓存表以及路由表

    [转载]详解网络传输中的三张表,MAC地址表.ARP缓存表以及路由表 虽然学过了计算机网络,但是这部分还是有点乱.正好在网上看到了一篇文章,讲的很透彻,转载过来康康. 本文出自 "邓奇的Bl ...

  3. Navicat MySql 连不上 本地开发环境 MySQL8.0

    原因:   新版mysql数据库的加密方式改变,进而导致Navicat连接输入的密码不能与安装时输入的密码匹配,那如何解决这个问题呢?很简单,只需要一句代码的事儿~ 1.打开MySQL 8.0 Com ...

  4. 解决包含在label标签下的checkbox在ie8及以下版本点击事件无效果兼容的问题

    问题描述:     在IE8及以下版本时,点击label标签无法自动触发checkbox的click事件,导致无法产生希望的效果. 原HTML代码: <div class="col-s ...

  5. JS 百度地图 地图线路描绘

    JS 百度地图 地图线路描绘 <script type="text/javascript" src="http://api.map.baidu.com/api?v= ...

  6. flex布局下img图片变形的解决方法

      图片正常效果   图片变形效果 一.flex-shrink: 0 给 img 设置 flex-shrink: 0; flex-shrink 的默认值为1,如果没有显示定义该属性,将会自动按照默认值 ...

  7. 第五章、Celery分布式系统

    Celery 官方 Celery 官网:http://www.celeryproject.org/ Celery 官方文档英文版:http://docs.celeryproject.org/en/la ...

  8. pycharm2017.3版本永久激活

    1.下载破解文件 链接:https://pan.baidu.com/s/1nwI278l 密码:j3gt 2.修改检测文件 ,在文件后缀是vmoptions的 文件中加入(注意是文件中,不是文件后缀上 ...

  9. Octave(数据绘制)

    >> t=[:0.01:0.98]; >> y1 = sin(*pi**t); >> plot(t,y1); >> y2 = cos(*pi**t); ...

  10. php截取特定字符前面或后面的内容

    1.php 截取特定字符后面的内容 可以使用函数strripos,获取一个字符串在另一个字符串中第一次出现的位置. $number = '1_0'; $result = substr($number, ...