题目:

BZOJ1483

分析:

(这题码了一下午,码了近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]梦幻布丁(平衡树启发式合并+并查集)的更多相关文章

  1. bzoj1483: [HNOI2009]梦幻布丁(vector+启发式合并)

    1483: [HNOI2009]梦幻布丁 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 4022  Solved: 1640[Submit][Statu ...

  2. bzoj1483: [HNOI2009]梦幻布丁(链表+启发式合并)

    题目大意:一个序列,两种操作. ①把其中的一种数修改成另一种数 ②询问有多少段不同的数如1 2 2 1为3段(1 / 2 2 / 1). 昨晚的BC的C题和这题很类似,于是现学现写居然过了十分开心. ...

  3. 【BZOJ1483】[HNOI2009]梦幻布丁 链表+启发式合并

    [BZOJ1483][HNOI2009]梦幻布丁 Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1,2 ...

  4. BZOJ 1483: [HNOI2009]梦幻布丁( 链表 + 启发式合并 )

    把相同颜色的串成一个链表, 然后每次A操作就启发式合并, 然后计算对答案的影响. ----------------------------------------------------------- ...

  5. BZOJ 1483: [HNOI2009]梦幻布丁 [链表启发式合并]

    1483: [HNOI2009]梦幻布丁 题意:一个带颜色序列,一种颜色合并到另一种,询问有多少颜色段 一种颜色开一个链表,每次遍历小的合并到大的里,顺带维护答案 等等,合并方向有规定? 令col[x ...

  6. bzoj 1483: [HNOI2009]梦幻布丁 (链表启发式合并)

    Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色. 例如颜色分别为1,2,2,1的四个布丁一共有3段颜色. Input ...

  7. 洛谷P3201 [HNOI2009]梦幻布丁(链表 + 启发式合并)

    题目链接 给出 \(n\) 个布丁,每个补丁都有其颜色.现在有 \(m\) 次操作,每次操作将第 \(x_i\) 种颜色全部变为第 \(y_i\) 种颜色. 操作中可能会插入询问,回答目前总共有多少段 ...

  8. [BZOJ 1483] [HNOI2009] 梦幻布丁 (线段树合并)

    [BZOJ 1483] [HNOI2009] 梦幻布丁 (线段树合并) 题面 N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1 ...

  9. BZOJ 2733: [HNOI2012]永无乡(treap + 启发式合并 + 并查集)

    不难...treap + 启发式合并 + 并查集 搞搞就行了 --------------------------------------------------------------------- ...

随机推荐

  1. PHP代码静态分析工具PHPStan

    最近发现自己写的PHP代码运行结果总跟自己预想的不一样,排查时发现大多是语法错误,在运行之前错误已经种下.可能是自己粗心大意,或者说php -l检测太简单,不过的确是有一些语法错误埋藏得太深(毕竟PH ...

  2. js中匿名函数的N种写法

    匿名函数没有实际名字,也没有指针,怎么执行? 关于匿名函数写法,很发散~ +号是让函数声明转换为函数表达式.汇总一下 最常见的用法: 代码如下: (function() {  alert('water ...

  3. 【Codeforces 369C】 Valera and Elections

    [链接] 我是链接,点我呀:) [题意] 给你一棵树 让你选择若干个修理点. 这些修理点被选中之后,节点i到1号节点之间的所有"坏路"都会被修好 问最少需要选择多少个点才能将所有的 ...

  4. ansible plugins简介

    ansible插件是增强ansible的核心功能的代码片段,ansible使用插件架构来实现丰富,灵活和可扩展的功能集. Ansible提供了许多方便的插件,您可以轻松编写自己的插件. 下边简单介绍A ...

  5. POJ 1226 Substrings

    Substrings Time Limit: 1000ms Memory Limit: 10000KB This problem will be judged on PKU. Original ID: ...

  6. RabbitMQ-基本概念(一)

    整体架构模型 Producer 消息生产者,生产者创建消息然后发布到RabbitM中,消息一般包含2个部分 消息体(payload)和标签 消息体就是带有业务逻辑结构的数据,消息标签用来表述这条消息, ...

  7. Java获取系统环境变量(System Environment Variable)和系统属性(System Properties)以及启动参数的方法

    系统环境变量(System Environment Variable): 在Linux下使用export $ENV=123指定的值.获取的方式如下: Map<String,String> ...

  8. SiteMesh2-sitemesh.xml的ParameterDecoratorMapper映射器的用法

    继续使用上一章http://www.cnblogs.com/EasonJim/p/7086916.html的例子,改造成使用ParameterDecoratorMapper映射器的方法,这个映射器不需 ...

  9. 使用SQL Profile及SQL Tuning Advisor固定运行计划

    SQL Profile就是为某一SQL语句提供除了系统统计信息.对象(表和索引等)统计信息之外的其它信息,比方执行环境.额外的更准确的统计信息,以帮助优化器为SQL语句选择更适合的执行计划. SQL ...

  10. jQuery的立即调用表达式

    立即调用表达式 任何库与框架设计的第一要要点就是解决命名空间与变量污染的问题.jquery就是用javascript函数作用域的特性,采用立即调用表达式包裹了自身的方法来解决这个问题的. jQuery ...