Description

有n种颜色的砖块,第i种颜色的砖块有a[i]个,你需要把他们放成一排,使得相邻两个砖块的颜色不相同,限定第一个砖块的颜色是start,最后一个砖块的颜色是end,请构造出一种合法的方案或判断无解。

HINT

【数据范围】

n,m≤1000000,1≤start,end≤n

∑ai<=1000000

Solution

全网唯一 一篇O(n)题解+bzoj最优解

这个题看大家都是优先队列,然后直接贪心放置。

还有用权值线段树来模拟堆过的%%%。

其实不用带logn也可以过的。

大家的方法是从左往右扫过去的。

对于这种插空排序的问题,还有一种考虑方法就是每个种类每个种类来考虑。

好处是,前面放过的种类放完了,和当前第i种永远不会产生冲突。

这就是我的大方向思路。

一、先不考虑端点固定的情况。

其实,不一定要先放最多的。

顺序可以随便。

假设放到完了前i种,那么,一共有sum[i]个。

对于后面的n-i种来说,前i种的方法对后面没有影响。

所以,肯定前i种放法中,选择相邻的情况最少的方案咯!

怎样凑出这个方案?

放完了前i种,设还剩下k个相邻位置。

1.对于第i种,肯定先插那k个位置中。这样每次相邻的-1,已经最优。

2.如果i种还剩下,那就从前面开始插空(不能和1中放的相邻)。这样相邻的数量不增不减。已经最优。

3.如果还剩下,那没有办法了。为了之后好处理,我们都把这些剩下的都放在末尾。

这样,不管你是数量较多的,还是数量较少的,

较多的,可以放在一起,由后面的再插空隔开。

较少的,就隔开之前相邻的。

至于怎么插空?

用一个最普通的链表就可以维护。

当然,我们每次要维护3中,开始连续的那一串的起始位置。方便下次直接访问。

二、有固定点呢?

两个端点比较麻烦。

所以我们就先放端点好了。

放的方法和上面差不多。

先放p,再放q

如果p的数量大于等于q。

那么放q的时候,直接插空,然后无论如何留下一个放末尾。

如果p的数量小于q。

那么放q的时候,插完空,直接往后放完即可。

(注意的是,这样的话有一个情况,就是在最后一个和倒数第二个之间还要插一个,后面放的时候特判一下)

然后放剩下k-2种。

按照刚才的策略即可。注意不能放在1前面,以及最后一个后面。

由于策略一直是最优的。

所以放完了之后,还有相邻元素,那就无解了。

三、一些细节

1.可能有两个端点颜色相同的情况。特判即可。bzoj上还有端点相同,且这个颜色只有一个的数据。。。。

2.刚才“二”中说的那个注意事项。

3.乱七八糟的各种边界情况和+1-1等等。

画个示意图就好理解了。

代码:

全程链表,所以复杂度线性。

(其实应该还有很多常数优化空间2333)

(这个题输入输出优化都很有用,输出优化快了400ms???)

