2-SAT问题,其实是一个逻辑互斥问题。
做了两道裸题之后仔细想来,和小时候做过的“有两个女生,如果A是女生,那么B一定不是女生。A和C性别相同,求A、B、C三人的性别。”几乎是一样的。

对于这道题我们来分析一下。
“如果A是女生,那么B一定不是女生”——A和B性别相反,假设“A为女生”为true,那么“B为女生”必定为false,于是一定有“B为男生”为true【注意是一定】。
对于这句话同理可得,假设“B为女生”为true,那么“A为男生”必定为true。
“A和C性别相同”——假设“A为女生”为true,那么“C为女生”必定为true; 假设“A为男生”为true,那么“C为男生”必定为true。
这道题的逻辑关系十分简单,我们可以肉眼看出答案,然而对于一些逻辑关系成千上万达到1E6数量的我们不能直观看出,所以针对比较复杂的情况,人力想出了列表的方法。

| A | B | C
——————————
女   |    |    |
——————————
男   |    |    |

之后对于随机选择一个格子进行假设保证不出现互斥的情况。

如何实现的?比如我在"A男"这一格打了勾,那么必定会在“B女”,“C男”打勾,发现不满足题意,再令A为女生重新打勾。

下面讨论2-SAT的算法。
先说一些概念:
一个命题:若P则Q。
它的逆否命题:若非Q则非P。
这两个是等价的(大多2-SAT图是对称的,一些不对称的图指定了某个条件的真假性)

一些“不同时为真”、“不同时为假”的逻辑关系都可以化成“若P则Q”的形式(“不同时为真”:若A为真则B为假,及其逆否命题 若B为真则A为假;若B为假则A为真,及其逆否命题 若A为假则B为真)

对于“若P则Q”,我们从P出发,连一条边到Q,代表P如果为真Q一定为真。对于所有这样的逻辑关系都建立这样的逻辑边,之后选取一个节点进行假设赋值,如果这个点无论真假都不满足条件,该问题就无解。

然后解题顺序:
1.建图(把每个点拆成点A 2*i 和点B2*i+1,对于每个点A和B必定有且仅有一个为真)
2.跑2-SAT

下面放代码模板

 void _clean(){

     for(int i=;i<=n*+;i++){
g[i].clear();
mark[i]=false;
}
} bool _dfs(int u){ if(mark[u^])return false;//如果该点的对立面为真,该点必定为假
if(mark[u])return true;//如果该点之前扫过,为真,那么直接返回
mark[u]=true;//如果这个点没讨论过(A/B两点均没赋值),那么把该点赋为真
stk[++top]=u;//进栈,这个我还没能理解,不过最后满栈应该是top==n,栈里的都为真
for(int i=;i<g[u].size();i++)
if(!_dfs(g[u][i]))return false;//该点为真,那么和这个点相连的每个点全必须为真,否则返回false
return true;
} bool _Twosat(){ _clean();
top=;
for(int i=;i<=n*+;i+=)
if(!mark[i+] && !mark[i]){//如果该点没讨论过
top=;
if(!_dfs(i)){假设A点为真失败
while(top)
mark[stk[top--]]=false;//栈里元素全出栈,并赋值为假
if(!_dfs(i+))return false;//如果假设B点为真也失败,那么无解。
}
}
return true;
}

补充:

1、2-SAT问题就是一个“不能同时为真”、“若满足状态1就必定满足状态2”的逻辑问题。这一类问题的特点大多是对于每个对象可以等价成两种互斥状态。

2、大多的2-SAT问题是对称的,即满足p->q,就一定满足-q->-p(逆否命题的互推),但也有个别例外,就是比如有一个对象与2-sat图的其他对象关联,但是题目已经决定了这个对象必须是某一个状态(不能自我假设了,这个时候可以对于这个点跑一个裸的dfs,一会儿我会讲到例题)

下边讲几道2-SAT的题目(都只讲建图不放代码了)

1、

Now or laterUVALive - 3211

另一篇博文里有详细题解:http://www.cnblogs.com/L-Excalibur/p/8513386.html

2、

AstronautsUVALive - 3713

题目链接:https://cn.vjudge.net/contest/209473#problem/K

题目大意:有一堆宇航员要去完成任务,A任务非常伟大,只有能力值大于等于平均能力值的人才能完成,B任务是普通任务,只有能力值低于平均能力值的人才会去完成,C任务不限制条件,所有能力值的人都可以做。但是这堆宇航员互相有一些憎恨关系,如果两个宇航员互相憎恨,那么就不能给他们分配同一个任务,给出一个合理方案,如果没有合理方案输出no solution。

