Link:

BZOJ 1805 传送门

Solution:

一道思路比较巧的线段树的题目

首先可以发现,答案和顺序是没有关系的,都是$\sum_{i=1}^n {H_i∗(H_i−1)/2}$。

那么可以比较容易得得到以下的贪心策略:

对于第$i$个船帆,对前$H_i$层中的前$K_i$小的数加1

感性证明:最优方案肯定是每层上数量尽量接近,那么每次放在当前数量最少的层上一定不会使答案变差

此题的难点还是在实现上,

为了对于每个$i$都能区间+1,序列一定要具有单调性

又由于$H_i$是从小到大排序的,每次区间加时注意维护序列从大到小递减即可

但每次对$[H_i-K_i+1,H_i]$+1是不一定能保证单调的,

于是我们找到$seg[H_i-K_i+1]$第一个出现的位置$posr$和结束的位置$posl$,

由于$seg[posl]$更小,我们先尽量选取$[posl,H_i]$,再从$posr$开始选取从而保证单调性

Code:

#include <bits/stdc++.h>

using namespace std;
typedef long long ll;
typedef pair<int,int> P;
#define mid ((l+r)>>1)
#define lc k<<1,l,mid
#define rc k<<1|1,mid+1,r
#define F first
#define S second const int MAXN=1e5+;
const int INF=<<;
int n,maxk=;
ll res=;
P dat[MAXN]; int seg[MAXN<<],tag[MAXN<<]; void push_up(int k){seg[k]=min(seg[k<<],seg[k<<|]);}
void push_down(int k)
{
if(!tag[k]) return;
seg[k<<]+=tag[k];seg[k<<|]+=tag[k];
tag[k<<]+=tag[k];tag[k<<|]+=tag[k];
tag[k]=;
} void build(int k,int l,int r)
{
if(l==r)
{
if(l==maxk) seg[k]=-INF; //要增加最后一位!
else seg[k]=;
return;
}
build(lc);build(rc);
push_up(k);
} int find_val(int x,int k,int l,int r)
{
if(l==r) return seg[k];
push_down(k);
if(x<=mid) return find_val(x,lc);
else return find_val(x,rc);
} int find_pos(int x,int k,int l,int r)
{
if(l==r) return l;
push_down(k);
if(seg[k<<]<=x) return find_pos(x,lc);
else return find_pos(x,rc);
} void Update(int a,int b,int k,int l,int r)
{
if(a<=l && r<=b)
{
seg[k]++;tag[k]++;
return;
}
push_down(k);
if(a<=mid) Update(a,b,lc);
if(b>mid) Update(a,b,rc);
push_up(k);
} void cnt(int k,int l,int r)
{
if(l==maxk) return;
if(l==r)
{
res+=1ll*seg[k]*(seg[k]-)/;
return;
}
push_down(k);
cnt(lc);cnt(rc);
} int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
scanf("%d%d",&dat[i].F,&dat[i].S),maxk=max(maxk,dat[i].F);
sort(dat+,dat+n+);maxk++;
build(,,maxk); for(int i=;i<=n;i++)
{
int num=find_val(dat[i].F-dat[i].S+,,,maxk); //当前位置
int posl=find_pos(num-,,,maxk); //第一个小于num的的数的位置
int posr=find_pos(num,,,maxk); //第一个等于num的数的位置 if(posl<=dat[i].F) Update(posl,dat[i].F,,,maxk);
Update(posr,posr+min(dat[i].S,posl-(dat[i].F-dat[i].S+))-,,,maxk);
} cnt(,,maxk);
printf("%lld",res);
return ;
}

Review:

1、如果序列单调,则可用线段树查询的方式找出第一个小于/大于/等于$x$的数

2、对边界问题还是要敏感些啊,

此题中如果不添加最后一位,并将$seg[maxk]=-INF$,则可能出现$posl=(dat[i].F-dat[i].S+1)$,导致溢出

3、算是又了解了一种在线段树上维护单调性的方法吧,

要先找到这个数第一个出现的位置$pos$,再从$pos$开始操作即可

