Codeforces Round #410 (Div. 2) E. Mike and code of a permutation(拓扑序+线段树)

2023-12-24 06:18

本文主要是介绍Codeforces Round #410 (Div. 2) E. Mike and code of a permutation(拓扑序+线段树),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原题链接:http://codeforces.com/contest/798/problem/E


题解:

大力补了一发E,看了半天的官方题解,结合了它的代码才看懂。。。还是菜啊?


首先很容易考虑到,可以通过拓扑序的方式来构造出答案,那么如何建图就成了问题的关键。


如果ai==-1,则让ai=n+1,如果某个点没有比它小的节点,就把它连到n+1


然后可以通过遍历一个有向图,每次遍历到某个边,就将该边删除。


考虑边(i,j)表示pi>pj,那么,在区间【1,a[i]-1】(i为当前节点)中的任意点j,如果存在边(j,k)且k>i,则必有pi>pj,如果不存在,那么表示已经不存在比点pi小的节点。


可以通过反证法给予证明:


考虑区间【1,a[i]-1】存边(j,k)其中,1<=j<=a[i]-1,1<=k<now,且pi>pj,因为我们是从1遍历到i,如果该边存在,那么一定有j>i,因为从i连出去的边我们会第一个遍历,否则这条边我们一定已经遍历过,那么考虑a[k],因为a[k]!=i,那么一定有pm>pk,而且m<k,在这种情况下,a[k]=m<i,所以不存在。


所以问题就转换成了,求一个区间中的最大值,同时支持修改区间中的某个点的模型。可以通过一个线段树来维护,每个点表示这个区间中的点连接的点集中,编号最大的点是那一个,并且维护该边的编号。删除操作只需要把该点连接到0就好。


时间复杂度O(nlogn)


#include<bits/stdc++.h>
#define xx first
#define yy second
#define mp make_pair
using namespace std;
const int MAXN=5e5+5;
pair<int,int> tree[MAXN<<2];
int a[MAXN],b[MAXN],ans[MAXN];
bool book[MAXN];
int n;
vector<int> sv;
void push_up(int rt)
{tree[rt]=max(tree[rt<<1],tree[rt<<1|1]);
}
void build(int l,int r,int rt)
{if(l==r){tree[rt]=mp(b[l],l);return ;}int mid=(l+r)>>1;build(l,mid,rt<<1);build(mid+1,r,rt<<1|1);push_up(rt);
}
void update(int pos,int val,int l,int r,int rt)
{if(l==r){tree[rt]=mp(val,tree[rt].yy);return ;}int mid=(l+r)>>1;if(pos<=mid){update(pos,val,l,mid,rt<<1);}else{update(pos,val,mid+1,r,rt<<1|1);}push_up(rt);
}
pair<int,int> query(int L,int R,int l,int r,int rt)
{pair<int,int> ret;if(L>R){return mp(0,0);}if(L<=l&&r<=R){return tree[rt];}int mid=(l+r)>>1;if(L<=mid){ret=max(ret,query(L,R,l,mid,rt<<1));}if(R>mid){ret=max(ret,query(L,R,mid+1,r,rt<<1|1));}return ret;
}
void dfs(int now)
{book[now]=true;update(now,0,1,n,1);if(b[now]!=n+1&&!book[b[now]]){dfs(b[now]);}while(1){pair<int,int> mx=query(1,a[now]-1,1,n,1);if(mx.xx>now){dfs(mx.yy);}else{break;}}sv.push_back(now);
}
int main()
{//freopen("in.txt","r",stdin);//freopen("out.txt","w",stdout);scanf("%d",&n);for(int i=1;i<=n;i++)scanf("%d",&a[i]);for(int i=1;i<=n;i++){b[i]=n+1;}for(int i=1;i<=n;i++){if(a[i]!=-1)b[a[i]]=i;}for(int i=1;i<=n;i++){if(a[i]==-1)a[i]=n+1;}build(1,n,1);for(int i=1;i<=n;i++){if(!book[i])dfs(i);}for(int i=0;i<sv.size();i++){ans[sv[i]]=i+1;}for(int i=1;i<=n;i++){printf("%d ",ans[i]);}printf("\n");return 0;
}


这篇关于Codeforces Round #410 (Div. 2) E. Mike and code of a permutation(拓扑序+线段树)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/530806

相关文章

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

hdu 1285(拓扑排序)

题意: 给各个队间的胜负关系,让排名次,名词相同按从小到大排。 解析: 拓扑排序是应用于有向无回路图(Direct Acyclic Graph,简称DAG)上的一种排序方式,对一个有向无回路图进行拓扑排序后,所有的顶点形成一个序列,对所有边(u,v),满足u 在v 的前面。该序列说明了顶点表示的事件或状态发生的整体顺序。比较经典的是在工程活动上,某些工程完成后,另一些工程才能继续,此时

poj 1127 线段相交的判定

题意: 有n根木棍,每根的端点坐标分别是 px, py, qx, qy。 判断每对木棍是否相连,当他们之间有公共点时,就认为他们相连。 并且通过相连的木棍相连的木棍也是相连的。 解析: 线段相交的判定。 首先,模板中的线段相交是不判端点的,所以要加一个端点在直线上的判定; 然后,端点在直线上的判定这个函数是不判定两个端点是同一个端点的情况的,所以要加是否端点相等的判断。 最后

HDU4737线段树

题目大意:给定一系列数,F(i,j)表示对从ai到aj连续求或运算,(i<=j)求F(i,j)<=m的总数。 const int Max_N = 100008 ;int sum[Max_N<<2] , x[Max_N] ;int n , m ;void push_up(int t){sum[t] = sum[t<<1] | sum[t<<1|1] ;}void upd

Codeforces Round #240 (Div. 2) E分治算法探究1

Codeforces Round #240 (Div. 2) E  http://codeforces.com/contest/415/problem/E 2^n个数,每次操作将其分成2^q份,对于每一份内部的数进行翻转(逆序),每次操作完后输出操作后新序列的逆序对数。 图一:  划分子问题。 图二: 分而治之,=>  合并 。 图三: 回溯:

zoj 1721 判断2条线段(完全)相交

给出起点,终点,与一些障碍线段。 求起点到终点的最短路。 枚举2点的距离,然后最短路。 2点可达条件:没有线段与这2点所构成的线段(完全)相交。 const double eps = 1e-8 ;double add(double x , double y){if(fabs(x+y) < eps*(fabs(x) + fabs(y))) return 0 ;return x + y ;