ICM Technex 2017 and Codeforces Round 400 D. The Door Problem 【2-SAT】

D. The Door Problem

D

题意

n n n 个门 和 m m m 个开关,每个门 恰好 2 2 2 个开关控制
起初每个门可能处于 开启( 1 1 1) 状态或 关闭( 0 0 0) 状态,每按下一个控制这扇门的开关,这扇门的状态都会翻转

问是否存在一种按下开关的方案,使得所有门都开启?

思路

考虑 2 − S A T 2-SAT 2SAT ,每个开关只会有两种状态:按下 ( 1 ) (1) (1) 和 不按 ( 0 ) (0) (0)(按两下等价于不按)
可以发现:

  1. 对于已经开启的门,其两个开关要么同时不按,要么同时按下
  2. 对于关闭的门,其两个开关有且仅有一个开关被按下(两个开关状态不同)

假设其两个开关为 u 、 v u、v uv
对于限制 1 1 1:连边 u ↔ v u \lrarr v uv u ˉ ↔ v ˉ \bar u \lrarr \bar v uˉvˉ
对于限制 2 2 2:连边 u ↔ v ˉ u \lrarr \bar v uvˉ u ˉ ↔ v \bar u \lrarr v uˉv

2 − S A T 2-SAT 2SAT 即可

时间复杂度: O ( n + m ) O(n + m) O(n+m)

#include<bits/stdc++.h>
#define fore(i,l,r)	for(int i=(int)(l);i<(int)(r);++i)
#define fi first
#define se second
#define endl '\n'
#define ull unsigned long long
#define ALL(v) v.begin(), v.end()
#define Debug(x, ed) std::cerr << #x << " = " << x << ed;

const int INF=0x3f3f3f3f;
const long long INFLL=1e18;

typedef long long ll;

struct TwoSat {
    int n; //属性数量
    std::vector<std::vector<int>> e;
    std::vector<bool> ans;
    TwoSat(int n) : n(n), e(2 * n), ans(n) {} //下标从0开始

    /* 建边表示 u为f 且 v为g */
    // void addedge(int u, bool f, int v, bool g) { //原变量和反变量相邻放
        // e[2 * u + !f].push_back(2 * v + g); //反变量在偶数位置,原变量在奇数位置
        // e[2 * v + !g].push_back(2 * u + f);

    // }

    void addedge(int u, int v, int same){
        e[2 * u + 1].push_back(2 * v + same);
        e[2 * v + 1].push_back(2 * u + same);
        e[2 * u].push_back(2 * v + (same ^ 1));
        e[2 * v].push_back(2 * u + (same ^ 1));
    }

    bool satisfiable() {
        std::vector<int> id(2 * n, -1), dfn(2 * n, -1), low(2 * n, -1);
        std::vector<int> stk;
        int now = 0, cnt = 0;
        std::function<void(int)> tarjan = [&](int u) {
            stk.push_back(u);
            dfn[u] = low[u] = now++;
            for (auto v : e[u]) {
                if (dfn[v] == -1) {
                    tarjan(v);
                    low[u] = std::min(low[u], low[v]);
                } else if (id[v] == -1) {
                    low[u] = std::min(low[u], dfn[v]);
                }
            }
            if (dfn[u] == low[u]) {
                int v;
                do {
                    v = stk.back();
                    stk.pop_back();
                    id[v] = cnt;
                } while (v != u);
                ++cnt;
            }
        };
        for (int i = 0; i < 2 * n; ++i) if (dfn[i] == -1) tarjan(i);
        for (int i = 0; i < n; ++i) {
            if (id[2 * i] == id[2 * i + 1]) return false;
            ans[i] = id[2 * i] > id[2 * i + 1]; //取依赖性更高的那个
        }
        return true;
    }

    std::vector<bool> get_ans() { return ans; }
};


