显然答案与杆的顺序无关(按每个高度考虑)。

从高到低考虑杆,设此时的状态为\(S\),\(S[i]\)是高度\(i\)的帆的数目,初始全为\(0\),当前杆的高度为\(h\),杆上需要放置的帆的数目为\(k\),贪心地,假设两个高度的同等宜选,优先选择更高的;帆尽量放置在\(S[i]=0(i\le h)\)的高度上,若还有剩余,设剩下\(t(t\le k)\)个,则放置在除去以选择的高度以外,\(S[i](i\le h)\)前\(t\)小的位置。

整理一下,每次选出\(S[1\cdots h]\)中前\(k\)小的(相同大小选下标较大的)高度放置帆。

{5,3} 0 0 1 1 1  +0
{4,3} 1 1 2 1 1 +1
{4,1} 1 2 2 1 1 +1
{3,2} 2 2 3 1 1 +3
{3,2} 3 3 3 1 1 +4
{2,1} 3 4 3 1 1 +3

但是此时\(S\)似乎不好维护。。考虑将过程倒过来,从低到高考虑杆,(\(S\)一开始为空集,其余定义相同),假设两个高度同等宜选,优先选择更低的,其余大致相同。即每次选出\(S[1\cdots h]\)中前\(k\)小的(相同大小选下标较小的)高度放置帆。

{2,1} 1 0        +0
{3,2} 1 1 1 +0
{3,2} 2 2 1 +2
{4,1} 2 2 1 1 +0
{4,3} 3 2 2 2 +4
{5,3} 3 3 3 2 1 +4

这样用平衡树维护\(S\)(初始大小为0,每次长度变化时补0节点),每次查询全局前\(k\)小然后整体加一即可。(省去了下标范围的约束)。进一步可发现,连下标都不用维护了


巧妙的分割线(之前的splay已经弃坑了)

观察各个时态的\(S\),发现它总是一个单调不增的序列,把某时态的\(S\)中相同且相邻分成一块,如图,蓝色方框表示将选出前\(k\)小的位置,注意被完全覆盖的块的高度可以直接区间+1,而割开的块区间+1的范围是反过来的,显然这样的块最多一块,于是可以上线段树来维护。

答案也不用每步累加,设\(S[i]\)表示最终状态上高度为\(i\)的帆的数目,显然总答案是\(\sum_i\frac{S[i](S[i]-1)}2\),这与因为从本文第一句照应。

#include <bits/stdc++.h>
#define LL long long
const int N=1e5+10; namespace sgt {
struct sgtNode {
int mx,mn,add;
} t[N<<2];
#define ls (x<<1)
#define rs (x<<1|1)
void update(int x) {
t[x].mx=std::max(t[ls].mx,t[rs].mx);
t[x].mn=std::min(t[ls].mn,t[rs].mn);
}
void pushr(int x,int add) {t[x].mn+=add,t[x].mx+=add,t[x].add+=add;}
void pushdown(int x) {
if(t[x].add) pushr(ls,t[x].add),pushr(rs,t[x].add),t[x].add=0;
}
LL calc(int x,int l,int r) {
if(l==r) return 1LL*t[x].mx*(t[x].mx-1)/2;
int mid=(l+r)>>1; pushdown(x);
return calc(ls,l,mid)+calc(rs,mid+1,r);
}
void build(int x,int l,int r) {
t[x].mn=+1e9,t[x].mx=-1e9;
if(l==r) return; int mid=(l+r)>>1;
build(ls,l,mid); build(rs,mid+1,r);
}
void insert(int x,int l,int r,int p) {
if(l==r) return void(t[x].mx=t[x].mn=0);
int mid=(l+r)>>1; pushdown(x);
if(p<=mid) insert(ls,l,mid,p);
else insert(rs,mid+1,r,p);
update(x);
}
void modify(int x,int l,int r,int L,int R) {
if(L>R) return;
if(L<=l&&r<=R) return pushr(x,1);
int mid=(l+r)>>1; pushdown(x);
if(L<=mid) modify(ls,l,mid,L,R);
if(mid<R) modify(rs,mid+1,r,L,R);
update(x);
}
int getVal(int x,int l,int r,int p) {
if(t[x].mn==t[x].mx) return t[x].mn;
int mid=(l+r)>>1; pushdown(x);
if(p<=mid) return getVal(ls,l,mid,p);
else return getVal(rs,mid+1,r,p);
}
int getRangeL(int x,int l,int r,int w) {
if(l==r) return l;
int mid=(l+r)>>1; pushdown(x);
if(t[ls].mn<=w) return getRangeL(ls,l,mid,w);
else return getRangeL(rs,mid+1,r,w);
}
int getRangeR(int x,int l,int r,int w) {
if(l==r) return l;
int mid=(l+r)>>1; pushdown(x);
if(t[rs].mx>=w) return getRangeR(rs,mid+1,r,w);
else return getRangeR(ls,l,mid,w);
}
} int n,m;
std::pair<int,int> a[N]; int main() {
scanf("%d",&n);
for(int i=1; i<=n; ++i) {
scanf("%d%d",&a[i].first,&a[i].second);
m=std::max(m,a[i].first);
}
std::sort(a+1,a+n+1);
sgt::build(1,1,m);
for(int i=1,R=0; i<=n; ++i) {
while(R<a[i].first) sgt::insert(1,1,m,++R);
int pos=R-a[i].second+1; //被割开的位置
int val=sgt::getVal(1,1,m,pos);
int rgl=sgt::getRangeL(1,1,m,val);
int rgr=sgt::getRangeR(1,1,m,val); // std::cout<<val<<' '<<rgl<<' '<<rgr<<std::endl; sgt::modify(1,1,m,rgr+1,R);
sgt::modify(1,1,m,rgl,rgl+a[i].second-(R-rgr)-1);
}
printf("%lld\n",sgt::calc(1,1,m));
return 0;
}

