BZOJ3600:没有人的算术
传送门
如果能给每个 \(pair\) 按照权值编号就好了
假设之前已经有了所有的权值的编号,现在考虑编号新的 \(pair\)
如果看过了陈立杰的论文的话,不难得到一个重量平衡树的做法
给树上每个子树一个实数权值区间 \([l,r]\),这个点权值为 \(mid=\frac{l+r}{2}\)
左子树 \([l,mid]\) 右子树 \([mid,r]\)
只需要选择一个树高 \(log\) 的树(treap/替罪羊树)使得满足精度要求即可
# include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn(5e5 + 5);
const double alpha(0.75);
int ls[maxn], rs[maxn], rt, tot, size[maxn], que[maxn], cnt, id[maxn], n, m;
double val[maxn];
pair <int, int> info[maxn];
int mx[maxn << 2];
inline int operator <(pair <int, int> a, pair <int, int> b) {
return val[a.first] == val[b.first] ? val[a.second] < val[b.second] : val[a.first] < val[b.first];
}
void Dfs(int u) {
if (!u) return;
Dfs(ls[u]), que[++cnt] = u, Dfs(rs[u]);
}
int Build(int l, int r, double vl, double vr) {
if (l > r) return 0;
double midv;
int mid, o;
mid = (l + r) >> 1, o = que[mid], midv = (vl + vr) * 0.5;
ls[o] = rs[o] = 0, val[o] = midv;
ls[o] = Build(l, mid - 1, vl, midv);
rs[o] = Build(mid + 1, r, midv, vr);
size[o] = size[ls[o]] + size[rs[o]] + 1;
return o;
}
int Rebuild(int x, double vl, double vr) {
cnt = 0, Dfs(x);
return Build(1, cnt, vl, vr);
}
int Insert(int &x, double vl, double vr, pair <int, int> v) {
double midv;
int ret;
midv = (vl + vr) * 0.5;
if (!x) {
x = ++tot, val[x] = midv, info[x] = v, size[x] = 1;
return x;
}
if (alpha * size[x] < max(size[ls[x]], size[rs[x]])) x = Rebuild(x, vl, vr);
if (v == info[x]) return x;
else if (v < info[x]) ret = Insert(ls[x], vl, midv, v);
else ret = Insert(rs[x], midv, vr, v);
size[x] = size[ls[x]] + size[rs[x]] + 1;
return ret;
}
void Modify(int x, int l, int r, int p) {
int mid;
if (l == r) mx[x] = l;
else {
mid = (l + r) >> 1;
p <= mid ? Modify(x << 1, l, mid, p) : Modify(x << 1 | 1, mid + 1, r, p);
mx[x] = val[id[mx[x << 1]]] >= val[id[mx[x << 1 | 1]]] ? mx[x << 1] : mx[x << 1 | 1];
}
}
int Query(int x, int l, int r, int ql, int qr) {
int mid, ret, v;
if (ql <= l && qr >= r) return mx[x];
mid = (l + r) >> 1, ret = -1, v;
if (ql <= mid) ret = Query(x << 1, l, mid, ql, qr);
if (qr > mid) {
v = Query(x << 1 | 1, mid + 1, r, ql, qr);
ret = (ret == -1 || val[id[v]] > val[id[ret]]) ? v : ret;
}
return ret;
}
int main() {
int i, l, r, k;
char op;
scanf("%d%d", &n, &m);
val[0] = -1, Insert(rt, 0, 1, make_pair(0, 0));
for (i = 1; i <= n; ++i) Modify(1, 1, n, i);
for (i = 1; i <= m; ++i) {
scanf(" %c%d%d", &op, &l, &r);
if (op == 'C') {
scanf("%d", &k);
id[k] = Insert(rt, 0, 1, make_pair(id[l], id[r]));
Modify(1, 1, n, k);
}
else printf("%d\n", Query(1, 1, n, l, r));
}
return 0;
}
BZOJ3600:没有人的算术的更多相关文章
- bzoj3600: 没有人的算术
题意:太难说了..手动去看吧反正不是权限题. 膜拜VFK大爷的神题! 其实一开始思路挺清楚的,如果我们能做到用一个实数去代表"数",这就是裸的动态区间最值查询. 关键是怎么用实数去 ...
- [BZOJ3600] 没有人的算术 [重量平衡树+权值线段树]
题面 传送门 思路 这道题目是陈立杰论文<重量平衡树和后缀平衡树在信息学奥赛中的应用 >中关于重量平衡树维护序列排名算法的一个应用 具体方法为:令根节点保存一个实数区间$[0,1]$ 若当 ...
- 「BZOJ3600」没有人的算术 替罪羊树+线段树
题目描述 过长--不想发图也不想发文字,所以就发链接吧-- 没有人的算术 题解 \(orz\)神题一枚 我们考虑如果插入的数不是数对,而是普通的数,这就是一道傻题了--直接线段树一顿乱上就可以了. 于 ...
- 【BZOJ3600】没有人的算术 - 替罪羊树+线段树
题意: 题解: Orz vfleaking……真·神题 做法大概是先把题意中定义的“数”都赋一个实数权值,用平衡树来维护整个从大到小排序过的序列,再用线段树查询最值: 这样做为什么是对的?考虑插入一个 ...
- 【BZOJ3600】没有人的算术(替罪羊树+线段树)
点此看题面 大致题意: 定义任意数对\(>0\),数对之间比大小先比第一位.后比第二位,一开始数列全为\(0\),要求你支持\(a_k=(a_x,a_y)\)和询问区间最大值所在位置两种操作. ...
- bzoj 3600 没有人的算术——二叉查找树动态标号
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3600 已知 l 和 r 的排名,想快速知道 k 的排名.那么建一个 BIT ,用已知的排名做 ...
- bzoj 3600 没有人的算术 - 替罪羊树 - 线段树
题目都是图片,就不给了,就给链接好了 由于bzoj比较慢,就先给[vjudge传送门] 有兴趣的可以去逛bzoj[bzoj传送门] 题目大意 有n个数a[1],a[2],...,a[n],它们开始都是 ...
- bzoj 3600: 没有人的算术
Description Solution 我们可以给每一个数钦定一个权值 , 这样就可以 \(O(1)\) 比较大小了. 考虑怎么确定权值: 用平衡树来维护 , 我们假设根节点管辖 \([1,2^{6 ...
- 【题解】BZOJ 3600: 没有人的算术——替罪羊树、线段树
题目传送门 题意 具体的自己去上面看吧...反正不是权限题. 简单来说,就是定义了一类新的数,每个数是0或者为 \((x_L, x_R)\) ,同时定义比较大小的方式为:非零数大于零,否则按字典序比较 ...
随机推荐
- 初学Oracle
初学Oracle,遇到了很多的问题,下载的是Oracle11g,没有找到合适的管理工具,所以用sql plus 创建表,以下是本人总结的一些sql plus的命令行的命令,希望对大家有用 与sql p ...
- C#-MVC基础-模型(Model)、视图(View)和控制器(Controller)
搜狗百科:http://baike.sogou.com/v25227.htm?fromTitle=MVC MVC全名是Model View Controller,是软件工程中的一种软件架构模式,把软件 ...
- opencv 将视频分解成图片和使用本地图片合成视频
代码如下: // cvTest.cpp : Defines the entry point for the console application. #include "stdafx.h&q ...
- linux下不同颜色文件的性质
绿色文件: 可执行文件,可执行的程序 红色文件:压缩文件或者包文件 蓝色文件:目录 白色文件:一般性文件,如文本文件,配置文件,源码文件等 浅蓝色文件:链接文件,主要是使用ln命令建立的文件 红色闪烁 ...
- mfix输出自定义数据
有时候需要输出一些自定义的网格或者DEM颗粒信息,比如输出颗粒的受力,这里举例颗粒自定义数据的输出.网格自定义输出方法类似. 首先用FileLocatorPro(网上很多绿色版),搜索一下代码里mod ...
- smtp自动发送邮件demo
using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net ...
- js中的正则表达式【常用】
正则表达式是一种用于处理字符串匹配的强大工具,正则的核心在于匹配语法. 以下是常用的匹配规则 . 除了换行符之外的任意一个字符 \ 转义符,取消后面一个字符的含义,使其成为一个普通字符 [] 括号里的 ...
- java回调方法之理解
以前经常看见"回调方法(或回调函数)"一词,但是没有了解过是什么意思,更不知道用法.现在从网络上搜集了一些很好的资料,自己又整理一下,作为自己的笔记,也作为学习过程中的一个小脚印. ...
- Elastic-Job源码分析之JobScheduler类分析
JobScheduler这个类是EJ中比较核心的一个类,我们现在开始解析这个类. 一.构造器 首先我们看一下JobScheduler的几个构造器. private JobScheduler(final ...
- JVM-类加载过程(Java类的生命周期)
什么是类加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构.类的 ...