int main(){
    std::ios::sync_with_stdio(false);
    std::cin.tie(nullptr);
    std::cout.tie(nullptr);
    int n, m;
    std::cin >> n >> m;
    TwoSat twosat(m);
    std::vector<int> a(n);
    std::vector<std::vector<int>> b(n);
    fore(i, 0, n) std::cin >> a[i];
    fore(i, 0, m){
        int k;
        std::cin >> k;
        while(k--){
            int x;
            std::cin >> x;
            --x;
            b[x].push_back(i);
        }
    }

    fore(i, 0, n){
        int u = b[i][0];
        int v = b[i][1];
        twosat.addedge(u, v, a[i]);
    }

    if(twosat.satisfiable()) std::cout << "YES";
    else std::cout << "NO";

    return 0;
}
  • 11
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: emd-icm42670-p_examples-2.1.2.pdf.rar 是一个压缩文件,其中包含了 ICM-42670-P 芯片的示例代码及文档。ICM-42670-P 是一款低功耗、高性能的惯性测量单元(IMU)传感器,可以用于测量物体的加速度、角速度和方向等参数。这个压缩文件包含了 ICM-42670-P 传感器的使用说明、接口定义、寄存器设置及示例代码。 ICM-42670-P 示例代码可以帮助开发者快速理解和使用该芯片。示例代码中提供了一些常见的功能示例,如初始化传感器、获取传感器数据、配置传感器的采样率和滤波器等。开发者可以根据自己的需求进行修改和扩展,以满足特定的应用场景。 除了示例代码,压缩文件中还包含了 ICM-42670-P 芯片的详细技术文档,比如数据手册和寄存器映射表。这些文档可以帮助开发者深入了解芯片的功能、性能和使用方法。通过阅读这些文档,开发者可以更好地理解芯片的工作原理,优化代码的实现,并解决可能出现的问题。 总之,emd-icm42670-p_examples-2.1.2.pdf.rar 包含了 ICM-42670-P 芯片的示例代码和技术文档,为开发者提供了使用该芯片的相关信息和参考实现。开发者可以通过阅读文档和使用示例代码,快速上手并应用该芯片于各种应用场景中。 ### 回答2: emd-icm42670-p_examples-2.1.2.pdf.rar是一个压缩文件,其中包含了ICM42670传感器的示例代码。ICM42670是一款高性能的惯性测量单元(IMU),可用于测量和监测物体的加速度和角速度。 该压缩文件中的示例代码可以帮助开发人员更好地理解和使用ICM42670传感器。代码中包含了各种功能的示例,比如初始化传感器、读取加速度和角速度数据、设置传感器的工作模式等。 这些示例代码使用了特定的编程语言和开发环境,通常会包含详细的说明和注释,以便开发人员能够轻松地理解和修改代码。同时,开发人员还可以根据自己的需求,将这些示例代码作为基础,进行进一步的开发和定制。 通过使用ICM42670传感器示例代码,开发人员可以快速构建和开发基于该传感器的应用程序,如运动跟踪、姿态检测、导航系统等。此外,示例中的代码还可以用于教育和学习目的,帮助初学者更好地理解和掌握传感器的工作原理和使用方法。 总之,emd-icm42670-p_examples-2.1.2.pdf.rar提供了ICM42670传感器的示例代码,为开发人员提供了一个方便的资源,帮助他们更好地利用和应用这个高性能的惯性测量单元。 ### 回答3: emd-icm42670-p_examples-2.1.2.pdf.rar 是一个压缩文件,主要用于存储和传输PDF格式的示例文件。该文件名包含以下信息: emd-:表示此文件与电子机械制动系统( Electronic Mechanical Braking System)相关。这可能是一个示例文件,用于展示电子机械制动系统的使用方法或演示其功能。 icm42670-:表示此文件与ICM42670有关。ICM42670是一种集成了三轴陀螺仪和三轴加速度计功能的传感器芯片。该文件可能包含有关如何使用ICM42670芯片的示例代码或应用程序。 p_examples-:表示此文件是一个示例文件集合(Examples)。这些示例文件可能包含不同的代码示例,演示了如何在特定平台上使用ICM42670芯片。 2.1.2:表示此文件的版本号为2.1.2。这可能意味着这是某个软件或平台的特定版本,其中包含了特定版本ICM42670芯片的示例文件。 .pdf:表示该文件是一个PDF格式的文件,这是一种广泛使用的文件格式,可用于显示文本和图像的电子文档。 .rar:表示该文件是一种压缩文件,采用了RAR格式进行压缩。RAR是一种常见的压缩文件格式,可用于减小文件的大小,便于存储和传输。 因此,emd-icm42670-p_examples-2.1.2.pdf.rar 可能是一个包含了ICM42670芯片使用示例的PDF文件集合,并使用RAR格式进行了压缩。要查看其中的内容,需要使用支持RAR格式的解压缩工具将其解压缩,并使用PDF阅读器打开PDF文件。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值