poj - 3683 - Priest John's Busiest Day(2-SAT)

时间:2022-12-20 03:15:05

题意:有N场婚礼,每场婚礼的开始时间为Si,结束时间为Ti,每场婚礼有个仪式,历时Di,这个仪式要么在Si时刻开始,要么在Ti-Di时刻开始,问能否安排每场婚礼举行仪式的时间,使主持人John能参加所有的这些仪式的全过程。

题目链接:http://poj.org/problem?id=3683

——>>每场婚礼的仪式,要么在开始段举行,要么在结束段举行,且一定要举行,要求各场婚礼仪式没冲突——>>2-SAT。。。

2-SAT挺神,针对此类问题,可谓手到擒来。。。

LJ《训练指南》上的写法挺容易理解的。。。于是用上了。。。(相对于2003年伍昱论文中O(n)的算法,在时间上《训练指南》中的写法比不上)。。。

对于2-SAT,建图非常重要。。。

——>>设一场婚礼为i,mark[2*i] == 1表示在其开始段举行仪式,mark[2*i+1] == 1表示在其结束段举行仪式。。。

建图思路:对于一个仪式i和另一个仪式j,若i与j有冲突,则说明i不能举行或者j不能举行。。。即i == 0 || j == 0,所以,i' -> j,j' -> i。。。

#include <cstdio>
#include <cstring>
#include <vector> using namespace std; const int maxn = 1000 + 10; int N;
int S[maxn], S1[maxn], S2[maxn], T[maxn], T1[maxn], T2[maxn], D[maxn];
int t[maxn][2]; struct Twoset {
int n;
vector<int> G[maxn*2];
bool mark[maxn*2];
int stk[maxn*2], c; void init(int n) {
this->n = n;
for(int i = 0; i < 2*n; i++) G[i].clear();
memset(mark, 0, sizeof(mark));
} void add_clause(int x, int xval, int y, int yval) {
x = x * 2 + xval;
y = y * 2 + yval;
G[x^1].push_back(y);
G[y^1].push_back(x);
} bool dfs(int x) {
if(mark[x^1]) return false;
if(mark[x]) return true;
mark[x] = true;
stk[++c] = x;
int sz = G[x].size();
for(int i = 0; i < sz; i++) {
int v = G[x][i];
if(!dfs(v)) return false;
}
return true;
} bool YES() {
for(int i = 0; i < 2*n; i += 2) if(!mark[i] && !mark[i+1]) {
c = 0;
if(!dfs(i)) {
while(c) mark[stk[c--]] = false;
if(!dfs(i+1)) return false;
}
}
return true;
} void solve() {
if(YES()) {
puts("YES");
for(int i = 0; i < 2*n; i++) if(mark[i]) {
int id = i / 2;
if(i&1) printf("%02d:%02d %02d:%02d\n", (T[id]-D[id])/60, (T[id]-D[id])%60, T[id]/60, T[id]%60);
else printf("%02d:%02d %02d:%02d\n", S[id]/60, S[id]%60, (S[id]+D[id])/60, (S[id]+D[id])%60);
}
}
else puts("NO");
}
} solver; bool isok(int L1, int R1, int L2, int R2) { //判断两个仪式是否冲突
return R1 <= L2 || R2 <= L1;
} void read() {
for(int i = 0; i < N; i++) {
scanf("%d:%d %d:%d %d", &S1[i], &S2[i], &T1[i], &T2[i], &D[i]);
S[i] = S1[i] * 60 + S2[i];
T[i] = T1[i] * 60 + T2[i];
t[i][0] = S[i];
t[i][1] = T[i] - D[i];
}
} void build() {
for(int i = 0; i < N; i++) for(int a = 0; a < 2; a++)
for(int j = i+1; j < N; j++) for(int b = 0; b < 2; b++)
if(!isok(t[i][a], t[i][a]+D[i], t[j][b], t[j][b]+D[j])) //有冲突,不能同时赋a,b,所以a^1或者b^1成立
solver.add_clause(i, a^1, j, b^1);
} int main()
{
while(scanf("%d", &N) == 1) {
solver.init(N);
read();
build();
solver.solve();
}
return 0;
}

