链接:https://www.nowcoder.net/acm/contest/71/E
来源:牛客网

题目描述

有一个长为 n 的数列 A,其中有 m 个限制条件,条件有两种:
1、对于区间 [l,r],其区间元素按位或和等于 x
2、对于区间 [l,r],其区间元素按位与和等于 x
求出一个数列 A,使得满足给定的 m 个条件,保证有解。

输入描述:

输入第一行两个正整数 n,m,意义如上
接下来 m 行,每行四个整数 op,l,r,x,表示一组限制
op = 1 表示是限制 1,op = 2 表示是限制 2

输出描述:

输出仅一行,n 个整数 a

i

 表示数列 A。要求 0 <= a

i

 < 10

9

输入例子:
4 3
1 1 2 9
2 3 4 2
1 2 3 11
输出例子:
1 9 2 6

-->

示例1

输入

4 3
1 1 2 9
2 3 4 2
1 2 3 11

输出

1 9 2 6

备注:

1<=n,m<=10^5, 1<=l<=r<=n, 0<=x<2^20

题解

差分约束系统,剪枝。

每一位分开考虑,可以列出一系列不等式,只要求出一组可行解。

剪枝:

对于某些位置,在没有跑差分约束系统之前,就可以确定一定是$1$,也就是可以多增加一些不等式,形如$sum[i] - sum[0] >= x$,用来剪枝。

#include <bits/stdc++.h>
using namespace std; const int maxn = 1e5 + 10;
int n, m;
int op[maxn], L[maxn], R[maxn], x[maxn];
int h[maxn], v[maxn * 10], w[maxn * 10], nx[maxn * 10];
int sz;
int ans[maxn]; int dis[maxn], f[maxn];
int sum[maxn]; void init() {
for(int i = 0; i <= n; i ++) {
h[i] = -1;
sum[i] = 0;
}
sz = 0;
} void add(int a, int b, int c) {
//printf("%d -> %d : %d\n", a, b, -c);
v[sz] = b;
w[sz] = -c;
nx[sz] = h[a];
h[a] = sz ++;
} void spfa() {
queue<int> q;
for(int i = 0; i <= n; i ++) {
dis[i] = maxn;
f[i] = 0;
}
dis[0] = 0;
q.push(0);
f[0] = 1;
while(!q.empty()) {
int top = q.front();
q.pop();
f[top] = 0;
for(int i = h[top]; i != -1; i = nx[i]) {
if(dis[top] + w[i] < dis[v[i]]) {
dis[v[i]] = dis[top] + w[i];
if(f[v[i]] == 0) {
f[v[i]] = 1;
q.push(v[i]);
}
}
}
}
} int main() {
scanf("%d%d", &n, &m);
for(int i = 1; i <= m; i ++) {
scanf("%d%d%d%d", &op[i], &L[i], &R[i], &x[i]);
}
for(int t = 0; t < 20; t ++) {
init();
// 0 <= sum[x] - sum[x - 1] <= 1
// ! sum[x] - sum[x - 1] >= 0
// ! sum[x - 1] - sum[x] >= -1
for(int i = 1; i <= n; i ++) {
add(i - 1, i, 0);
add(i, i - 1, -1);
}
for(int i = 1; i <= m; i ++) {
if(op[i] == 1) {
if(x[i] & (1 << t)) {
// [L, R] 至少有一个1
// sum[R] - sum[L - 1] >= 1
add(L[i] - 1, R[i], 1);
} else {
// [L, R] 全为0
// 0 <= sum[R] - sum[L - 1] <= 0
// ! sum[R] - sum[L - 1] >= 0
// ! sum[L - 1] - sum[R] >= 0
add(L[i] - 1, R[i], 0);
add(R[i], L[i] - 1, 0);
}
} else {
if(x[i] & (1 << t)) {
// [L, R] 全为1
// R - L + 1 <= sum[R] - sum[L - 1] <= R - L + 1
// ! sum[R] - sum[L - 1] >= R - L + 1
// ! sum[L - 1] - sum[R] >= -(R - L + 1)
add(L[i] - 1, R[i], R[i] - L[i] + 1);
add(R[i], L[i] - 1, -(R[i] - L[i] + 1));
sum[L[i]] ++;
sum[R[i] + 1] --;
} else {
// [L, R] 不全为1
// 0 <= sum[R] - sum[L - 1] <= R - L
// ! sum[R] - sum[L - 1] >= 0
// ! sum[L - 1] - sum[R] >= L - R
add(L[i] - 1, R[i], 0);
add(R[i], L[i] - 1, L[i] - R[i]);
}
}
}
for(int i = 1; i <= n; i ++) {
sum[i] += sum[i - 1];
}
for(int i = 1; i <= n; i ++) {
if(sum[i]) sum[i] = 1;
}
for(int i = 1; i <= n; i ++) {
sum[i] += sum[i - 1];
add(0, i, sum[i]);
}
spfa();
for(int i = 1; i <= n; i ++) {
dis[i] = -dis[i];
}
for(int i = n; i >= 1; i --) {
dis[i] = dis[i] - dis[i - 1];
}
for(int i = 1; i <= n; i ++) {
ans[i] = ans[i] + dis[i] * (1 << t);
}
}
for(int i = 1; i <= n; i ++) {
printf("%d", ans[i]);
if(i < n) printf(" ");
else printf("\n");
}
return 0;
} /*
v[j] - v[i] >= k, 问v[t] - v[s]最小值
建边 i -> j, 权值为k, s到t的最长路就是答案
*/