#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<cstring>
#include<iostream>
#define reg register int
#define il inline
#define numb (ch^'0')
using namespace std;
typedef long long ll;
const int N=+;
int nxt[N],id[N];//nxt后继,id颜色编号
int tot;
int k,p,q;
int a[N];
il void rd(int &x){
char ch;x=;
while(!isdigit(ch=getchar()));
for(x=numb;isdigit(ch=getchar());x=x*+numb);
}
il void prin(int x){
if(x/) prin(x/);
putchar(x%+'');
}
il void upda(int o,int to,int d){//初始化链表元素
nxt[o]=to;id[o]=d;
}
int main(){
rd(k);rd(p);rd(q);
for(reg i=;i<=k;++i)rd(a[i]);
if(p==q&&a[p]==){//特判一个点
if(k>) printf("");
else printf("%d",p);
return ;
}
int las=;
for(reg i=;i<=a[p];++i){//放p
upda(++tot,,p);
if(las) nxt[las]=tot;las=tot;
}
las=;
int pos=;//pos是每一次最后的连续一部分同种相邻颜色的起始位置
if(p!=q){//放q
int i;
for(i=;i<=a[q]-&&las<=a[p];++i){
upda(++tot,nxt[las],q);
nxt[las]=tot;++las;
}
if(a[p]>=a[q]){
nxt[a[p]]=++tot;
upda(tot,,q);
pos=a[q];
}
else{
pos=tot;
while(i<=a[q]){
nxt[tot]=tot+;
upda(++tot,,q);
++i;
}
}
}
else{
pos=;
}
int nd=tot;//末尾的编号
for(reg i=;i<=k;++i){//放其他的
if(i==p||i==q) continue;
int tmp=a[i];
while(a[i]&&nxt[pos]!=nd){//插后面的空
upda(++tot,nxt[pos],i);
nxt[pos]=tot;
a[i]--;++pos;
}
if(a[i]&&nxt[pos]==nd&&id[pos]==q){//细节2
upda(++tot,nxt[pos],i);
nxt[pos]=tot;
pos=tot;//warning!!!
a[i]--;
}
if(a[i]){//从前面插空
int now=;//start from a[p]
while(a[i]&&id[nxt[now]]!=i&&nxt[now]!=nd){
upda(++tot,nxt[now],i);
int to=nxt[now];
nxt[now]=tot;
now=to;
a[i]--;
}
if(a[i]){//如果还有剩余
int las=pos;
if(id[pos]!=i) pos=tot+;//warning!!! tot+1
while(a[i]){
upda(++tot,nxt[las],i);
nxt[las]=tot;
las=tot;
a[i]--;
}
}
}
}
for(reg i=;i!=nd;i=nxt[i]){//判断不合法
if(id[i]==id[nxt[i]]){
printf("");return ;
}
}
for(reg i=;i!=nd;i=nxt[i]){
prin(id[i]);putchar(' ');
}prin(id[nd]);
return ;
}

总结:

注意处理排序插空问题的两个大方法:

1.从左到右扫描。期间往往用数据结构维护。

2.分类别,同一个类别一起考虑。往往用到对插入的物品排序(当然本题不用)

