题目描述

给定一个长度为n的正整数序列a,每个数都在1到10^9范围内,告诉你其中s个数,并给出m条信息,每条信息包含三个数l,r,k以及接下来k个正整数,表示a[l],a[l+1],...,a[r-1],a[r]里这k个数中的任意一个都比任意一个剩下的r-l+1-k个数大(严格大于,即没有等号)。
请任意构造出一组满足条件的方案,或者判断无解。

输入

第一行包含三个正整数n,s,m(1<=s<=n<=100000,1<=m<=200000)。
接下来s行,每行包含两个正整数p[i],d[i](1<=p[i]<=n,1<=d[i]<=10^9),表示已知a[p[i]]=d[i],保证p[i]递增。
接下来m行,每行一开始为三个正整数l[i],r[i],k[i](1<=l[i]<r[i]<=n,1<=k[i]<=r[i]-l[i]),接下来k[i]个正整数x[1],x[2],...,x[k[i]](l[i]<=x[1]<x[2]<...<x[k[i]]<=r[i]),表示这k[i]个数中的任意一个都比任意一个剩下的r[i]-l[i]+1-k[i]个数大。Σk <= 300,000

输出

若无解,则输出NIE。
否则第一行输出TAK,第二行输出n个正整数,依次输出序列a中每个数。

样例输入

5 2 2
2 7
5 3
1 4 2 2 3
4 5 1 4

样例输出

TAK
1 7 5 4 3


题解

线段树优化建图+差分约束系统+拓扑排序

先想朴素的建图方法:利用差分约束的思想,如果要求x比y大,那么连边y->x,长度为1。对于每条信息,将非给定节点向一个新建节点连边,长度为0;新建节点向所有给定节点连边,长度为1。然后拓扑排序,如果有点没被更新,或者与给定的点权冲突(即要求比给出的大)则无解,否则出解。

这样建图边数爆炸。考虑到每次给出的都是一段区间,所以可以使用线段树优化这个建图过程。

建立线段树,从子节点到父节点连长度为0的边;对于每条信息,把非给定节点分成k+1个区间,每个区间在线段树中找到对应位置,这些位置向新建节点连边,长度为0;新建节点向给定节点对应的叶子结点连边,长度为1。然后拓扑排序即可。

至于拓扑排序求解的方法,需要记录f数组和t数组,分别表示点的权值和点的已知权值,如果遇到一个点存在已知权值,且f>t,则无解,否则f取f与t的max。

另外,“每个数都在1到10^9范围内”这个10^9是有用的,当f值大于10^9时也应算作无解。

#include <cstdio>
#include <cstring>
#include <queue>
#define N 1000010
#define M 3000010
#define lson l , mid , x << 1
#define rson mid + 1 , r , x << 1 | 1
using namespace std;
queue<int> q;
int head[N] , to[M] , len[M] , next[M] , cnt = 1 , tot , p[N] , f[N] , t[N] , x[N] , rd[N];
void add(int x , int y , int z)
{
to[++cnt] = y , len[cnt] = z , next[cnt] = head[x] , head[x] = cnt , rd[y] ++ ;
}
void build(int l , int r , int x)
{
if(l == r)
{
p[l] = x;
return;
}
int mid = (l + r) >> 1;
build(lson) , build(rson);
add(x << 1 , x , 0) , add(x << 1 | 1 , x , 0);
}
void update(int b , int e , int t , int l , int r , int x)
{
if(b <= l && r <= e)
{
add(x , t , 0);
return;
}
int mid = (l + r) >> 1;
if(b <= mid) update(b , e , t , lson);
if(e > mid) update(b , e , t , rson);
}
int main()
{
int n , s , m , i , u , c , d , l , r , k;
scanf("%d%d%d" , &n , &s , &m);
build(1 , n , 1) , tot = 4 * n;
while(s -- ) scanf("%d%d" , &c , &d) , f[p[c]] = t[p[c]] = d;
while(m -- )
{
scanf("%d%d%d" , &l , &r , &k) , x[0] = l - 1 , x[k + 1] = r + 1 , tot ++ ;
for(i = 1 ; i <= k ; i ++ ) scanf("%d" , &x[i]) , add(tot , p[x[i]] , 1);
for(i = 0 ; i <= k ; i ++ ) if(x[i + 1] - x[i] > 1) update(x[i] + 1 , x[i + 1] - 1 , tot , 1 , n , 1);
}
for(i = 1 ; i <= tot ; i ++ )
if(!rd[i])
f[i] = max(f[i] , 1) , q.push(i);
while(!q.empty())
{
u = q.front() , q.pop();
for(i = head[u] ; i ; i = next[i])
{
f[to[i]] = max(f[to[i]] , f[u] + len[i]) , rd[to[i]] -- ;
if(t[to[i]] && f[to[i]] > t[to[i]])
{
printf("NIE\n");
return 0;
}
if(!rd[to[i]]) q.push(to[i]);
}
}
for(i = 1 ; i <= n ; i ++ )
{
if(!f[p[i]] || f[p[i]] > 1000000000)
{
printf("NIE\n");
return 0;
}
}
printf("TAK\n");
for(i = 1 ; i < n ; i ++ ) printf("%d " , f[p[i]]);
printf("%d" , f[p[n]]);
return 0;
}

