Solution -「九省联考 2018」IIIDX
Description
Link.
给出一个堆,然后让你填数进去,使得其满足小根堆的性质,并使编号靠前的点的数最大。
Solution
考虑贪心,把原数列降序排序,然后因为这个东西是整除分块的形式,所以一个结点的子树一定对应的是原序列的一个子区间。不过这个东西并不能用根号分治来做。
然后对于一个子树的根 \(u\),我们给他 \([l,r]\) 这个区间,\(\text{subtree}(u)-\{u\}=a_{[l,r)}\),结点 \(u=a_{r}\),\([l,r]\) 按需分配,反正就优先队列维护就行了。
代码大概长成这样子:
void Search(int x) {
for(int y in SonOf(x)) Search(y);
if(x != VirtualRoot) {
ans[x] = PriorityQueue.top();
PriorityQueue.pop();
}
}
那个 VirtualRoot
是为了编码方便弄出来的一个虚根,不用管。同时发现这个做法只有在 \(\forall i,j,s.t.d_{i}\neq d_{j}\) 的时候才是对的。Hack 数据网上找找应该有。
对于一个与 \(u\) 同层的结点,可能会出现这个结点与 \(u\) 的儿子交换点权后更优的情况。
对值域 \([1,n]\) 建出一棵线段树,同时定义 \(pos_{i}\) 为 \(\max\{j\mid a_{i}=a_{i+1}=\cdots=a_{j}\}\)。然后每次查找能用的个数不小于该结点子树大小的位置,有多解跑到最右边,然后把这个位置的能用个数减去子树大小。描述的不清楚,建议做代码阅读理解领略一下精神。
#include<bits/stdc++.h>
struct node
{
int mn,tag;
node(int A=0,int B=0)
{
mn=A;
tag=B;
}
}nodes[2000010];
std::vector<std::vector<int> > e;
int n,siz[500010],ans[500010];
double k;
void dfs(int x)
{
siz[x]=1;
for(int i=0;i<int(e[x].size());++i)
{
int y=e[x][i];
dfs(y);
siz[x]+=siz[y];
}
}
void build(int l,int r,int x)
{
if(l^r)
{
int mid=(l+r)>>1;
build(l,mid,x<<1);
build(mid+1,r,x<<1|1);
nodes[x].mn=std::min(nodes[x<<1].mn,nodes[x<<1|1].mn);
}
else nodes[x].mn=l;
}
void push_down(int x)
{
if(nodes[x].tag)
{
int &cur=nodes[x].tag;
nodes[x<<1].mn+=cur;
nodes[x<<1|1].mn+=cur;
nodes[x<<1].tag+=cur;
nodes[x<<1|1].tag+=cur;
cur=0;
}
}
void ins(int l,int r,int x,int fr,int ba,int delta)
{
if(l>ba || r<fr) return;
if(l>=fr && r<=ba)
{
nodes[x].mn+=delta;
nodes[x].tag+=delta;
}
else
{
int mid=(l+r)>>1;
push_down(x);
ins(l,mid,x<<1,fr,ba,delta);
ins(mid+1,r,x<<1|1,fr,ba,delta);
nodes[x].mn=std::min(nodes[x<<1].mn,nodes[x<<1|1].mn);
}
}
int find(int l,int r,int x,int d)
{
if(l^r)
{
int mid=(l+r)>>1;
push_down(x);
if(d<=nodes[x<<1|1].mn) return find(l,mid,x<<1,d);
else return find(mid+1,r,x<<1|1,d);
}
else return l+(nodes[x].mn<d);
}
int getDiv(int x,double k)
{
return int(std::floor(double(x)/k));
}
int main()
{
scanf("%d %lf",&n,&k);
e.resize(n+1);
std::vector<int> a(n+10),pos(n+10);
for(int i=1;i<=n;++i) scanf("%d",&a[i]);
std::sort(a.begin()+1,a.begin()+n+1,std::greater<int>());
for(int i=n-1;i;--i)
{
if(a[i]==a[i+1]) pos[i]=pos[i+1]+1;
}
for(int i=1;i<=n;++i) e[getDiv(i,k)].emplace_back(i);
dfs(0);
build(1,n,1);
for(int i=1;i<=n;++i)
{
if(getDiv(i,k)^getDiv(i-1,k)) ins(1,n,1,ans[getDiv(i,k)],n,siz[getDiv(i,k)]-1);
int tmp=find(1,n,1,siz[i]);
tmp+=pos[tmp];
++pos[tmp];
tmp-=pos[tmp]-1;
ans[i]=tmp;
ins(1,n,1,tmp,n,-siz[i]);
}
for(int i=1;i<=n;++i) printf("%d ",a[ans[i]]);
return 0;
}
Solution -「九省联考 2018」IIIDX的更多相关文章
- 「九省联考 2018」IIIDX 解题报告
「九省联考 2018」IIIDX 这什么鬼题,送的55分要拿稳,实测有60? 考虑把数值从大到小摆好,每个位置\(i\)维护一个\(f_i\),表示\(i\)左边比它大的(包括自己)还有几个数可以选 ...
- loj2472 「九省联考 2018」IIIDX
ref #include <algorithm> #include <iostream> #include <cstdio> using namespace std ...
- LOJ #2473. 「九省联考 2018」秘密袭击
#2473. 「九省联考 2018」秘密袭击 链接 分析: 首先枚举一个权值W,计算这个多少个连通块中,第k大的数是这个权值. $f[i][j]$表示到第i个节点,有j个大于W数的连通块的个数.然后背 ...
- Loj #2479. 「九省联考 2018」制胡窜
Loj #2479. 「九省联考 2018」制胡窜 题目描述 对于一个字符串 \(S\),我们定义 \(|S|\) 表示 \(S\) 的长度. 接着,我们定义 \(S_i\) 表示 \(S\) 中第 ...
- LOJ 2743(洛谷 4365) 「九省联考 2018」秘密袭击——整体DP+插值思想
题目:https://loj.ac/problem/2473 https://www.luogu.org/problemnew/show/P4365 参考:https://blog.csdn.net/ ...
- [loj 2478][luogu P4843]「九省联考 2018」林克卡特树
传送门 Description 小L 最近沉迷于塞尔达传说:荒野之息(The Legend of Zelda: Breath of The Wild)无法自拔,他尤其喜欢游戏中的迷你挑战. 游戏中有一 ...
- @loj - 2478@「九省联考 2018」林克卡特树
目录 @description@ @solution@ @part - 1@ @part - 2@ @accepted code@ @details@ @description@ 小 L 最近沉迷于塞 ...
- 【LOJ】#2479. 「九省联考 2018」制胡窜
题解 老了,国赛之前敲一个后缀树上LCT和线段树都休闲的很 现在后缀树上线段树合并差点把我写死 主要思路就是后缀树+线段树合并+容斥,我相信熟练的OIer看到这已经会了 但就是不想写 但是由于我过于老 ...
- LOJ#2471「九省联考 2018」一双木棋 MinMax博弈+记搜
题面 戳这里 题解 因为每行取的数的个数是单调不增的,感觉状态数不会很多? 怒而记搜,结果过了... #include<bits/stdc++.h> #define For(i,x,y) ...
- Solution -「六省联考 2017」「洛谷 P3750」分手是祝愿
\(\mathcal{Description}\) Link. 有 \(n\) 盏编号为 \(1\sim n\),已知初始状态的灯,每次操作选取 \(x\in[1,n]\),使得所有编号为 \ ...
随机推荐
- 图解三代测序(SMRT Sequencing)
目前主流三代测序平台除了Oxford 家的 Nanopore,还有 Pacific Biosciences(简称 PacBio)公司的 Single Molecule Real-Time(SMRT)S ...
- k8s实战案例之部署redis单机和redis cluster
1.在k8s上部署redis单机 1.1.redis简介 redis是一款基于BSD协议,开源的非关系型数据库(nosql数据库),作者是意大利开发者Salvatore Sanfilippo在2009 ...
- Mybatis的ResultMap对column和property的理解
Mybatis的ResultMap对column和property的理解 首先,先看看这张图,看能不能一下看明白: select元素有很多属性(这里说用的比较多的): id:命名空间唯一标识,可以被用 ...
- Rust的语句与表达式
Rust 语句与表达式 Rust 中的语法分为两大类: 语句 (statement) 和表达式 (Expression). 语句:指的是要执行的一些操作和产生副作用的表达式. 表达式:主要用于计算求值 ...
- 浅谈 thinkphp composer 扩展包加载原理
浅谈 thinkphp composer 扩展包加载原理 本文将介绍 ThinkPHP 中 Composer 扩展包的加载原理,帮助读者更好地理解和应用该功能. 前言 如题,今天感觉好久没有更新博客了 ...
- 【HMS Core】Health Kit健康数据采样, 原子采样数据问题
[问题描述] 1.体脂数据中的肌肉量和水份量是如何获得的,都有些什么?体脂数据中的体重,体脂是用户自己上传的,然后通过计算公式得到数据吗 2.日常活动统计数据包含什么内容,怎么获取这些数据? 3. 锻 ...
- 不成生DOM的非主流Blazor UI开源啦!
作者之前介绍了开发中的PixUI,为了适配Web应用采用了将C#通过Roslyn语法语义分析后转换为Javascript的方案,但是这样带来的问题是工程量较大,在短时间内无法达到生产级质量.因此在 ...
- [ARM 汇编]高级部分—性能优化与调试—3.4.3 使用模拟器进行调试与测试
在ARM汇编程序开发过程中,使用模拟器(emulator)进行调试和测试是一种非常有效的方法.模拟器可以在不同的处理器上测试代码,帮助我们发现潜在的问题,并提供丰富的调试功能.本节将介绍如何使用QEM ...
- XTTS系列之二:不可忽略的BCT
重要系统Oracle数据库U2L迁移场景中,如果客户来问我建议,我都会回复说首选就是XTTS,除非XTTS经测试实在是无法满足停机窗口,否则就不要考虑OGG这类方案. 换句话说,选择OGG做迁移的场景 ...
- K8S | 容器和Pod组件
对比软件安装和运行: 一.场景 作为研发人员,通常自己电脑的系统环境都是非常复杂,在个人的习惯上,是按照下图的模块管理电脑的系统环境: 对于「基础设施」.「主机操作系统」.「系统软件」来说,通常只做配 ...