[BZOJ3523][Poi2014]KLO-Bricks——全网唯一 一篇O(n)题解+bzoj最优解的更多相关文章

  1. [SDOI2009]Bill的挑战——全网唯一 一篇容斥题解

    全网唯一一篇容斥题解 Description Solution 看到这个题,大部分人想的是状压dp 但是我是个蒟蒻没想到,就用容斥切掉了. 并且复杂度比一般状压低, (其实这个容斥的算法,提出来源于y ...

  2. [JSOI2008]Blue Mary的战役地图——全网唯一一篇dp题解

    全网唯一一篇dp题解 网上貌似全部都是哈希+二分(反正我是大概baidu了翻了翻)(还有人暴力AC了的..) 哈希还是相对于dp还是比较麻烦的. 而且正确性还有可能被卡(当然这个题不会) 而且还容易写 ...

  3. BZOJ3523 [Poi2014]Bricks 【贪心】

    题目链接 BZOJ3523 题解 简单的贪心题 优先与上一个不一样且数量最多的,如果有多个相同,则优先选择非结尾颜色 比较显然,但不知怎么证 #include<algorithm> #in ...

  4. BZOJ3523[Poi2014]Bricks——贪心+堆

    题目描述 有n种颜色的砖块,第i种颜色的砖块有a[i]个,你需要把他们放成一排,使得相邻两个砖块的颜色不相同,限定第一个砖块的颜色是start,最后一个砖块的颜色是end,请构造出一种合法的方案或判断 ...

  5. scrapy-redis+selenium+webdriver解决动态代理ip和user-agent的问题(全网唯一完整代码解决方案)

    问题描述:在爬取一些反爬机制做的比较好的网站时,经常会遇见一个问题就网站代码是通过js写的,这种就无法直接使用一般的爬虫工具爬取,这种情况一般有两种解决方案 第一种:把js代码转为html代码,然后再 ...

  6. 全网唯一正常能用的centos7 安装mysql5.7.35 22 33 25

    CentOS7.4用yum安装并配置MySQL5.7   1.配置YUM源 下载MySQL源安装包 wget http://dev.mysql.com/get/mysql57-community-re ...

  7. (分享)视频压缩Free Video Compressor 汉化版/中文版【全网唯一】

    介绍:Free Video Compressor 是一个免费视频压缩软件,可以帮您有效的压缩视频.电影文件的体积大小,减小占用的磁盘空间,使之更容易放到手机中保存播放Free Video Compre ...

  8. 全网唯一的纯前端实现的canvas支持多图压缩并打包下载的工具

    技术栈: canvas jszip.js(网页端压缩解压缩插件JSZIP库) FileSaver.js(文件保存到本地库) 直接解读源码: <div class="cont" ...

  9. 【微信小程序】mpvue中页面之间传值(全网唯一真正可行的方法,中指推了一下隐形眼镜)

    摘要: mpvue中页面之间传值(注意:是页面之间,不是组件之间) 场景:A页面跳转B页面,在B页面选择商品,将商品名带回A页面并显示 使用api: getCurrentPages step1: A页 ...

随机推荐

  1. 180608-Git工具之Stash

    git stash 暂存 背景: 实际开发过程中,经常可能遇到的一个问题,当你在dev分支上正开发得happy的时候:突然来了个线上bug,得赶紧从release分支上切一个bugfix分支来解决线上 ...

  2. Selenium(Python)生成Html测试报告

    由于Python3已经不支持HTMLTestRunner了, 无论是PyCharm还是pip都无法安装成功, 所以只能去 http://tungwaiyip.info/software/HTMLTes ...

  3. 制作一个App的完整流程是哪些

    APP开发流程其实并不复杂,但是对于客户来说,.一般移动APP开发都离不开UI设计师.前端开发.后端开发.测试专员.产品经理等,由于他们的工作性质都不一样,我们且先把APP软件开发项目分为三个阶段: ...

  4. hackerrank Project Euler #210: Obtuse Angled Triangles

    传送门 做出一个好几个星期屯下来的题目的感觉就是一个字: 爽! 上图的黄点部分就是我们需要求的点 两边的部分很好算 求圆的地方有一个优化,由于圆心是整数点,我们可以把圆分为下面几个部分,阴影部分最难算 ...

  5. JavaScript --经典问题

    JavaScript中如何检测一个变量是一个String类型?请写出函数实现 方法1. function isString(obj){ return typeof(obj) === "str ...

  6. lintcode413 反转整数

    反转整数   将一个整数中的数字进行颠倒,当颠倒后的整数溢出时,返回 0 (标记为 32 位整数). 您在真实的面试中是否遇到过这个题? Yes 样例 给定 x = 123,返回 321 给定 x = ...

  7. JDK源码分析:Integer.java部分源码解析

    1)声明部: public final class Integer extends Number implements Comparable<Integer> extends Number ...

  8. FPGA学习-VGA接口

    一般FPGA开发板的VGA会向用户暴露两共五个种接口,第一种是时序信号,用于同步传输和显示:第二种是色彩信号,用于随着时序把色彩显示到显示器上 时序接口 行同步信号-用于指示一行内像素的显示 场同步信 ...

  9. Mybatis ResultMap(2)

    SQL 映射XML 文件是所有sql语句放置的地方.需要定义一个workspace,一般定义为对应的接口类的路径.写好SQL语句映射文件后,需要在MyBAtis配置文件mappers标签中引用,例如: ...

  10. 【转】Keepalived+Tengine实现高可用集群

    原文出处:http://502245466.blog.51cto.com/7559397/1301772 概述 近年来随着Nginx在国内的发展潮流,越来越多的互联网公司使用Nginx:凭Nginx的 ...