题目:

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. 第十节:Web爬虫之数据存储与MySQL8.0数据库安装和数据插入

    用解析器解析出数据之后,接下来就是存储数据了,保存的形式可以多种多样,最简单的形式是直接保存为文本文件,如 TXT.JSON.csv 另外,还可以保存到数据库中,如关系型数据库MySQL ,非关系型数 ...

  2. noip模拟赛 whzzt-Confidence

    分析:做着感觉像脑筋急转弯一样......因为空间的限制,存不下每一个数,所以用数学方法来解. 设t1=Σai - Σbi = aj - bj,t2=Σi*ai - Σi*bi = j*(aj - b ...

  3. 终于又弄完一个DJANGO BY EXAMPLE的测试

    感觉学到了很多东东,都是以前开发不曾用到的. 知识的作用是到用时,知道到哪里去查相关的功能..:) 这个bookmarks就有很多功能点值得学习呢...

  4. JAVA NIO 之 Selector 组件

    NIO 重要功能就是实现多路复用.Selector是SelectableChannel对象的多路复用器.一些基础知识: 选择器(Selector):选择器类管理着一个被注册的通道集合的信息和它们的就绪 ...

  5. poj——1006 生理周期

    生理周期 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 138291   Accepted: 44300 Descripti ...

  6. sql-server-internals-architecture

    http://kevinekline.com/slides/sql-server-internals-architecture/

  7. Java Number类(数据类型的包装类)

    Java Number 一般地,当需要使用数字的时候,我们通常使用内置数据类型,如:byte.int.long.double等. 例如: int i = 5000; float gpa = 13.65 ...

  8. 【CV知识学习】Fisher Vector

    在论文<action recognition with improved trajectories>中看到fisher vector,所以学习一下.但网上很多的资料我觉得都写的不好,查了一 ...

  9. 猫猫学iOS 之CoreLocation反地理编码小Demo输入经纬度得到城市

    猫猫分享,必须精品 原创文章,欢迎转载.转载请注明:翟乃玉的博客 地址:http://blog.csdn.net/u013357243 一:效果 输入经纬度,能够得到相应的地名 二:思路 跟地里编码差 ...

  10. Javascript中双等号(==)隐性转换机制 JS里charCodeAt()和fromCharCode()方法拓展应用:加密与解密

    Javascript中双等号(==)隐性转换机制   在Javascript中判断相等关系有双等号(==)和三等号(===)两种.其中双等号(==)是值相等,而三等号(===)是严格相等(值及类型是否 ...