【bzoj4383】[POI2015]Pustynia 线段树优化建图+差分约束系统+拓扑排序的更多相关文章

  1. BZOJ4383 [POI2015]Pustynia[线段树优化建边+拓扑排序+差分约束]

    收获挺大的一道题. 这里的限制大小可以做差分约束,从$y\to x$连$1$,表示$y\le x-1$即$y<x$,然后跑最长路求解. 但是,如果这样每次$k+1$个小区间每个点都向$k$个断点 ...

  2. BZOJ_4383_[POI2015]Pustynia_线段树优化建图+拓扑排序

    BZOJ_4383_[POI2015]Pustynia_线段树优化建图+拓扑排序 Description 给定一个长度为n的正整数序列a,每个数都在1到10^9范围内,告诉你其中s个数,并给出m条信息 ...

  3. [POI2015][bzoj4383] Pustynia [线段树优化建图+拓扑排序]

    题面 bzoj权限题传送门 luogu传送门 思路 首先,这个题目显然可以从所有小的点往大的连边,然后如果没环就一定可行,从起点(入读为0)开始构造就好了 但是问题来了,如果每个都连的话,本题中边数是 ...

  4. [POI2015]PUS [线段树优化建图]

    problem 线段树优化建图,拓扑,没了. #include <bits/stdc++.h> #define ls(x) ch[x][0] #define rs(x) ch[x][1] ...

  5. BZOJ5017 [SNOI2017]炸弹 - 线段树优化建图+Tarjan

    Solution 一个点向一个区间内的所有点连边, 可以用线段树优化建图来优化 : 前置技能传送门 然后就得到一个有向图, 一个联通块内的炸弹可以互相引爆, 所以进行缩点变成$DAG$ 然后拓扑排序. ...

  6. 【BZOJ3681】Arietta 树链剖分+可持久化线段树优化建图+网络流

    [BZOJ3681]Arietta Description Arietta 的命运与她的妹妹不同,在她的妹妹已经走进学院的时候,她仍然留在山村中.但是她从未停止过和恋人 Velding 的书信往来.一 ...

  7. 【ARC069F】Flags 2-sat+线段树优化建图+二分

    Description ​ 数轴上有 n 个旗子,第 ii 个可以插在坐标 xi或者 yi,最大化两两旗子之间的最小距离. Input ​ 第一行一个整数 N. ​ 接下来 N 行每行两个整数 xi, ...

  8. 【bzoj5017】[Snoi2017]炸弹 线段树优化建图+Tarjan+拓扑排序

    题目描述 在一条直线上有 N 个炸弹,每个炸弹的坐标是 Xi,爆炸半径是 Ri,当一个炸弹爆炸时,如果另一个炸弹所在位置 Xj 满足:  Xi−Ri≤Xj≤Xi+Ri,那么,该炸弹也会被引爆.  现在 ...

  9. 【bzoj4699】树上的最短路(树剖+线段树优化建图)

    题意 给你一棵 $n$ 个点 $n-1$ 条边的树,每条边有一个通过时间.此外有 $m$ 个传送条件 $(x_1,y_1,x_2,y_2,c)$,表示从 $x_1$ 到 $x_2$ 的简单路径上的点可 ...

随机推荐

  1. WebService学习之旅(六)使用Apache Axis2实现WebService客户端调用

    上节介绍了如何使用Axis2 发布一个WebService,Axis2除了为我们编写WebService应用带来了便利,也同样简化的客户端调用的过程,本节在上节的基础上使用Axis2自带的工具生成客户 ...

  2. ssh框架出现Java.lang.NoSuchMethodError: antlr.collections.AST.getLine()I错误

    原因:因为Struts自带的antlr-2.7.2.jar,比Hibernate自带的antlr-2.7.7.jar的版本要低,存在jar包冲突现象,因此要删除前一个低版本的. 由于myeclipse ...

  3. JSP serverlet区别与联系

    jsp是html包含java servlet是java包含html jsp请求到tomcat---tomcat封装了jsp到servlet实现. 所以jsp请求时候,会自动创建session 而不用在 ...

  4. POJ Washing Clothes 洗衣服 (01背包,微变型)

    题意:有多种颜色的衣服,由两个人合作来洗,必须洗完一种颜色才能洗下一种,求需要多少时间能洗完. 思路:将衣服按颜色分类,对每种颜色进行01背包,容量上限是该种颜色衣服全部洗完的耗时长一半,其实就是在最 ...

  5. Tensorflow_入门学习_2_一个神经网络栗子

    3.0 A Neural Network Example 载入数据: from tensorflow.examples.tutorials.mnist import input_data mnist ...

  6. SQL与脚本语言

    SQL是人类与数据库沟通的语言https://zhidao.baidu.com/question/413397944.html我个人认为SQL是一种专门对数据库进行操作的特殊的脚本语言.因为SQL语句 ...

  7. v-if与v-show的区别与选择

      v-if与v-show的区别与选择 官网给的区别 v-if 是“真正”的条件渲染,因为它会确保在切换过程中条件块内的事件监听器和子组件适当地被销毁和重建. v-if也是惰性的:如果在初始渲染时条件 ...

  8. Mac 安装和卸载 Mysql5.7.11 的方法

    安装 去http://www.mysql.com/downloads/, 选择最下方的MySQL Community Edition,点击MySQL Community Server的download ...

  9. C语言中函数参数传递

    C语言中函数参数传递的三种方式 (1)值传递,就是把你的变量的值传递给函数的形式参数,实际就是用变量的值来新生成一个形式参数,因而在函数里对形参的改变不会影响到函数外的变量的值.(2)地址传递,就是把 ...

  10. 在64位的linux中运行32位的应用程序

    常规做法,先添加32bit架构: sudo dpkg --add-architecture i386 sudo apt-get update sudo apt-get install libc6:i3 ...