什么,tarjan?那是什么?

码量太大,我选择放弃

为什么不用dfs写2-sat呢?他会伤心的说

这题2-sat的过程大佬们已经讲得非常清楚了,我就略微提一下,主要讲dfs的原理

2_sat原理

我们知道,如果要求 \(a\) 或 \(b\) , 那么如果 \(a\) 不成立,我们可以保证 \(b\) 成立.

换成式子: \(a||b\) = \(!a&&b\) || \(a&&!b\)

于是,我们只需要将!a和b连边,!b和a连边就能确定必然要走的路线

dfs原理

dfs的原理是对于每个点,我们将所有能拿的边都拿了,然后判断是否满足某一个点 \(a\) 必然要使得另外一点要同时满足 \(b\) 和 \(!b\) .如果存在这样的一个点,那么这个图必然不成立

具体实现

首先是连边:

for (int i=0;i<m;i++){
string s1,s2; cin >> s1 >> s2;
int b = (s1[0]=='m') ? 1 : 0, d = (s2[0]=='m') ? 1 : 0;
int a = get_num(s1),c = get_num(s2);//get_num是将这个string后面的数字转化为int
adj[a+((b+1)%2)*n].push_back(c+n*d);
adj[c+((d+1)%2)*n].push_back(a+b*n);
}

然后正常dfs

bool dfs(int posi){
memset(vis,0,sizeof(vis));
queue<int> q;
q.push(posi);
vis[posi] = true;
while (!q.empty()){
int qf = q.front();q.pop();
for (int v : adj[qf]){
if (!vis[v]){
vis[v] = true;//没去过下标
q.push(v);
}
}
}
for (int i=1;i<=n;i++) if (vis[i] && vis[i+n]) return false;//这里相当于b&!b,因为i+n代表的是!i
return true;
}

然后呢?

然后就完了啊

其实就对每个点判断是否成立就好了

完整代码:

