Codeforces1083CMaxMex
作者:吻过彩虹的脸_378 | 来源:互联网 | 2024-09-25 19:57
Description一棵\(N\)个节点的树,每个节点上都有互不相同的\([0,~N-1]\)的数。定义一条路径上的数的集合为\(S\),求一条路径使得\(Mex(S)\)最大。
Description
一棵\(N\)个节点的树, 每个节点上都有 互不相同的 \([0, ~N-1]\) 的数。
定义一条路径上的数的集合为 \(S\), 求一条路径使得 \(Mex(S)\) 最大。
带修改, \(M\) 次查询
Solution
用一棵权值线段树维护。
节点 \([L,R]\)存储信息:是否有一条路径包含 \([L,R]\) 内的所有数 以及 路径两个端点。
合并两个区间: 在\(4\)个点中 枚举新路径的端点, 然后判断另外两个点是否在路径上即可。
正解能 \(O(1)\) 合并, 然而这个解法需要\(O(logN)\)
所以总复杂度为 \(O(Nlog^2N)\)
Code
#include
#include
#include
#include
#include
#define up(a, b) (a = a > b ? a : b)
#define down(a, b) (a = a > b ? b : a)
#define cmax(a, b) (a > b ? a : b)
#define cmin(a, b) (a > b ? b : a)
#define Abs(a) ((a) > 0 ? (a) : -(a))
#define rd read()
#define db double
#define LL long long
using namespace std;
typedef pair P;
inline char nc(){
static char buf[1<<14],*p1=buf,*p2=buf;
return (p1==p2)&&(p2=(p1=buf)+fread(buf,1,1<<14,stdin),p1==p2)?EOF:*p1++;
}
inline int read(){
char c=nc();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1;c=nc();}
while(c>='0'&&c<='9'){x=x*10+c-'0',c=nc();}
return x*f;
}
const int N = 2e5 + 5;
int n, m, a[N], b[N], f[N][20], dep[N];
int timer, tin[N], tout[N];
vector to[N];
void dfs(int u) {
tin[u] = ++timer; // dfs序
for (int i = 0, up = to[u].size(); i int nt = to[u][i];
if (nt == f[u][0])
continue;
dep[nt] = dep[u] + 1;
dfs(nt);
}
tout[u] = timer;
}
inline bool isanc(int u, int v) { // 判断u是否是v的祖先
return tin[u] <= tin[v] && tout[u] >= tout[v];
}
int getlca(int u, int v) {
if (isanc(u, v))
return u;
for (int i = 19; ~i; --i)
if (f[u][i] && (!isanc(f[u][i], v)))
u = f[u][i];
return f[u][0];
}
inline bool isondownpath(int u, P v) { //判断u是否是 竖直路径v() 上的点
return isanc(v.first, u) && isanc(u, v.second);
}
inline bool isonpath(int u, P v) { //判断u是否是 路径v()上的点
int lca = getlca(v.first, v.second);
return isondownpath(u, P(lca, v.second)) || isondownpath(u, P(lca, v.first));
}
P merge(P u, P v) { //合并两条路径
if (u.first == -1 || v.first == -1)
return P(-1, -1);
int g[4] = {u.first, u.second, v.first, v.second};
for (int i = 0; i <4; ++i) {
for (int j = i + 1; j <4; ++j) {
bool flag = true;
for (int k = 0; k <4; ++k) {
if (k != i && k != j && !isonpath(g[k], P(g[i], g[j])))
flag = false;
}
if (flag)
return P(g[i], g[j]);
}
}
return P(-1, -1);
}
namespace SegT {
#define mid ((l + r) >> 1)
#define lson u <<1
#define rson u <<1 | 1
P t[N <<2];
void build(int l, int r, int u) {
if (l == r) {
t[u].first = t[u].secOnd= b[l];
return;
}
build(l, mid, lson); build(mid + 1, r, rson);
t[u] = merge(t[lson], t[rson]);
}
void update(int c, int l, int r, int u) {
if (l == r) {
t[u].first = t[u].secOnd= b[l];
return;
}
if (c <= mid)
update(c, l, mid, lson);
else update(c, mid + 1, r, rson);
t[u] = merge(t[lson], t[rson]);
}
int query(int l, int r, int u, P tmp) {
if (l == r) {
if (tmp.first == -1)
return t[u].first != -1;
P g = merge(tmp, t[u]);
return g.first != -1;
}
P g = tmp.first == -1 ? t[lson] : merge(tmp, t[lson]);
if (g.first == -1)
return query(l, mid, lson, tmp);
else return mid - l + 1 + query(mid + 1, r, rson, g);
}
}using namespace SegT;
int main()
{
n = rd;
for (int i = 1; i <= n; ++i)
a[i] = rd + 1, b[a[i]] = i;
for (int i = 2; i <= n; ++i) {
int u = rd;
f[i][0] = u;
to[u].push_back(i);
to[i].push_back(u);
}
dep[1] = 1; dfs(1);
for (int i = 1; i <20; ++i)
for (int j = 1; j <= n; ++j)
f[j][i] = f[f[j][i - 1]][i - 1];
;
build(1, n, 1);
m = rd;
for (; m; --m) {
int typ = rd;
if (typ == 1) {
int u = rd, v = rd;
swap(a[u], a[v]);
swap(b[a[u]], b[a[v]]);
update(a[u], 1, n, 1);
update(a[v], 1, n, 1);
}
else printf("%d\n", query(1, n, 1, P(-1, -1)));
}
}
Max Mex
推荐阅读
-
DescriptionInputOutputSampleInput44#****#****#*xxx#SampleOutput5DataConstraint分析非常眼熟 ...
[详细]
蜡笔小新 2024-09-28 18:03:01
-
图形绘制是标记和可视化数据的重要方法.通过在Matlab中集成画板绘图功能,可为科学计算提供便利.1设置Matlab支持Opencv编译操作系统:麒麟14.04(基于Ubu ...
[详细]
蜡笔小新 2024-09-28 11:01:26
-
-
这篇文章主要讲解了“GradeBook类怎么定义”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Grad ...
[详细]
蜡笔小新 2024-09-28 08:26:33
-
题目883.三维形体投影面积题目大意在nxn的网格grid中,我们放置了一些与x,y,z三轴对齐的1x1x1立方体。每个值vgri ...
[详细]
蜡笔小新 2024-09-27 13:50:27
-
编译lib手动编译cmake编译gtest测试程序断言和caseFixture使用gmock编译gmock测试程序参考GtestGithub使用gtest(gmock)方便我们编写 ...
[详细]
蜡笔小新 2024-09-27 13:27:28
-
日期:2012-4-7来源:GBin1.com在线演示本地下载今天我们介绍一个超棒的创建快速动态互动HTML5可视化图形效果的javascript类库-Envision.j ...
[详细]
蜡笔小新 2024-09-27 12:50:24
-
#import挂载对象所需要的参数(UIAlertView挂载对象)staticconstcharkRepresente ...
[详细]
蜡笔小新 2024-09-28 16:28:32
-
publicclassJoinThreadextendsThread{staticThreadthreadAnull,threadBnull;publicstaticvoidmai ...
[详细]
蜡笔小新 2024-09-28 15:26:04
-
这篇文章将为大家详细讲解有关如何使用C语言strcmp函数,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一 ...
[详细]
蜡笔小新 2024-09-28 12:35:27
-
下载器,就是一种网络工具,从网络中接收自己想要的数据。下载器是一个网络客户端。它的下载流程无非就是客户端连接服务器端,然后发送资源下载请求 ...
[详细]
蜡笔小新 2024-09-28 11:59:38
-
原题我们定义“区间的价值”为一段区间的最大值*最小值。一个区间左端点在L,右端点在R,那么该区间的长度为(R−L+1)。求长度分别为1~n的区间的最大价值。保证数据随机因为保证数据随 ...
[详细]
蜡笔小新 2024-09-27 22:04:53
-
::命名管道不但能实现同一台机器上两个进程通信,还能在网络中不同机器上的两个进程之间的通信机制。与邮槽不同,命名管道是采用基于连接并且可靠的传输方式,所以命名管道传输数据只能一对一 ...
[详细]
蜡笔小新 2024-09-27 17:36:49
-
wyh2000andastringproblemTimeLimit:20001000MS(JavaOthers)MemoryLimit:13107265 ...
[详细]
蜡笔小新 2024-09-27 15:22:22
-
7.1实验环境VM配置:Ubuntu12.04(x86)7.2实验原理什么是爆破?使用爆破技巧,来绕过共享库地址随机化。7.3实验过程7. ...
[详细]
蜡笔小新 2024-09-27 13:59:14
-
The“travellingsalesmanproblem”asksthefollowingquestion:“Givenalistofcitiesandthedistancesb ...
[详细]
蜡笔小新 2024-09-27 12:33:28
-
吻过彩虹的脸_378
这个家伙很懒,什么也没留下!