题意:

      给你一个数组b[][],在给你一些关系,问是否可以找到一个满足限制的a[],

关系如下(图片):




思路:

      说到限制,而且还是两个两个之间的限制,那么很容易想到2-sat但是这个题目

扎一看还不像,b[i][j]不是只 0 1 2,怎么办呢,其实我们可以一位一位枚举,最多

也就32,对于每一位我们都判断下,只有所有的位数都满足了,才算存在a[],下面说下关键,就是怎么建图。

a[i] | a[j] == 0 说明两个都是0,则           a  ~a ,b ~b.

a[i] | a[j] == 1 说明两个至少有一个是1则    ~a  b ,~b a.

a[i] & a[j] == 1 说明两个都是1,则          ~a  a ,~b b.

a[i] & a[j] == 0 说明至少有一个0 则         b ~a  ,a ~b.

a[i] ^ a[j] == 1 说明两个不同 则            a ~b ,b ~a ,~a b ,~b a

a[i] ^ a[j] == 0 说明两个相同 则            a b ,b ,a ,~a ~b ,~b ~a

然后强连通判断是否可行就行了,这里说一下,之前我强连通用的全是双深搜的那个,一直都可以,知道今天这个题目一直超时,我一开始想不出超时的地方,只能是换了个强连通的算法,用的Tarjan结果1625ms AC了.蛋疼。


#include<stdio.h>
#include<string.h>
#include<stack> #define N_node 1000 + 50
#define N_edge 1000000 + 100

using namespace
std; typedef struct
{
int
to ,next;
}
STAR; STAR E[N_edge];
int
list[N_node] ,tot;
int
DFN[N_node] ,LOW[N_node];
int
Belong[N_node];
int
Index ,num ,okk;
int
instack[N_node];
int
B[550][550];
stack<int>st; void add(int a ,int b)
{

E[++tot].to = b;
E[tot].next = list[a];
list[a] = tot;
} int
minn(int x ,int y)
{
return
x < y ? x : y;
} void
Tarjan(int s)
{

DFN[s] = LOW[s] = Index ++;
st.push(s);
instack[s] = 1;
for(int
k = list[s] ;k ;k = E[k].next)
{
int
to = E[k].to;
if(!
DFN[to])
{

Tarjan(to);
LOW[s] = minn(LOW[to] ,LOW[s]);
}
else if(
instack[to])
{

LOW[s] = minn(DFN[to] ,LOW[s]);
}
}
if(
LOW[s] == DFN[s])
{

num ++;
while(
1)
{
int
v = st.top();
Belong[v] = num;
st.pop();
instack[v] = 0;
if(
v == s) break;
}
}
} bool
ok(int n)
{

memset(instack ,0 ,sizeof(instack));
memset(DFN ,0 ,sizeof(DFN));
memset(LOW ,0 ,sizeof(LOW));
while(!
st.empty()) st.pop();
Index = 1 ,num = 0;
for(int
i = 0 ;i < n * 2 ;i ++)
{
if(
DFN[i]) continue;
Tarjan(i);
}
for(int
i = 0 ;i < n * 2 ;i += 2)
if(
Belong[i] == Belong[i^1]) return 0;
return
1;
} bool
solve(int n )
{
for(int
i = 0 ;i < n ;i ++)
if(
B[i][i]) return 0;
__int64
Key = 1;
for(int
ii = 1 ;ii <= 32 ;ii ++ ,Key *= 2)
{

memset(list ,0 ,sizeof(list));
tot = 1;
for(int
i = 0 ;i < n ;i ++)
for(int
j = 0 ;j < n ;j ++)
{
if(
i == j) continue;
int
now = B[i][j] & Key;
if(
i % 2 && j % 2)
{
if(!
now)
add(i * 2 ,i * 2 + 1) ,add(j * 2 ,j * 2 + 1);
else
add(i * 2 + 1 ,j * 2) ,add(j * 2 + 1 ,i * 2);
}
else if(
i % 2 == 0 && j % 2 == 0)
{
if(!
now)
add(j * 2 ,i * 2 + 1) ,add(i * 2 ,j * 2 + 1);
else
add(i * 2 + 1 ,i * 2) ,add(j * 2 + 1 ,j * 2);
}
else
{
if(!
now)
add(i * 2 ,j * 2) ,add(i * 2 + 1 ,j * 2 + 1),
add(j * 2 ,i * 2) ,add(j * 2 + 1 ,i * 2 + 1);
else

add(i * 2 ,j * 2 + 1) ,add(j * 2 ,i * 2 + 1),
add(i * 2 + 1 ,j * 2) ,add(j * 2 + 1 ,i * 2);
}
}
if(!
ok(n)) return 0;
}
return
1;
} int main ()
{
int
n ,i ,j;
while(~
scanf("%d" ,&n))
{
for(
i = 0 ;i < n ;i ++)
for(
j = 0 ;j < n ;j ++)
scanf("%d" ,&B[i][j]);
solve(n) ? puts("YES") : puts("NO");
}
return
0;
}