poj - 3683 - Priest John's Busiest Day(2-SAT)的更多相关文章

  1. POJ 3683 Priest John&&num;39&semi;s Busiest Day (2-SAT)

    Priest John's Busiest Day Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 6900   Accept ...

  2. POJ 3683 Priest John&&num;39&semi;s Busiest Day(2-SAT&plus;方案输出)

    Priest John's Busiest Day Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 10010   Accep ...

  3. POJ 3683 Priest John&&num;39&semi;s Busiest Day(2-SAT 并输出解)

    Description John is the only priest in his town. September 1st is the John's busiest day in a year b ...

  4. POJ 3683 Priest John&&num;39&semi;s Busiest Day &sol; OpenJ&lowbar;Bailian 3788 Priest John&&num;39&semi;s Busiest Day&lpar;2-sat问题&rpar;

    POJ 3683 Priest John's Busiest Day / OpenJ_Bailian 3788 Priest John's Busiest Day(2-sat问题) Descripti ...

  5. POJ 3683 Priest John&&num;39&semi;s Busiest Day &lpar;2-SAT&rpar;

    题意:有n对新人要在同一天结婚.结婚时间为Ti到Di,这里有时长为Si的一个仪式需要神父出席.神父可以在Ti-(Ti+Si)这段时间出席也可以在(Di-Si)-Si这段时间.问神父能否出席所有仪式,如 ...

  6. POJ 3683 Priest John&&num;39&semi;s Busiest Day &lpar;2-SAT,常规&rpar;

    题意: 一些人要在同一天进行婚礼,但是牧师只有1个,每一对夫妻都有一个时间范围[s , e]可供牧师选择,且起码要m分钟才主持完毕,但是要么就在 s 就开始,要么就主持到刚好 e 结束.因为人数太多了 ...

  7. POJ 3683 Priest John&&num;39&semi;s Busiest Day&lbrack;2-SAT 构造解&rsqb;

    题意: $n$对$couple$举行仪式,有两个时间段可以选择,问是否可以不冲突举行完,并求方案 两个时间段选择对应一真一假,对于有时间段冲突冲突的两人按照$2-SAT$的规则连边(把不冲突的时间段连 ...

  8. POJ 3683 Priest John&&num;39&semi;s Busiest Day 【2-Sat】

    这是一道裸的2-Sat,只要考虑矛盾条件的判断就好了. 矛盾判断: 对于婚礼现场 x 和 y,x 的第一段可以和 y 的第一段或者第二段矛盾,同理,x 的第二段可以和 y 的第一段或者第二段矛盾,条件 ...

  9. POJ 3683 Priest John's Busiest Day

    看这个题目之前可以先看POJ2186复习一下强联通分量的分解 题意:给出N个开始时间和结束时间和持续时间三元组,持续时间可以在开始后或者结束前,问如何分配可以没有冲突. -----–我是分割线---- ...

随机推荐

  1. Mac按键

    ⌘——Command ⌃ ——Control ⌥——Option (alt) ⇧——Shift ⇪——Caps Lock ⌫——Delete

  2. linux平台上面python调用c

    不能免俗,先打印个helloworld出来,c代码的函数 hello.c #include <stdio.h> int helloworld() { printf("hello ...

  3. json数据相对于xml数据&period;

    JSON is a valid subset of JavaScript, Python, and YAML JSON parsing is generally faster than XML par ...

  4. ViewPager欢迎页

    布局  <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:to ...

  5. poj3642 Charm Bracelet&lpar;0-1背包&rpar;

    题目意思: 给出N,M,N表示有N个物品,M表示背包的容量.接着给出每一个物品的体积和价值,求背包可以装在的最大价值. http://poj.org/problem? id=3624 题目分析: o- ...

  6. 手机电话号码吉凶查询原理及ASP算法源码 转

    随着手机的快速普及,越来越多的人都在使用手机,而号码的挑选也是用户越来越关心的事情.虽然号码只是个代号而已,但几千年的传统积淀仍给号码赋予其各种含义,至于号码的吉凶也是见仁见智的一种个人喜好问题,或许 ...

  7. BZOJ 1018&colon; &lbrack;SHOI2008&rsqb;堵塞的交通traffic &lbrack;线段树 区间信息&rsqb;

    1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 3064  Solved: 1027[Submi ...

  8. Android Studio 2&period;2 新功能详解

    Tamic /文 -译 http://blog.csdn.net/sk719887916/article/details/52672688 Android的Studio 2.2 已经可以在官网下载了. ...

  9. luogu5290 春节十二响

    题目链接 思路 先考虑一条链的情况怎么做. 因为只有两个子树,并且两个子树都是链.所以可以把这两条链找出来,然后\(sort\)一下.合并起来. 然后推广到树上 对于每一棵树都可以按照和上面同样的方法 ...

  10. C语言之链表的使用

    C语言链表初学者都说很难,今天就来为大家讲讲链表 讲链表之前不得不介绍一下结构体,在链表学习之前大家都应该已经学了结构体,都知道结构体里面能有许多变量,每个变量可以当做这个结构体的属性,例如: str ...