#include <iostream>
#include <algorithm>
#include <queue>
#include <vector>
#include <cstring>
#include <math.h>
using namespace std;
#define pp pair<int,int>
int n,m,pos[100005];
vector<int> adj[200005];
bool vis[200005];
bool dfs(int posi){
memset(vis,0,sizeof(vis));
queue<int> q;
q.push(posi);
vis[posi] = true;
while (!q.empty()){
int qf = q.front();q.pop();
for (int v : adj[qf]){
if (!vis[v]){
vis[v] = true;
q.push(v);
}
}
}
for (int i=1;i<=n;i++) if (vis[i] && vis[i+n]) return false;
return true;
}
int get_num(string s){
int tmp = 0;
for (int i=s.length()-1;i>=1;i--){
tmp+=(int)(s[i]-48)*pow(10,(int)s.length()-i-1);
}//从右往左拿数
return tmp;
}
int main(){
int T; cin >> T;
while(T--){ cin >> n >> m;
for (int i=1;i<=n*2;i++) adj[i].clear();
for (int i=0;i<m;i++){
string s1,s2; cin >> s1 >> s2;
int b = (s1[0]=='m') ? 1 : 0, d = (s2[0]=='m') ? 1 : 0;//满汉的情况下b是1,否则是2
int a = get_num(s1),c = get_num(s2);
adj[a+((b+1)%2)*n].push_back(c+n*d);//a+((b+1)%2)*n表示!a
adj[c+((d+1)%2)*n].push_back(a+b*n);
}
for (int i=1;i<=n;i++) { if(!dfs(i) && !dfs(i+n)){ cout << "BAD" << endl;goto abcd;//有一个跑不了就输出BAD然后走人} }
cout << "GOOD" << endl;//到这还没走人就是可以输出
abcd:;
} }

好题(双倍经验?):P3007

建议刷完这题去写,难度基本上一样,加几行多一题紫题不好么

题解 P4171 【[JSOI2010]满汉全席】的更多相关文章

  1. 洛谷 P4171 [JSOI2010]满汉全席 解题报告

    P4171 [JSOI2010]满汉全席 题目描述 满汉全席是中国最丰盛的宴客菜肴,有许多种不同的材料透过满族或是汉族的料理方式,呈现在數量繁多的菜色之中.由于菜色众多而繁杂,只有极少數博学多闻技艺高 ...

  2. 洛谷P4171 [JSOI2010] 满汉全席 [2-SAT,Tarjan]

    题目传送门 满汉全席 题目描述 满汉全席是中国最丰盛的宴客菜肴,有许多种不同的材料透过满族或是汉族的料理方式,呈现在數量繁多的菜色之中.由于菜色众多而繁杂,只有极少數博学多闻技艺高超的厨师能够做出满汉 ...

  3. [洛谷P4171][JSOI2010]满汉全席

    题目大意:有$n$个点,每个点可以选或不选,有$m$组约束,形如$a,u,b,v$,表示$u=a,v=b$中至少要满足一个条件,问是否存在一组解,多组询问 题解:$2-SAT$,感觉是板子题呀,最后判 ...

  4. P4171 [JSOI2010]满汉全席

    简要的学了一下2-sat,然而不会输出方案. 就是个sb模板题啦 // luogu-judger-enable-o2 #include<bits/stdc++.h> #define il ...

  5. 【题解】JSOI2010满汉全席

    ~bzoj1823 第一次接触2-SAT——SAT,即适定性(Satisfiability)的缩写.像名称所说,即满足需求的可能性问题,而k-SAT即每个人有k种需求,已经证明k>2时是一个NP ...

  6. P4171 [JSOI2010]满汉全席(2-SAT)

    传送门 2-SAT裸题 把每一道菜拆成两个点分别表示用汉式或满式 连边可以参考板子->这里 然后最尴尬的是我没发现$n<=100$然后化成整数的时候只考虑了$s[1]$结果炸掉了2333 ...

  7. Luogu P4171 [JSOI2010]满汉全席 2-sat

    终于搞懂了\(2-sat\).实际上是个挺简单的东西,像网络流一样关键在于建模. 问题:\(n\)个数\(A\),可以选择\(0\)和\(1\),现在给你\(m\)组条件\(A\),\(B\),对每个 ...

  8. LUOGU P4171 [JSOI2010]满汉全席

    传送门 解题思路 2-SAT 裸题. 代码 #include<iostream> #include<cstdio> #include<cstring> #inclu ...

  9. 【BZOJ1823】[JSOI2010]满汉全席(2-sat)

    [BZOJ1823][JSOI2010]满汉全席(2-sat) 题面 BZOJ 洛谷 题解 很明显的\(2-sat\)模板题,还不需要输出方案. 对于任意两组限制之间,检查有无同一种石材要用两种不同的 ...

  10. 【BZOJ1823】[JSOI2010]满汉全席 2-SAT

    [BZOJ1823][JSOI2010]满汉全席 Description 满汉全席是中国最丰盛的宴客菜肴,有许多种不同的材料透过满族或是汉族的料理方式,呈现在數量繁多的菜色之中.由于菜色众多而繁杂,只 ...

随机推荐

  1. HDU 1542 线段树离散化+扫描线 平面面积计算

    也是很久之前的题目,一直没做 做完之后觉得基本的离散化和扫描线还是不难的,由于本题要离散x点的坐标,最后要计算被覆盖的x轴上的长度,所以不能用普通的建树法,建树建到r-l==1的时候就停止,表示某段而 ...

  2. 三、jsx简化教程

    1)使用 JSX 的好处 1.提供更加语意化且易懂的标签 与html对比 <!--HTML写法--> <form class="messageBox"> & ...

  3. java基础源码 (3)--Annotation(注解)

    借鉴博客地址:https://www.cnblogs.com/skywang12345/p/3344137.html /** * The common interface extended by al ...

  4. mysql第六篇 : MySQL索引原理与慢查询优化

    浏览目录 一.索引介绍 二.索引方法 三.索引类型 四.聚合索引和辅助索引 五.测试索引 六.正确使用索引 七.组合索引 八.注意事项 九.查询计划 十.慢日志查询 十一.大数据量分页优化 一.索引介 ...

  5. javascript中的私有作用域

    我们知道js中所有的块级作用域都是无效的,块级作用域内的变量,在外部仍然可以被读取,其实是申明在外部的.如何实现变量的私有化,只在块级作用域起效,避免污染全局的变量呢.而且,挂载在全局的变量很难被回收 ...

  6. CGridCtrl添加右键菜单

    头文件下添加: afx_msg void OnMergeCell(); afx_msg void OnContextMenu(CWnd* /*pWnd*/, CPoint /*point*/); 添加 ...

  7. 大数据高可用集群环境安装与配置(10)——安装Kafka高可用集群

    1. 获取安装包下载链接 访问https://kafka.apache.org/downloads 找到kafka对应版本 需要与服务器安装的scala版本一致(运行spark-shell可以看到当前 ...

  8. ORA-00911

    直接在PLSQL运行没问题,在java程序里面运行就报错:ORA-00911 select * from mytable; 亲测,改为: select * from mytable 看到区别没,去掉: ...

  9. python虚拟环境配置(下)

    前言 嘿,各位小伙伴们,晚上好呀,新年快乐,注意预防流感哈,就不要出去浪了,万一中奖了,嗯...,还是当个宅男,宅男无敌,哈哈哈, 过年了,村都被封了,哎,出都出不去,想着干点啥,就把以前没完善的继续 ...

  10. 二分+半平面交——poj1279

    /* 二分距离,凸包所有边往左平移这个距离,半平面交后看是否还有核存在 */ #include<iostream> #include<cstring> #include< ...