解题思路:每个能力值大于等于平均值的人(以下简称第一类人)划分为两种状态a.选择任务A,b.选择任务C;能力值小于平均值(第二类人)划分为两种状态a.选择任务B,b.选择任务C。如果第一类人或第二类人有内部矛盾,那么如果其中一个为状态a,另一个一定为状态b(对称建图,不能同时为a和b,共四条)。如果第一类人和第二类人有矛盾,那么不能同时为b,一个为b另一个一定为a。

浅谈2-SAT(待续)的更多相关文章

  1. 【WebApi系列】浅谈HTTP

    [01]浅谈HTTP在WebApi开发中的运用 [02]聊聊WebApi体系结构 [03]详解WebApi如何传递参数 [04]详解WebApi测试和PostMan [05]浅谈WebApi Core ...

  2. 【WebApi系列】浅谈HTTP在WebApi开发中的运用

    WebApi系列文章 [01]浅谈HTTP在WebApi开发中的运用 [02]聊聊WebApi体系结构 [03]详解WebApi参数的传递 [04]详解WebApi测试和PostMan [05]浅谈W ...

  3. C#中,用HashTable,DataTable等复制和克隆浅谈

    如有雷同,不胜荣欣,若转载,请注明 在C#中,用HashTable,DataTable等复制和克隆浅谈,下面直接看例子 HashTable ht = null; ht = new HashTable( ...

  4. C# 复制和克隆认识浅谈

    如有雷同,不胜荣欣.若转载,请注明 在C#中,用HashTable,DataTable等复制和克隆浅谈,以下直接看样例 HashTable ht = null; ht = new HashTable( ...

  5. 浅谈 Fragment 生命周期

    版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Fragment 文中如有纰漏,欢迎大家留言指出. Fragment 是在 Android 3.0 中 ...

  6. 浅谈 LayoutInflater

    浅谈 LayoutInflater 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/View 文中如有纰漏,欢迎大家留言指出. 在 Android 的 ...

  7. 浅谈Java的throw与throws

    转载:http://blog.csdn.net/luoweifu/article/details/10721543 我进行了一些加工,不是本人原创但比原博主要更完善~ 浅谈Java异常 以前虽然知道一 ...

  8. 浅谈SQL注入风险 - 一个Login拿下Server

    前两天,带着学生们学习了简单的ASP.NET MVC,通过ADO.NET方式连接数据库,实现增删改查. 可能有一部分学生提前预习过,在我写登录SQL的时候,他们鄙视我说:“老师你这SQL有注入,随便都 ...

  9. 浅谈WebService的版本兼容性设计

    在现在大型的项目或者软件开发中,一般都会有很多种终端, PC端比如Winform.WebForm,移动端,比如各种Native客户端(iOS, Android, WP),Html5等,我们要满足以上所 ...

随机推荐

  1. NO.4: 确定对象被使用前已被初始化

    1.为内置对象进行 "手工初始化",因为C++不保证初始化他们(内置类型在赋值与初始化销毁基本相同,最好还是进行初始化列表),在内置类型过多情况下,可选择private函数统一初始 ...

  2. 在android手机上通过Html5Plus调用java类。

    关于html5plus的资料参考http://www.html5plus.org/ 最近通过html5做手机app,其中涉及到网络通过,必须采用原生的socket,websocket无法满足要求,ht ...

  3. Your Prediction Gets As Good As Your Data

    Your Prediction Gets As Good As Your Data May 5, 2015 by Kazem In the past, we have seen software en ...

  4. 获取web页面xpath

    1. Open Chrome 2. Right click the element that you want to get xpath 3. select "Inspector" ...

  5. RabbitMq简单应用

    服务端: <?php //配置信息 $conn_args = array( 'host' => '127.0.0.1', 'port' => '5672', 'login' => ...

  6. [转]linux各文件夹介绍

    本文来自linux各文件夹的作用的一个精简版,作为个人使用笔记. 下面简单看下linux下的文件结构,看看每个文件夹都是干吗用的? /bin 二进制可执行命令 /dev 设备特殊文件 /etc 系统管 ...

  7. AC自动机(Keywords Search)

    题目链接:https://cn.vjudge.net/contest/280743#problem/A 题目大意:首先给你T组测试样例,然后给你n个字符串,最后再给你一个模式串,然后问你这一些字符串中 ...

  8. C++(vs)多线程调试 (转)

      在一个程序中,这些独立运行的程序片断叫作“线程”(Thread),利用它编程的概念就叫作“多线程处理”.利用线程,用户可按下一个按钮,然后程序会立即作出响应,而不是让用户等待程序完成了当前任务以后 ...

  9. weblogica

  10. gentoo系统安装

    1. Gentoo常用镜像   ===>http://www.linuxsir.org/bbs/thread263600.html 2. 安装方式1 http://www.linuxidc.co ...