hdu4421 2-sat(枚举二进制每一位)的更多相关文章

  1. Java编程的逻辑 (4) - 整数的二进制表示与位运算

    ​本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http: ...

  2. USACO2.1 Hamming Codes【枚举+二进制处理+输出格式+题意理解】

    这道题加了2个看起来奇奇怪怪的$tag$ 1.输出格式:不得不说这个格式输出很恶心,很像$UVA$的风格,细节稍微处理不好就会出错. 因为这个还$WA$了一次: ,m=n; ) { ;i<=t+ ...

  3. POJ 1753. Flip Game 枚举or爆搜+位压缩,或者高斯消元法

    Flip Game Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 37427   Accepted: 16288 Descr ...

  4. 《CLR via C#》---枚举标志和标志位

    枚举类型 枚举类型定义了一组符号名称/值配对. 如 private enum Color /* : byte */ { White, // Assigned a value of 0 Red, // ...

  5. たくさんの数式 / Many Formulas AtCoder - 2067 (枚举二进制)

    Problem Statement You are given a string S consisting of digits between 1 and 9, inclusive. You can ...

  6. img-图片二进制流 64位前端显示

    碰到的场景:因为使用iframe子窗口打开,多张的二维码图片创建方法调用,导致页面打开缓慢, 所以将调取方式转换成<img src="data:image/png;base64,@it ...

  7. poj 3977 Subset(折半枚举+二进制枚举+二分)

    Subset Time Limit: 30000MS   Memory Limit: 65536K Total Submissions: 5721   Accepted: 1083 Descripti ...

  8. 趣题: 按二进制中1的个数枚举1~2^n (位运算技巧)

    ; ; k <= n; k++){ << k)-,u = << n; s < u;){ ;i < n;i++) printf(-i)&); print ...

  9. C# 枚举的flags 标志位应用

    枚举有个特性叫标志位,使用方法如下 [Flags] enum Foo { a =1, b = 2, c = 4, d = 8 } 每个值需要为2的n次方,保证多个值的组合不会重复. 这样在判断其中一个 ...

随机推荐

  1. 树莓派4b通过外接ssd硬盘启动系统失败的排查和解决

    树莓派4b通过外接ssd硬盘启动系统失败,症状: 屏幕卡在黑屏或提示 mmc1:Controller never released inhibit bit(s).... 先说如何设置硬盘启动,后面是解 ...

  2. 25个关键技术点,带你熟悉Python

    摘要:本文收纳了Python学习者经常使用的库和包,并介绍了Python使用中热门的问题. 01.Python 简介 什么是 Python 一种面向对象的高级动态可解释型脚本语言. Python 解释 ...

  3. MyBatis中模糊查询

    接口 // 模糊查询 List<User> getUserLike(String value); Mapper.xml文件 <!-- 模糊查询 --> <select i ...

  4. ASP.NET Core 在 .NET 6 Preview 2 中的更新

    原文:<ASP.NET Core updates in .NET 6 Preview 2>,作者 Daniel Roth .NET 6 预览版 2 现已推出,其中包括许多对 ASP.NET ...

  5. JAVA视频资料百度网盘分享

    1.javascript视频教程 链接: http://pan.baidu.com/s/1gd57FVH 密码: d9ei 2.JPA视频教程 链接: http://pan.baidu.com/s/1 ...

  6. Nodejs学习笔记(5) 文件上传系统实例

    目录 2018.8.4更新:  MySQL可以存放几乎任何类型的数据(图片.文档.压缩包等),但这不是最好的解决方案,正常情况下都是在数据库中存放文件路径,图片.音乐.视频.压缩包.文档等文件存放在硬 ...

  7. PTA 带头结点的链式表操作集

    6-2 带头结点的链式表操作集 (20 分)   本题要求实现带头结点的链式表操作集. 函数接口定义: List MakeEmpty(); Position Find( List L, Element ...

  8. Mysql之索引选择及优化

    索引模型 哈希表 适用于只有等值查询的场景,Memory引擎默认索引 InnoDB支持自适应哈希索引,不可干预,由引擎自行决定是否创建 有序数组:在等值查询和范围查询场景中的性能都非常优秀,但插入和删 ...

  9. Flex属性你真的搞清楚了吗?我深表怀疑

    背景 在使用弹性布局实现两侧宽度固定,中间宽度自适应的效果时,发现自己理解的和实际效果不一致,所以亲自实践验证了一个flex属性的诸多场景的表现,不仅解开了我之前使用过程遇到的疑惑,而且发现了许多自己 ...

  10. 如何从 dump 文件中提取出 C# 源代码?

    一:背景 相信有很多朋友在遇到应用程序各种奇葩问题后,拿下来一个dump文件,辛辛苦苦分析了大半天,终于在某一个线程的调用栈上找到了一个可疑的方法,但 windbg 常常是以 汇编 的方式显示方法代码 ...