【BZOJ1483】[HNOI2009]梦幻布丁(平衡树启发式合并+并查集)
题目:
分析:
(这题码了一下午,码了近250行,但是意外跑的比本校各位神仙稍快,特写博客纪念)
首先能看出一个显然的结论:颜色段数只会变少不会变多。
我们考虑用并查集维护区间,对于每个区间维护它的起点和终点。建\(n\)棵平衡树,第\(i\)棵存颜色为\(i\)的区间。把\(x\)变成\(y\)时进行启发式合并,同时对于\(x\)上的每个结点\([a,b]\),在\(y\)中找\(a-1\)和\(b+1\)所在区间。如果存在则合并,并答案减\(1\);若不存在则向\(y\)中插入新结点。时间复杂度\(O(m\log ^2n)\)
代码:
代码不难写,就是长……
#include <cstdio>
#include <cstring>
#include <cctype>
#include <algorithm>
using namespace std;
namespace zyt
{
template<typename T>
inline void read(T &x)
{
char c;
bool f = false;
x = 0;
do
c = getchar();
while (c != '-' && !isdigit(c));
if (c == '-')
f = true, c = getchar();
do
x = x * 10 + c - '0', c = getchar();
while (isdigit(c));
if (f)
x = -x;
}
template<typename T>
inline void write(T x)
{
static char buf[20];
char *pos = buf;
if (x < 0)
putchar('-'), x = -x;
do
*pos++ = x % 10 + '0';
while (x /= 10);
while (pos > buf)
putchar(*--pos);
}
const int N = 1e6 + 10;
int n, m, p[N], st[N], ed[N];
int f(const int x)
{
return x == p[x] ? x : p[x] = f(p[x]);
}
int ans = 0;
class Splay
{
private:
struct node
{
int val, size;
node *fa, *s[2];
node(const int _val, node *_fa)
: val(_val), fa(_fa)
{
size = 1;
s[0] = s[1] = NULL;
}
}*head;
bool dir(const node *rot)
{
return rot == rot->fa->s[1];
}
void update(node *rot)
{
rot->size = 1;
if (rot->s[0])
rot->size += rot->s[0]->size;
if (rot->s[1])
rot->size += rot->s[1]->size;
}
void rotate(node *rot)
{
node *f = rot->fa, *ff = f->fa;
bool d = dir(rot);
rot->fa = ff;
if (ff)
ff->s[dir(f)] = rot;
else
head = rot;
f->s[d] = rot->s[!d];
if (rot->s[!d])
rot->s[!d]->fa = f;
rot->s[!d] = f;
f->fa = rot;
update(f);
}
void splay(node *rot, const node *goal = NULL)
{
while (rot && rot->fa && rot->fa != goal)
{
node *f = rot->fa, *ff = f->fa;
if (ff == goal)
rotate(rot);
else if (dir(rot) ^ dir(f))
rotate(rot), rotate(rot);
else
rotate(f), rotate(rot);
}
update(rot);
}
void del(node *rot, Splay &s)
{
if (!rot)
return;
int x = f(rot->val);
bool flag = false;
if (st[x] > 1 && s.find(f(st[x] - 1)))
{
p[x] = f(st[x] - 1);
ed[f(st[x] - 1)] = ed[x];
--ans, flag = true;
}
x = f(x);
if (ed[x] < n && s.find(f(ed[x] + 1)))
{
p[x] = f(ed[x] + 1);
st[f(ed[x] + 1)] = st[x];
--ans, flag = true;
}
if (!flag)
s.insert(rot->val);
if (rot->s[0])
del(rot->s[0], s);
if (rot->s[1])
del(rot->s[1], s);
delete rot;
}
node *find(const int val)
{
node *rot = head;
while (1)
{
if (!rot || rot->val == val)
return rot;
if (val < rot->val)
rot = rot->s[0];
else
rot = rot->s[1];
}
}
public:
void insert(const int val)
{
if (!head)
{
head = new node(val, NULL);
return;
}
node *rot = head;
while (1)
{
if (val < rot->val)
{
if (rot->s[0])
rot = rot->s[0];
else
{
rot->s[0] = new node(val, rot);
splay(rot->s[0]);
break;
}
}
else
{
if (rot->s[1])
rot = rot->s[1];
else
{
rot->s[1] = new node(val, rot);
splay(rot->s[1]);
break;
}
}
}
}
size_t size()
{
if (head)
return head->size;
else
return 0;
}
friend void merge(Splay &a, Splay &b);
}tree[N];
inline void merge(Splay &a, Splay &b)
{
if (a.size() < b.size())
swap(a, b);
b.del(b.head, a);
b.head = NULL;
}
int arr[N], tmp[N];
int work()
{
read(n), read(m);
int last = 1;
for (int i = 1; i <= n; i++)
read(arr[i]);
p[1] = st[1] = ed[1] = 1;
for (int i = 2; i <= n; i++)
{
if (arr[i] != arr[i - 1])
{
tree[arr[i - 1]].insert(last);
st[i] = last = i, ++ans;
}
p[i] = last;
ed[last] = i;
}
tree[arr[n]].insert(last), ++ans;
while (m--)
{
int opt;
read(opt);
if (opt == 1)
{
int x, y;
read(x), read(y);
if (x != y)
merge(tree[y], tree[x]);
}
else
write(ans), putchar('\n');
}
return 0;
}
}
int main()
{
return zyt::work();
}
【BZOJ1483】[HNOI2009]梦幻布丁(平衡树启发式合并+并查集)的更多相关文章
- bzoj1483: [HNOI2009]梦幻布丁(vector+启发式合并)
1483: [HNOI2009]梦幻布丁 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 4022 Solved: 1640[Submit][Statu ...
- bzoj1483: [HNOI2009]梦幻布丁(链表+启发式合并)
题目大意:一个序列,两种操作. ①把其中的一种数修改成另一种数 ②询问有多少段不同的数如1 2 2 1为3段(1 / 2 2 / 1). 昨晚的BC的C题和这题很类似,于是现学现写居然过了十分开心. ...
- 【BZOJ1483】[HNOI2009]梦幻布丁 链表+启发式合并
[BZOJ1483][HNOI2009]梦幻布丁 Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1,2 ...
- BZOJ 1483: [HNOI2009]梦幻布丁( 链表 + 启发式合并 )
把相同颜色的串成一个链表, 然后每次A操作就启发式合并, 然后计算对答案的影响. ----------------------------------------------------------- ...
- BZOJ 1483: [HNOI2009]梦幻布丁 [链表启发式合并]
1483: [HNOI2009]梦幻布丁 题意:一个带颜色序列,一种颜色合并到另一种,询问有多少颜色段 一种颜色开一个链表,每次遍历小的合并到大的里,顺带维护答案 等等,合并方向有规定? 令col[x ...
- bzoj 1483: [HNOI2009]梦幻布丁 (链表启发式合并)
Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色. 例如颜色分别为1,2,2,1的四个布丁一共有3段颜色. Input ...
- 洛谷P3201 [HNOI2009]梦幻布丁(链表 + 启发式合并)
题目链接 给出 \(n\) 个布丁,每个补丁都有其颜色.现在有 \(m\) 次操作,每次操作将第 \(x_i\) 种颜色全部变为第 \(y_i\) 种颜色. 操作中可能会插入询问,回答目前总共有多少段 ...
- [BZOJ 1483] [HNOI2009] 梦幻布丁 (线段树合并)
[BZOJ 1483] [HNOI2009] 梦幻布丁 (线段树合并) 题面 N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1 ...
- BZOJ 2733: [HNOI2012]永无乡(treap + 启发式合并 + 并查集)
不难...treap + 启发式合并 + 并查集 搞搞就行了 --------------------------------------------------------------------- ...
随机推荐
- 利用postman进行接口测试并发送带cookie请求的方法
做web测试的基本上都用用到postman去做一些接口测试,比如测试接口的访问权限,对于某些接口用户A可以访问,用户B不能访问:比如有时需要读取文件的数据.在postman上要实现这样测试,我们就必要 ...
- 什么是restful
- Spring核心技术(三)——Spring的依赖及其注入(续)
本文将继续前文,针对依赖注入的细节进行描述 依赖注入细节 如前文所述,开发者可以通过定义Bean的依赖的来引用其他的Bean或者是一些值的,Spring基于XML的配置元数据通过支持一些子元素< ...
- hadoop 3.0.0新特性
1.Minimum required Java version increased from Java 7 to Java 8 java最低支持版本变成java8 2.Support for eras ...
- Extended symmetrical multiprocessor architecture
An architecture for an extended multiprocessor (XMP) computer system is provided. The XMP computer s ...
- 2102 石子归并 2codevs
2102 石子归并 2codevs 题目描述 Description 在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为 ...
- linux top进程状态D
什么是D状态 运行在KVM虚拟机里的一些进程突然出了问题,这些出了问题的进程无法用kill杀掉,使用ps可以看到这些进程处于D状态: [build@kbuild-john ~]$ ps -a -o p ...
- mybatis指定jdbctype
MyBatis 插入空值时,需要指定JdbcType mybatis insert空值报空值异常,但是在pl/sql不会提示错误,主要原因是mybatis无法进行转换 所以在MyBatis映射文件中要 ...
- Xcode9自带版本控制系统(Git)的使用
情形一:新项目工程 前提是首先你得有个Github账号. 1.创建新工程,选中使用git,创建本地仓库. 2.在偏好设置添加git账号. 3.创建Github远程仓库,xcode会自动帮你在Githu ...
- Magento 模块开发之DispatchEvent
在这一章节中.我们来了解 Magento 中的事件分发机制 Mage::dispatchEvent() 在创建自己的模块时, Event 事件的分发将会变成十分实用且有效 以个人的经验. 事件的分发使 ...