[IOI2007] sails 船帆的更多相关文章

  1. luoguP4647 [IOI2007] sails 船帆

    https://www.luogu.org/problemnew/show/P4647 首先发现答案与顺序无关,令 $ x_i $ 表示高度为 $ i $ 的那一行帆的个数,第 $ i $ 行对答案的 ...

  2. BZOJ1805[Ioi2007]Sail船帆——线段树+贪心

    题目描述 让我们来建造一艘新的海盗船.船上有 N个旗杆,每根旗杆被分成单位长度的小节.旗杆的长度等于它被分成的小节的数目.每根旗杆上会挂一些帆,每张帆正好占据旗杆上的一个小节.在一根旗杆上的帆可以任意 ...

  3. BZOJ.1805.[IOI2007]sail船帆(贪心 线段树)

    BZOJ 洛谷 首先旗杆的顺序没有影响,答案之和在某一高度帆的总数有关.所以先把旗杆按高度排序. 设高度为\(i\)的帆有\(s_i\)个,那么答案是\(\sum\frac{s_i(s_i-1)}{2 ...

  4. bzoj1805: [Ioi2007]Sail 船帆

    可以发现旗杆的顺序是没有用的,对于每列,它的答案是它的最大值mx*(mx+1)/2 高度由小到大排序旗杆,问题可以转化为在前h行选k个最小的值 考虑激情splay乱搞(我只会splay......) ...

  5. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  6. 通过sails和阿里大于实现短信验证

    通过sails与阿里大于来实现注册短信验证码的发送,逻辑图如下 1.用户在客户端发送手机号给服务器,服务器接收到手机号,生成对应时间戳,随机四位数验证码 2.服务器将电话号码和验证码告诉阿里大于服务器 ...

  7. Node.js之sails框架

    先开一坑,有空更新,记录最近钉钉项目上对node及sails框架的学习记录和理解

  8. Node.js~sails.js~package.json的作用

    回到目录 我们在sails框架进行node.js开发时,会涉及到项目的迁移,当迁移后可能你的module即丢失,这时,希望快速的安装所有的包包,可以使用下面命令 1 cd 你当前的sails项目 2 ...

  9. Node.js与Sails~项目结构与Mvc实现

    回到目录 Sails是一个Node.js的中间件架构,帮助我们很方便的构建WEB应用程序,网址:http://www.sailsjs.org/,它主要是在Express框架的基础上发展起来的,扩展了新 ...

随机推荐

  1. Linux计划任务及压缩归档

    计划任务介绍 自动执行,备份数据. crontab    和   at : at:它是一个可以处理仅执行一次就结束的指令 crontab:它是会把你指定的工作或任务,比如:脚本等,按照你设定的周期一直 ...

  2. CAShapLayer的使用1

    1.添加橙色圆环 - (CAShapeLayer *)shapeLayer { if (!_shapeLayer) { _shapeLayer = [CAShapeLayer layer]; CGRe ...

  3. C++11 带来的新特性 (4)—— 匿名函数(Lambdas)

    1 语法 Lambdas并不是新概念,在其它语言中已经烂大街了.直接进入主题,先看语法: [ captures ] ( params ) specifiers exception attr -> ...

  4. C&C++ Calling Convention

    tkorays(tkorays@hotmail.com) 调用约定(Calling Convention) 是计算机编程中一个比较底层的设计,它主要涉及: 函数参数通过寄存器传递还是栈? 函数参数从左 ...

  5. java-数组排序--插入排序

    插入排序 想象着你的左手拿着一手好牌[1,1,1,2,6,6,6,9,9],此时你从桌面上又抽出一张牌[1],你将抽出的牌,从又往左,依次与左手的牌进行比较(只以数字进行对比),当抽出的牌第一次不再大 ...

  6. 学习使人快乐8--Maven

    一.maven基操: MAVEN依赖之 坐标: 二.MAVEN依赖 type:依赖的类型,比如是jar包还是war包等 默认为jar,表示依赖的jar包 optional:标记依赖是否可选.默认值fa ...

  7. poj 2253 floyd最短路

    题目链接 : http://poj.org/problem?id=2253: 思路:这个题主要是理解了意思就行,题目意思是有两只青蛙和若干块石头,现在已知这些东西的坐标,两只青蛙A坐标和青蛙B坐标是第 ...

  8. babel分析

    现在都用 ES6 新语法以及 ES7 新特性来写应用了,但是浏览器和相关的环境还不能友好的支持,需要用到 Babel 转码器来转换成 ES5 的代码 相信大家都看到过如下的名词: babel-pres ...

  9. MYSQL可调用执行自定义SQL的代码

    DELIMITER $$ USE `mysql_wispeed01`$$ DROP PROCEDURE IF EXISTS `sp_execSQL`$$ CREATE DEFINER=`sa`@`%` ...

  10. mstsc的事 随笔

    当个备份吧, 记不得了,就翻一下自己的博客. MSTSC 设置, 平台:Windows 10 企业版 Windows 10 企业版,功能最全.