@description@

给定一个长度为 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 个数大(严格大于,即没有等号)。

请任意构造出一组满足条件的方案,或者判断无解。

input

第一行包含三个正整数 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

output

若无解,则输出NIE。

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

sample input

5 2 2

2 7

5 3

1 4 2 2 3

4 5 1 4

sample output

TAK

6 7 1000000000 6 3

@solution@

首先不考虑任何时间会炸空间会炸等种种问题,这是一道差分约束题。

然后我们来优化一下。

其一是建图的时候,边的数量过多这一问题,我们使用线段树来优化建图。

其二则是数据太大跑最短路跑不过的问题。我们根据图的特殊性来优化。

对于我们建出的图,会产生两类环:

第一类是源点连入某个点再由这个点连回来的情况,含义是一个数的上下界。

这种情况,我们可以仅保留它的下界(或者是上界,看差分约束的具体实现方法),事后再来判断它的上界是否合法。

第二类是不经过源点的环,这个环必然无解(因为它表示 a > b > c > ... > a,而这显然是不可能的)。

综上,我们可以将这个图改成无环的图。

然后就可以拓扑排序了。

@accepted code@

为了防止它溢出,我判断上界 10^9 是在拓扑排序里面判断的。

#include<cstdio>
#include<algorithm>
using namespace std;
const int MAXN = 100000;
const int MAXM = 200000;
const int MAXK = 300000;
const int MAXV = 4*MAXN + MAXK;
const int INF = int(1E9);
struct edge{
int to, dis;
edge *nxt;
}edges[20*MAXK + 10*MAXN + 5], *adj[MAXV + 5], *ecnt=&edges[0];
void addedge(int u, int v, int w) {
edge *p = (++ecnt);
p->to = v, p->dis = w, p->nxt = adj[u], adj[u] = p;
//printf("(%d %d %d)\n", u, v, w);
}
int id[MAXN + 5], num[4*MAXN + 5], cnt;
void build_segtree(int x, int l, int r) {
num[x] = (++cnt);
if( l == r ) {
id[l] = num[x];
return ;
}
int mid = (l + r) >> 1;
build_segtree(x<<1, l, mid);
addedge(num[x<<1], num[x], 0);
build_segtree(x<<1|1, mid + 1, r);
addedge(num[x<<1|1], num[x], 0);
}
void build_edge_segment(int x, int l, int r, int pl, int pr, int p) {
if( pl <= l && r <= pr ) {
addedge(num[x], p, 0);
return ;
}
if( pl > r || pr < l )
return ;
int mid = (l + r) >> 1;
build_edge_segment(x<<1, l, mid, pl, pr, p);
build_edge_segment(x<<1|1, mid + 1, r, pl, pr, p);
}
int x[MAXN + 5], key[MAXN + 5], n;
int ind[MAXV + 5], dis[MAXV + 5], stk[MAXV + 5], tp;
void solve() {
for(int i=0;i<=cnt;i++) {
for(edge *p=adj[i];p;p=p->nxt)
ind[p->to]++;
dis[i] = INF;
}
dis[0] = 0; stk[++tp] = 0;
while( tp ) {
int t = stk[tp--];
for(edge *p=adj[t];p;p=p->nxt) {
dis[p->to] = min(dis[p->to], dis[t] + p->dis);
if( dis[p->to] < -INF ) {
puts("NIE");
return ;
}
ind[p->to]--;
if( ind[p->to] == 0 ) stk[++tp] = p->to;
}
}
for(int i=0;i<=cnt;i++)
if( ind[i] ) {
puts("NIE");
return ;
}
for(int i=1;i<=n;i++)
if( key[i] != -1 && key[i] + dis[id[i]] ) {
puts("NIE");
return ;
}
puts("TAK");
for(int i=1;i<=n;i++)
printf("%d%c", -dis[id[i]], (i == n) ? '\n' : ' ');
}
int main() {
int s, m;
scanf("%d%d%d", &n, &s, &m);
build_segtree(1, 1, n);
for(int i=1;i<=n;i++) {
addedge(0, id[i], -1);
key[i] = -1;
}
for(int i=1;i<=s;i++) {
int p, d; scanf("%d%d", &p, &d);
addedge(0, id[p], -d);
key[p] = d;
}
for(int i=1;i<=m;i++) {
int l, r, k; cnt++;
scanf("%d%d%d", &l, &r, &k);
x[0] = l - 1, x[k + 1] = r + 1;
for(int j=1;j<=k;j++) scanf("%d", &x[j]);
for(int j=1;j<=k+1;j++) build_edge_segment(1, 1, n, x[j-1] + 1, x[j] - 1, cnt);
for(int j=1;j<=k;j++) addedge(cnt, id[x[j]], -1);
}
solve();
}

@details@

什么?题目中还要求了它不能超过 10^9?

我一开始并没有看到 QAQ。调了好久啊 QAQ。

