首页 > 其他分享 >D. 2+ doors(构造 二分图) CF 1715D

D. 2+ doors(构造 二分图) CF 1715D

时间:2022-09-02 11:33:40浏览次数:61  
标签:f0 f1 ch 1715D int CF 位上 doors define

题目:

​ 现在有一个长度为n的序列待构造,给出m对关系\(i, j,x\),表示\(a_i|a_j=x\),请在满足这m对关系的情况下构造出的最小字典序的序列。

分析:

​ 每当我们看到最小字典序的时候,基本都是贪心的思想。本题可以知道,我们要让序列前面的数尽可能的小。对于他给出的关系,需要按位来考虑,但是有一些麻烦的就是你确定一个数的一位的时候,他会影响到与他有关系的数,感觉就是一个二分图的思想。我们可以用\(f0[i][j]\)表示第\(i\)个数在第\(j\)位必定填0,\(f1[i][j]\)同理必定填1。顺序遍历序列,枚举位,能填0就填0。

实现:

​ 对于给出的关系 若x在第\(k\)位上为0,那么\(i\)和\(j\)在第k位上都必须填0,若x在第\(k\)位上为1,那么\(i\)和\(j\)在第k位上至少要有一个人是1。同时不能忽略\(i=j\)的情况,若\(i==j\)且x在第\(k\)位上为1,那么\(i\)在第k位上必须填1。那么现在我们预处理出了必须填的情况,但是还有一种情况我们是不能确定的,就是当某位可以填0或者填1的时候,怎么办?基于贪心的思想,肯定是填0最好,但是不一定能填上。为什么?因为如果你这一位填0,那么和他有关的那个数的这一位必须就填1,所以要在合法的情况下,才能填上0。这就是用到了二分图的思想,实现上我们将这种模糊的关系连一条边,对于当前\(a_i\)决定要填0的时候,遍历一下,若目标数在该位上必须填0,那么我当前\(a_i\)就只能填1了,因为模糊的优先级是不如必定的。

#include <bits/stdc++.h>
    
using namespace std;
#define rep(i, a, n) for(int i = a; i < n; i++)
#define all(x) x.begin(), x.end()
#define pb push_back
#define ios ios::sync_with_stdio(false);cin.tie(0);
#define debug(x)    cout << x << endl;
#define SZ(x)    (int)x.size()
typedef long long LL;
typedef unsigned long long ULL;
typedef pair<int, int> PII;
const int inf = 0x3f3f3f3f;
void read(int &x) {int s = 0, f = 1; char ch = getchar(); while(!isdigit(ch)) {f = (ch == '-' ? -1 : f); ch = getchar();} while(isdigit(ch)) {s = s * 10 + ch - '0'; ch = getchar();} x = s * f;}

const int N = 100005;
int n, m;
vector<int> g[N][30];
int f0[N][30], f1[N][30]; //第i个数的第j位必须填0或者必须填1

signed main()
{
    cin >> n >> m;
    rep(i, 1, m + 1)
    {
        int u, v, c;
        cin >> u >> v >> c;
        if(u > v)   swap(u, v); 
        for(int j = 29; j >= 0; j --)
        {
            if(c >> j & 1)
            {
                if(u == v)
                    f1[u][j] = f1[v][j] = 1;
                else
                    g[u][j].push_back(v);
            }
            else
                f0[u][j] = f0[v][j] = 1;
        }
    }

    rep(i, 1, n + 1)
    {
        int res = 0;
        for(int j = 29; j >= 0; j --)
        {
            bool can0 = 1;
            if(f1[i][j])
                can0 = 0;
            else if(!f1[i][j] && !f0[i][j])
            {
                for(int v : g[i][j])
                    if(f0[v][j])    
                        can0 = 0;
            }

            if(can0) //这位填0
                for(int v : g[i][j])
                    f1[v][j] = 1;
            else
                res += 1 << j;
        }
        cout << res << ' ';
    }
    cout << '\n';
}  

标签:f0,f1,ch,1715D,int,CF,位上,doors,define
From: https://www.cnblogs.com/DM11/p/16649209.html

相关文章

  • 关于使用命令行 cf login 登录 SAP BTP CloudFoundry 环境的问题
    在SAPBTP平台CloudFoundry环境找到APIendpoint:然后使用命令行cfapi,后面跟上这个APIendpoint:然后使用cflogin命令行登录:如果password输入错误,会遇到上......
  • CF1720E. Misha and Paintings
    题意给出n*n的矩阵,ai,j∈[1,n*n],现在要矩形覆盖若干次,每次把一个正方形的ai,j改为x,求最少的次数使得最后有k种不同的数n<=500题解设sum为初始不同的数,若sum<k则显然只......
  • CF464E The Classic Problem
    传送门思路\(2^{100000}\)?别想了,普通高精度肯定不行但我们发现,求最短路的过程中,其实是用到了比较大小和加法操作细想比较大小的过程,当长度相同的数,我们会先略过前面......
  • 最近的一些 CF 题(9.1起)
    1.CF623BB.ArrayGCD先考虑没有操作2的情况,由于不允许全删,所以至少会留下\(a_1\)与\(a_n\)中的一个,那么它们的质因数中必有一个需要成为公因数,由于最大公因数......
  • C. Monoblock(贡献 子段) CF 1715C
    题目:​ 给出长度为n的序列,计算其所有子段的答案和\((\sum_{l=1}^{n}\sum_{r=l}^ng(l,r))\)。对于子段\([l,r]\)的计算公式\(g(l,r)\)=l到r之间合并后的块数。​ 合并......
  • CF1722G 题解
    题目构造一个长度为\(n\)的数列,数列中每个数各不相同且都不超过\(2^{31}\),使得奇数项和偶数项的异或和相等。思路我提供一种比较神奇的构造方法。首先,两个数相等可......
  • CF1114F Please, another Queries on Array?
    CF1114FPlease,anotherQueriesonArray?题目大意你有一个数组\(a_1,a_2,\dots,a_n\)。现在你需要完成\(q\)次操作,有以下两种操作形式:MULTIPLYlrx,对于所有\(i(......
  • ansible 002 连接被控端 inventory ansible.cfg ansible-adhoc ansible原理
    ssh用普通用户连接被控端配置主机清单(/etc/hosts域名解析为前提)[root@workstationansible]#cathostsserveraserverb[root@workstationansible]#pwd/etc/ans......
  • CF1083C Max Mex
    传送门思路对线段树的功能理解又加深了假设我们枚举答案为\(x\),那么要满足有一条链包含了\(1\)~\(x-1\)的数我们考虑建立一棵线段树,下标为点权,区间记录的是\([l......
  • CF351B Jeff and Furik 题解
    CF351BJeffandFurik有一个长度为n的排列p,Jeff每次操作选两个相邻元素交换,Furik每次操作随机执行下列操作之一:1.对于所有p[i]<p[i+1],随机取一对p[i]与p[i+1]交换2.......