zoukankan      html  css  js  c++  java
  • hdoj 1824 Let's go home(2-SAT)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1824

    思路分析:该问题为2-SAT问题;需要注意逻辑推理的等价性;

    (1)题目第一个条件:每一个队或者队长留下或者其与两名队员同时留下,或者表明只能为两种情况中的一种;假设三人为A,B,C,队长为A,0表示不留下,1表示留下,因为B与C同时留下或者不留下,只要B,C中其中一个没有留下或者留下,则B,C中另一个也同样留下或者不留下,所以可以从该条件中推导出六条等价关系,即A不留下->B,C同时留下,A留下->B,C同时不留下,B留下->C留下,A不留下,B留下->C留下,A不留下,C留下->B留下,A不留西,C不留下->B不留下,A留下;

    (2)题目中第二个条件:每一对队员,如果队员A留下,则B必须回家休息,或者B留下,A必须回家休息;则可以推导出两条等价式:A留下->B不留下,B留下->A不留下,注意在这个条件中可以A,B都不留下;

    代码如下:

    #include <cstdio>
    #include <vector>
    #include <cstring>
    #include <iostream>
    using namespace std;
    
    const int MAX_N = 2 * 5000 + 10;
    struct TwoSAT {
        int n;
        vector<int> G[2 * MAX_N];
        bool mark[2 * MAX_N];
        int S[2 * MAX_N], c;
    
        void Init(int n)
        {
            this->n = n;
            for (int i = 0; i <= 2 * n; ++i)
                G[i].clear();
            memset(mark, 0, sizeof(mark));
        }
    
        bool Dfs(int x)
        {
            if (mark[x ^ 1])  return false;
            if (mark[x])      return true;
            mark[x] = true;
            S[c++] = x;
    
            for (int i = 0; i < G[x].size(); ++i)
            {
                if (!Dfs(G[x][i]))
                    return false;
            }
            return true;
        }
    
        void AddClause(int x, int y)
        {
            int a = 2 * x;
            int b = 2 * y;
            G[a ^ 1].push_back(b);
            G[b ^ 1].push_back(a);
        }
    
        void AddClauseTeam(int i, int j, int k)
        {
            int a = 2 * i;
            int b = 2 * j;
            int c = 2 * k;
            G[a].push_back(b ^ 1);
            G[a].push_back(c ^ 1);
            G[b].push_back(a ^ 1);
            G[b].push_back(c);
            G[c].push_back(a ^ 1);
            G[c].push_back(b);
            G[a ^ 1].push_back(b);
            G[a ^ 1].push_back(c);
            G[b ^ 1].push_back(a);
            G[b ^ 1].push_back(c ^ 1);
            G[c ^ 1].push_back(a);
            G[c ^ 1].push_back(b ^ 1);
        }
    
        bool Solve()
        {
            for (int i = 0; i < 2 * n; i += 2)
            {
                if (!mark[i] && !mark[i + 1])
                {
                    c = 0;
                    if (!Dfs(i))
                    {
                        while (c > 0) mark[S[--c]] = false;
                        if (!Dfs(i + 1))
                            return false;
                    }
                }
            }
            return true;
        }
    };
    
    TwoSAT sat;
    int main()
    {
        int n, m;
    
        while (scanf("%d %d", &n, &m) != EOF)
        {
            int a, b, c;
            sat.Init(3 * n);
            for (int i = 0; i < n; ++i)
            {
                scanf("%d %d %d", &a, &b, &c);
                sat.AddClauseTeam(a, b, c);
            }
            for (int i = 0; i < m; ++i)
            {
                scanf("%d %d", &a, &b);
                sat.AddClause(a, b);
            }
    
            bool ok = sat.Solve();
            if (ok)
                printf("yes
    ");
            else
                printf("no
    ");
        }
        return 0;
    }
  • 相关阅读:
    org.eclipse.swt.SWTException: Invalid thread access问题解决方法
    V3700系列存储数据恢复成功
    导致磁盘阵列数据丢失的7个常见原因/早做准备哦
    服务器分区丢失数据恢复过程(阵列数据恢复)
    EFS加密文件无法打开怎么办
    raid5硬盘硬件修复;条带分析方法;阵列重组
    程序员节/技术党福利:ORACLE 环境故障数据恢复方案
    HP MSA存储 raid组lvm下vxfs文件系统数据恢复方案
    如何排除服务器中RAID5故障/服务器数据恢复案例
    linux服务器数据恢复方法_服务器硬盘故障解决方案
  • 原文地址:https://www.cnblogs.com/tallisHe/p/4681259.html
Copyright © 2011-2022 走看看