@bzoj - 4378@ [POI2015] Pustynia的更多相关文章

  1. bzoj 4383: [POI2015]Pustynia

    复习了一下线段树优化建图的姿势,在线段树上连边跑拓扑排序 这题竟然卡vector……丧病 #include <bits/stdc++.h> #define N 1810000 using ...

  2. bzoj 4378: [POI2015]Logistyka ——树桩数组+离散化

    Description 维护一个长度为n的序列,一开始都是0,支持以下两种操作:1.U k a 将序列中第k个数修改为a.2.Z c s 在这个序列上,每次选出c个正数,并将它们都减去1,询问能否进行 ...

  3. @bzoj - 4378@ [POI2015] Logistyka

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 维护一个长度为 n 的序列,一开始都是 0,支持以下两种操作: ...

  4. [POI2015]Pustynia

    [POI2015]Pustynia 题目大意: 给定一个长度为\(n(n\le10^5)\)的正整数序列\(a\),每个数都在\(1\)到\(10^9\)范围内,告诉你其中\(s\)个数,并给出\(m ...

  5. 【BZOJ4383】[POI2015]Pustynia 线段树优化建图

    [BZOJ4383][POI2015]Pustynia Description 给定一个长度为n的正整数序列a,每个数都在1到10^9范围内,告诉你其中s个数,并给出m条信息,每条信息包含三个数l,r ...

  6. 洛谷P3588 - [POI2015]Pustynia

    Portal Description 给定一个长度为\(n(n\leq10^5)\)的正整数序列\(\{a_n\}\),每个数都在\([1,10^9]\)范围内,告诉你其中\(s\)个数,并给出\(m ...

  7. bzoj 4386: [POI2015]Wycieczki

    bzoj 4386: [POI2015]Wycieczki 这题什么素质,爆long long就算了,连int128都爆……最后还是用long double卡过的……而且可能是我本身自带大常数吧,T了 ...

  8. BZOJ 4385: [POI2015]Wilcze doły

    4385: [POI2015]Wilcze doły Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 648  Solved: 263[Submit][ ...

  9. BZOJ 4384: [POI2015]Trzy wieże

    4384: [POI2015]Trzy wieże Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 217  Solved: 61[Submit][St ...

随机推荐

  1. 小希的迷宫 HDU - 1272 (并查集)

    思路: 当图中的集合(连通子图)个数为1并且边数等于顶点数-1(即改图恰好为一棵生成树)时,输出Yes. 此题的坑:(1) 如果只输入0 0算作一组数据的话答案应该输出Yes (2) 输入数据可能并不 ...

  2. 弹性盒布局(flex)

    一.Flex 布局是什么? Flex 是 Flexible Box 的缩写,意为"弹性布局",用来为盒状模型提供最大的灵活性. 任何一个容器都可以指定为 Flex 布局. .box ...

  3. 手残,盘符前边多打一个空格导致的message d:\WEB_APP_QuChongFu\file\五月.xlsx (文件名、目录名或卷标语法不正确。)

    尝试读取并解析一个excel文件,一直提示错误 但是有个原始数据,导入就没问题 对比了一下,好像也就是字母d的大小写有区别 我先把大写的D改成小写的试试,如果是大小写问题,那应该抛出异常 好吧,好像并 ...

  4. Django项目:CRM(客户关系管理系统)--09--04PerfectCRM实现King_admin注册功能01

  5. dijkstra算法为什么不能有负边?

    因为Dijkstra算法在计算最短路径时,不会因为负边的出现而更新已经计算过(收录过)的顶点的路径长度, 这样一来,在存在负边的图中,就可能有某些顶点最终计算出的路径长度不是最短的长度. 假设前两个数 ...

  6. css清除浮动各方法与原理

    说到清除浮动的方法,我想网络上应该有不下7,8的方法,介绍这些方法之前,想下为什么清除浮动? 再次回到float这个属性,浮动元素(floats)会被移出文档流,不会影响到块状盒子的布局而只会影响内联 ...

  7. jdbc连接数据库的步骤(转载)

    http://hzy3774.iteye.com/blog/1689525   1.加载JDBC驱动程序: 在连接数据库之前,首先要加载想要连接的数据库的驱动到JVM(Java虚拟机), 这通过jav ...

  8. ELK背景介绍1

    一.elasticsearch背景介绍 1.问题引入:搜索所有天安门相关的内容,大数据量的判断,加索引orm,歌词怎么做?等等问题,大公司上亿条数据怎样开发处理日志? 2.ELK框架,目前先学习E(e ...

  9. js中的replace问题和textarea回车符问题

    在textarea中输入回车符 在js读取textarea中的值有\r\n然后到业务层转换到string中就有可能变成空格形式然后被存入数据库,当在取出此值的时候则会变成空格的形式,因此我们需要将不显 ...

  10. JS放在body与head中的不同

    放在body和head其实差不多的,只不过是文档解析的时间不同.浏览器解析html是从上到下的.如果把javascript放在head里的话,则先被解析,但这时候body还没有解析,所以$(#btn) ...