SDOI2018IIIDX
/*
题目转换为 n个节点的一片森林,n个权值,要给每个节点分配一个权值,保证子节点的权值不小于父节点的权值,并且1~n的权值的字典序最大。 考场上的贪心很显然 建立出 树来 将所有数值从大到小排序 然后后序遍历依次填进去 这样就可以处理 互不相同的数据 然后一旦有相同的就会gg 正解
考虑从前往后贪心 我们把所有的数字按照从大到小排序, 然后从前往后贪心 每个位置贪心找当前能放的最大的,也就是保证子树元素都比改点大, 如果有相同的, 令我们选择在序列中最靠右的,这样才有可能将较大的留给后面的 对于数列构建线段树 令res[i] 表示第i个元素以及他左边 空余的元素有多少个 每次查询在线段树上二分 找到这个元素之后 体会一下他的所有孩子都会比这个元素大 所以res[i - n] 都要减去sz[i] 这个限制 当我们遍历该节点的第一个孩子时 将其去除即可 */
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<iostream>
#define M 500010
#define fa(x) (int) floor((x) / (k))
#define inf 1000000000
using namespace std; int read() {
int nm = , f = ;
char c = getchar();
for(; !isdigit(c); c = getchar()) if(c == '-') f = -;
for(; isdigit(c); c =getchar() ) nm = nm * + c - '';
return nm * f;
}
int n, res[M << ], lazy[M << ], note[M], b[M], sz[M], ans[M], cnt[M];
double k; void pushup(int now) {
res[now] = min(res[now << ], res[now << | ]);
} void pushdown(int now) {
if(!lazy[now]) return;
lazy[now << ] += lazy[now], res[now << ] += lazy[now], lazy[now << | ] += lazy[now], res[now << | ] += lazy[now];
lazy[now] = ;
} void build(int l, int r, int now) {
if(l > r) return;
if(l == r) {
res[now] = l;
return;
}
int mid = (l + r) >> ;
build(l, mid, now << );
build(mid + , r, now << | );
pushup(now);
} void modify(int l, int r, int now, int ln, int rn, int ver) {
if(l > rn || r < ln) return ;
if(l >= ln && r <= rn) {
res[now] += ver;
lazy[now] += ver;
return;
}
int mid = (l + r) >> ;
pushdown(now);
modify(l, mid, now << , ln, rn , ver);
modify(mid + , r, now << | , ln, rn,ver);
pushup(now);
} int query(int l, int r, int now, int k) {
if(l == r) return (res[now] >= k) ?l : l + ;
int mid = (l + r) >> ;
pushdown(now);
if(k <= res[now << | ]) return query(l, mid ,now << , k);
return query(mid + , r, now << | , k);
} int main() {
n = read();
cin >> k;
for(int i = ; i <= n; i++) note[i] = read(), sz[i] = ;
sort(note + , note + n + , greater<int>() );
for(int i = n; i >= ; i--) {
if(note[i] == note[i + ]) cnt[i] = cnt[i + ] + ;
else cnt[i] = ;
sz[fa(i)] += sz[i];
}
build(, n, );
for(int i = ; i <= n; i++) {
if(fa(i) && fa(i) != fa(i - )) modify(, n, , ans[fa(i)], n, sz[fa(i)] - );
int op = query(, n, , sz[i]);
op += cnt[op],cnt[op]++,op -= (cnt[op] - ),ans[i] = op;
modify(, n, , op, n, -sz[i]);
}
for(int i = ; i <= n; i++) cout << note[ans[i]] << " ";
return ;
}
SDOI2018IIIDX的更多相关文章
随机推荐
- window.external的使用
我们在js中可以通过window.external.myfunc()来调用浏览器提供的外部方法myfunc.可以让网页内的js脚本中能调用C++代码. window.external 是外部对象,如你 ...
- UML图类,接口之间的关系
UML图类之间的关系(来自大话设计模式中的一张图,代表了类,接口之间的各种关系)
- linux 添加用户并赋予root权限
1.添加用户,首先用adduser命令添加一个普通用户,命令如下: #adduser tommy //添加一个名为tommy的用户#passwd tommy //修改密码Changing pass ...
- Excel技巧--让折线图带面积更直观生动
上图为带面积的折线图,比单纯的折线图更直观更好看. 制作办法: 1.先制作纯折线图: 2.再复制一遍数据列,选中图表,点击粘贴(ctrl+v): 3.右键图表,点击“更改图表类型”,对话框如下设置: ...
- RGB颜色空间、HSV颜色空间的理解
HSV是把H(色相),S(饱和度),V(亮度)当做色值来定位颜色的空间. 1.HSV模型 色相:取值范围是0~360度,用来表示颜色的类别.其中红色是0度,绿色是120度,蓝色是240度.饱和度:取值 ...
- Linux环境下查看线程数的几种方法
1.cat /proc/${pid}/status 2.pstree -p ${pid} 3.top -p ${pid} 再按H,或者直接输入 top -bH -d 3 -p ${pid} top ...
- 运行startup.bat的启动过程
一.前言 一般启动tomcat都是找到bin目录下的startup.bat(windows)下或者startup.sh(linux下),所以要研究tomcat的启动,startup脚本是避不开的.那么 ...
- JS和AS交互
这个话题在网上应该说很好找,有很多这方面的资料.比者也看过一些,不过大部分都是AS2与JS进行交互.很少提到AS3,即使有,也让人感觉浅尝辄止.很多东西并没有介绍的太清楚.所有笔者萌生了写这样一篇教程 ...
- ssh config配置
使用ssh config文件可以简化ssh连接输入参数,直接从config读取 (1)建立config文件 config文件位置在~/.ssh/config 如果不存在,可以创建一个 (2)confi ...
- Mongodb条件查询Query的用法
Query.All("name", "a", "b");//通过多个元素来匹配数组Query.And(Query.EQ("name ...