Problem - 2043E - Codeforces EDU173

Problem - E - Codeforces

这道题目显然每一位之间的影响是独立的,所以我们可以拆位 ,拆成30位然后逐一判断每一位是否合法,对于每一位,此时的矩阵就变成了01矩阵,对于每一行&操作,如果&1 那么相当于不变,&0相当于整行变成0;对于列|操作,如果|1,相当于全变成1 ,如果|0 ,相当于不变;

对于矩阵中的某个位置 ,如果他为1 而最终的矩阵为0,那么需要进行 &0操作,相反需要进行|1操作,假设某一行,进行了行操作,某些列进行了列操作,那么行列之间就会产生交点,对于交点,如果这个交点最终为1 ,那么显然我们要先行后列,反之先列后行,就相当于建立了一条有向边,要严格按照有向图的顺序进行操作才能得到合法答案,因此找到所有的这种节点后,我们可以进行拓扑排序,判断操作顺序是否是一个DAG如果是,说明可以通过DAG的顺序得到合法答案,反之说明存在环,无法得到正确答案,

代码实现如下:

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
#define int long long

void solve(){
    int n,m;
    cin>>n>>m;
    vector<vector<int>>a(n+1,vector<int>(m+1,0)),b(n+1,vector<int>(m+1,0));
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++){
            cin>>a[i][j];
        }
    }
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++){
            cin>>b[i][j];
        }
    }
    
    vector<vector<int>>aa(n+1,vector<int>(m+1,0)),bb(n+1,vector<int>(m+1,0));
    for(int k=0;k<=30;k++){
        vector<bool>row(n+1,0),col(m+1,0);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=m;j++){
                aa[i][j]=a[i][j]&(1<<k);
                bb[i][j]=b[i][j]&(1<<k);
                if(aa[i][j]!=bb[i][j]){
                    if(bb[i][j]==0){
                        row[i]=1;
                    }else col[j]=1;
                }
            }
        }
        vector<vector<int>>e(n+1+m);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=m;j++){
                int b_bit=(b[i][j]>>k)&1;
                if(b_bit==1){
                    e[i].push_back(j+n);
                }else{
                    e[j+n].push_back(i);
                }
            }
        }
        vector<bool>vis(m+n+1,0);
        queue<int>q;
        for(int i=1;i<=n;i++){
            if(row[i]){
                vis[i]=true;
                q.push(i);
            }
        }
        for(int j=1;j<=m;j++){
            if(col[j]){
                vis[j+n]=true;
                q.push(j+n);
            }
        }
        while(!q.empty()){
            int u=q.front();q.pop();
            for(int v:e[u]){
                if(!vis[v]){
                    vis[v]=true;
                    q.push(v);
                }
            }
        }
        int num=0;
        for(int i=1;i<=n+m;i++){
            if(vis[i])num++;
        }
        vector<int>deg(n+m+1,0);
        for(int u=1;u<=n+m;u++){
            if(vis[u]){ 
                for(int v:e[u]){
                    if(vis[v]){ 
                        deg[v]++;
                    }
                }
            }
        }
        
        for(int i=1;i<=n+m;i++){
            if(deg[i]==0&&vis[i]){
                q.push(i);
            }
        }

        vector<int>id;
        while(q.size()){
            int x=q.front();q.pop();
            id.push_back(x);
            for(auto v:e[x]){
          
                if(vis[v] && --deg[v]==0){
                    q.push(v);
                }
            }
        }
        if(id.size()!=num){
            cout<<"No\n";
            return;
        }
    }
    cout<<"Yes\n";

}
signed main(){
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
	int t=1;
	cin>>t;
	while(t--)solve();
	return 0;
}
相关推荐
云泽8085 小时前
C++ 可调用对象通关指南:深度解析 Lambda 表达式、function 包装器与 bind 绑定器
开发语言·c++·算法
wlsh155 小时前
Go 迭代器
算法
语戚5 小时前
力扣 3161. 块放置查询:线段树解法(Java 实现)
java·算法·leetcode·面试·线段树·力扣·
CS创新实验室6 小时前
从顺序表到动态数组:数据结构的永恒基石与现代语言的优雅封装
数据结构·算法
Black蜡笔小新7 小时前
自动化AI算法训练服务器DLTM训推一体化平台助力农业生产管理实现安全智能化
人工智能·算法·自动化
8Qi87 小时前
LeetCode 23. 合并 K 个升序链表 —— 小顶堆(PriorityQueue)
数据结构·算法·leetcode·链表·
QiLinkOS8 小时前
《打破“用爱发电”:一种基于 Gitee 与时间戳的开源权益分配机制探索》
c语言·数据结构·c++·科技·算法·gitee·开源
松间听晚8 小时前
Agentic RL 环境和代码学习:以HGPO为例
算法
智者知已应修善业8 小时前
【51单片机用T0定时器方式1,实现0.5S的时间间隔实现第一次一个灯亮、第二次二个灯亮,直到全部灯亮,然后重复整个过程】2023-12-29
c++·经验分享·笔记·算法·51单片机
小许同学记录成长9 小时前
几何体编辑与布尔运算
算法·无人机