[BZOJ 1805] Sail 船帆的更多相关文章

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

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

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

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

  3. bzoj1805: [Ioi2007]Sail 船帆

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

  4. bzoj AC倒序

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

  5. bzoj月赛1805

    题目在最后,FG还不会做,等着$NicoDafaGood$和$Achen$给我讲 A 对于每一个质因子建一棵线段树,直接查询就好了 主要是看到所有数的大小都不是很大,然后质因子最多只有log个,复杂度 ...

  6. BZOJ 2127: happiness [最小割]

    2127: happiness Time Limit: 51 Sec  Memory Limit: 259 MBSubmit: 1815  Solved: 878[Submit][Status][Di ...

  7. BZOJ 3275: Number

    3275: Number Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 874  Solved: 371[Submit][Status][Discus ...

  8. BZOJ 2879: [Noi2012]美食节

    2879: [Noi2012]美食节 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 1834  Solved: 969[Submit][Status] ...

  9. bzoj 4610 Ceiling Functi

    bzoj 4610 Ceiling Functi Description bzoj上的描述有问题 给出\(n\)个长度为\(k\)的数列,将每个数列构成一个二叉搜索树,问有多少颗形态不同的树. Inp ...

随机推荐

  1. JavaScript词法作用域与调用对象

    关于 Javascript 的函数作用域.调用对象和闭包之间的关系很微妙,关于它们的文章已经有很多,但不知道为什么很多新手都难以理解.我就尝试用比较通俗的语言来表达我自己的理解吧. 作用域 Scope ...

  2. [SDOI2016] 排列计数 (组合数学)

    [SDOI2016]排列计数 题目描述 求有多少种长度为 n 的序列 A,满足以下条件: 1 ~ n 这 n 个数在序列中各出现了一次 若第 i 个数 A[i] 的值为 i,则称 i 是稳定的.序列恰 ...

  3. POI2007 MEG-Megalopolis [树状数组]

    MEG-Megalopolis 题目描述 Byteotia has been eventually touched by globalisation, and so has Byteasar the ...

  4. [hdu 3652]数位dp解决数的倍数问题

    原以为很好的理解了数位dp,结果遇到一个新的问题还是不会分析,真的是要多积累啊. 解决13的倍数,可以根据当前余数来推,所以把当前余数记为一个状态就可以了. #include<bits/stdc ...

  5. 第九届蓝桥杯C/C++B组题解附代码

    1.标题:第几天 2000年的1月1日,是那一年的第1天.那么,2000年的5月4日,是那一年的第几天? 125天 打开日历就ok 2. 标题:明码 汉字的字形存在于字库中,即便在今天,16点阵的字库 ...

  6. [Usaco2015 dec]Max Flow 树上差分

    [Usaco2015 dec]Max Flow Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 353  Solved: 236[Submit][Sta ...

  7. Win7/8 绿色软件开机启动

    在查找番茄工作法PC端软件时,发现了淡高的文章win8绿色软件开机启动,试用了一下wintabs,的确好用! 另外,office软件中有一款 OFFICE tabs的插件,标签式的管理,非常方便快捷, ...

  8. 【Atcoder】AGC 016 C - +/- Rectangle

    [题意]给定大矩阵的边长H和W,给每格填数(<=|10^9|),要求大矩形总和为正数,而每个h*w的小矩形总和为负数,求构造方式. [算法]数学 [题解]结论题. ★当h|H&& ...

  9. Ubuntu 15.10 安装比特币客户端

    下载 git clone https://github.com/bitcoin/bitcoin.git cd bitcoin ./autogen.sh 安装依赖包: ++-dev sudo apt-g ...

  10. Python爬虫学习 - day1 - 爬取图片

    利用Python完成简单的图片爬取 最近学习到了爬虫,瞬时觉得很高大上,想取什么就取什么,感觉要上天.这里分享一个简单的爬取汽车之家文章列表的图片教程,供大家学习. 需要的知识点储备 本次爬虫脚本依赖 ...