Wannafly挑战赛9 E - 组一组的更多相关文章

  1. wannafly 挑战赛9 E 组一组 (差分约束)

    链接:https://www.nowcoder.com/acm/contest/71/E 时间限制:C/C++ 3秒,其他语言6秒 空间限制:C/C++ 65536K,其他语言131072K Spec ...

  2. Wannafly挑战赛25游记

    Wannafly挑战赛25游记 A - 因子 题目大意: 令\(x=n!(n\le10^{12})\),给定一大于\(1\)的正整数\(p(p\le10000)\)求一个\(k\)使得\(p^k|x\ ...

  3. 【Wannafly挑战赛4】F 线路规划 倍增+Kruskal+归并

    [Wannafly挑战赛4]F 线路规划 题目描述 Q国的监察院是一个神秘的组织.这个组织掌握了整个帝国的地下力量,监察着Q国的每一个人.监察院一共有N个成员,每一个成员都有且仅有1个直接上司,而他只 ...

  4. 【Wannafly挑战赛24E】旅行

    [Wannafly挑战赛24E]旅行 题面 牛客 题解 首先有一个非常显然的\(dp\):我们直接把\(s\rightarrow t\)的路径抠出来然后设\(f_{i,j}\)表示到第\(i\)个点, ...

  5. Wannafly挑战赛13 zzf的好矩阵 题解 答案解释

    Wannafly挑战赛13 zzf的好矩阵 题解 文章目录 Wannafly挑战赛13 zzf的好矩阵 题解 分析 结论1 结论2 结论3 C数组对应带子说明 空白长度论述 后续黑色长度论述 能&qu ...

  6. 【Nowcoder71E】组一组(差分约束,最短路)

    [Nowcoder71E]组一组(差分约束,最短路) 题面 Nowcoder 题解 看到二进制显然就直接拆位,那么区间的按位或和按位与转成前缀和之后,可以写成两个前缀和的值的差的大小关系,那么直接差分 ...

  7. Wannafly挑战赛27

    Wannafly挑战赛27 我打的第一场$Wannafly$是第25场,$T2$竟然出了一个几何题?而且还把我好不容易升上绿的$Rating$又降回了蓝名...之后再不敢打$Wannafly$了. 由 ...

  8. Wannafly 挑战赛 19 参考题解

    这一次的 Wannafly 挑战赛题目是我出的,除了第一题,剩余的题目好像对大部分算法竞赛者来说好像都不是特别友好,但是个人感觉题目质量还是过得去的,下面是题目链接以及题解. [题目链接] Wanna ...

  9. Wannafly挑战赛21A

    题目链接 Wannafly挑战赛21A 题解 代码 #include <cstdio> #include <cmath> #define MAX 1000005 #define ...

随机推荐

  1. CentOS7搭建FastDFS V5.11分布式文件系统-第二篇

    1.CentOS7 FastDFS搭建 前面已下载好了要用到的工具集,下面就可以开始安装了: 如果安装过程中出现问题,可以下载我提供的,当前测试可以通过的工具包: 点这里点这里 1.1 安装libfa ...

  2. 【loj6437】 【PKUSC2018】 PKUSC 计算几何

    题目大意:给你一个m个点的简单多边形.对于每个点i∈[1,n],作一个以O点为原点且过点i的圆,求该圆在多边形内的圆弧长度/圆长. 其中n≤200,m≤500. 我们将n个点分开处理. 首先,我们要判 ...

  3. C++的开源跨平台日志库glog学习研究(一)

    作为C++领域中为数不多的好用.高效的.跨平台的日志工具,Google的开源日志库glog也算是凤毛麟角了.glog 是一个C++实现的应用级日志记录框架,提供了C++风格的流操作. 恰巧趁着五一我也 ...

  4. Java之集合(二)ArrayDeque

    转载请注明源出处:http://www.cnblogs.com/lighten/p/7283928.html 1.前言 上章讲解了Java中的集合接口和相关实现抽象类,本章开始介绍一些具体的实现类,第 ...

  5. 程序使用suid应注意的策略

    安全要求程序应使用最小权限执行,然而有的程序由于特殊性,往往在执行某段代码的时候需要提高权限. 由于suid的容易使用特性,很多编程人员往往会直接使用它来实现提高权限的功能,而不是去做代码权限分离.这 ...

  6. 【jQuery源码】事件委托

    jQuery的事件绑定有几个比较优秀的特点: 1. 可以绑定不限数量的处理函数 2. 事件可以委托到祖先节点,不必一定要绑到对应的节点,这样后添加的节点也照样能被处理. 3. 链式操作 下面主要分析事 ...

  7. Attr.checkId()方法

    1.符号sym是TYP02 举个例子,如下: package bazola; class Point { // ... } class Tree<A> { class AttrVisito ...

  8. java 实现 HTTP请求(GET、POST)的方法

    使用Java进行服务调用时,避免不了要使用模拟HTTP请求来实现模拟,我在开发过程中恰巧遇到了这类的业务需求,所以就对这类的方法进行了一次总结,原理层次的东西暂时不考虑,毕竟HTTP的底层实现啥的,东 ...

  9. linux ping 命令解析

    不管在windows平台,还是在linux平台,ping都是非常常用的网络命令:ping命令通过ICMP(Internet控制消息协议)工作:ping可以用来测试本机与目标主机是否联通.联通速度如何. ...

  10. JVM笔记6-垃圾回收器

    JVM进行垃圾回收时要考虑哪的问题如下: 1.如何判定对象为垃圾对象? 1.引用计数法:在对象中添加一个引用计数器,当有地方引用这个对象的时候,引用计数器的值就+1,引用失效的时候,计数器的值就-1, ...