首页 > 代码库 > BZOJ 2120: 数颜色
BZOJ 2120: 数颜色
2120: 数颜色
Time Limit: 6 Sec Memory Limit: 259 MBSubmit: 3623 Solved: 1396
[Submit][Status][Discuss]
Description
墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问。墨墨会像你发布如下指令: 1、 Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同颜色的画笔。 2、 R P Col 把第P支画笔替换为颜色Col。为了满足墨墨的要求,你知道你需要干什么了吗?
Input
第1行两个整数N,M,分别代表初始画笔的数量以及墨墨会做的事情的个数。第2行N个整数,分别代表初始画笔排中第i支画笔的颜色。第3行到第2+M行,每行分别代表墨墨会做的一件事情,格式见题干部分。
Output
对于每一个Query的询问,你需要在对应的行中给出一个数字,代表第L支画笔到第R支画笔中共有几种不同颜色的画笔。
Sample Input
1 2 3 4 5 5
Q 1 4
Q 2 6
R 1 2
Q 1 4
Q 2 6
Sample Output
4
3
4
HINT
对于100%的数据,N≤10000,M≤10000,修改操作不多于1000次,所有的输入数据中出现的所有整数均大于等于1且不超过10^6。
2016.3.2新加数据两组by Nano_Ape
Source
这真是一道好题,貌似有好多种不同的做法。有人用树套树过的,有人用整体二分过的,有人用带修莫队过的,还有像我一样的蒟蒻用常数优化暴力过的。
虽然2016有新加数据,然而依然卡不住暴力,23333。
RunID | User | Problem | Result | Memory | Time | Language | Code_Length | Submit_Time |
1734100 | YOUSIKI | 2120 | Accepted | 34496 kb | 2340 ms | C++/Edit | 1411 B | 2016-12-11 12:45:43 |
1 #include <bits/stdc++.h> 2 3 const int siz = 10000000; 4 5 char buf[siz], *bit = buf; 6 7 inline int nextInt (void) { 8 register int ret = 0; 9 register int neg = 0;10 11 while (*bit < ‘0‘)12 if (*bit++ == ‘-‘)13 neg ^= true;14 15 while (*bit >= ‘0‘)16 ret = ret*10 + *bit++ - ‘0‘;17 18 return neg ? -ret : ret;19 }20 21 inline char nextChar (void) {22 while (*bit != ‘Q‘ && *bit != ‘R‘)++bit;23 return *bit ++;24 }25 26 const int maxn = 2000000 + 5;27 28 int n, m;29 int cases;30 int total;31 int answer;32 int col[maxn];33 int map[maxn];34 int vis[maxn];35 36 signed main (void) {37 fread (buf, 1, siz, stdin);38 39 n = nextInt ();40 m = nextInt ();41 42 for (register int i = 1; i <= n; ++i) {43 int color = nextInt ();44 if (map[color] == 0)45 map[color] = ++total;46 col[i] = map[color];47 }48 49 for (register int i = 1; i <= n; ++i) {50 switch (nextChar ()) {51 case ‘R‘ : {52 int a = nextInt ();53 int b = nextInt ();54 if (map[b] == 0)55 map[b] = ++total;56 col[a] = map[b];57 break;58 }59 case ‘Q‘ : {60 int a = nextInt ();61 int b = nextInt ();62 answer = 0, ++cases;63 for (register int j = a; j <= b; ++j)64 if (vis[col[j]] != cases)++answer, vis[col[j]] = cases;65 printf("%d\n", answer);66 }67 }68 }69 }
然而不太理解为啥离散化一下就有这么大的用处。
RunID | User | Problem | Result | Memory | Time | Language | Code_Length | Submit_Time |
1734122 | YOUSIKI | 2120 | Time_Limit_Exceed | 26684 kb | 7632 ms | C++/Edit | 1289 B | 2016-12-11 13:00:40 |
1 #include <bits/stdc++.h> 2 3 const int siz = 10000000; 4 5 char buf[siz], *bit = buf; 6 7 inline int nextInt (void) { 8 register int ret = 0; 9 register int neg = 0;10 11 while (*bit < ‘0‘)12 if (*bit++ == ‘-‘)13 neg ^= true;14 15 while (*bit >= ‘0‘)16 ret = ret*10 + *bit++ - ‘0‘;17 18 return neg ? -ret : ret;19 }20 21 inline char nextChar (void) {22 while (*bit != ‘Q‘ && *bit != ‘R‘)++bit;23 return *bit ++;24 }25 26 const int maxn = 2000000 + 5;27 28 int n, m;29 int cases;30 int answer;31 int col[maxn];32 int vis[maxn];33 34 signed main (void) {35 fread (buf, 1, siz, stdin);36 37 n = nextInt ();38 m = nextInt ();39 40 for (register int i = 1; i <= n; ++i)41 col[i] = nextInt ();42 43 for (register int i = 1; i <= m; ++i) {44 switch (nextChar ()) {45 case ‘R‘ : {46 int a = nextInt ();47 int b = nextInt ();48 col[a] = b;49 break;50 }51 case ‘Q‘ : {52 int a = nextInt ();53 int b = nextInt ();54 answer = 0, ++cases;55 for (register int j = a; j <= b; ++j)56 if (vis[col[j]] != cases)++answer, vis[col[j]] = cases;57 printf ("%d\n", answer);58 }59 }60 }61 }
当然,小生来写这道题不是为了练习暴力的,是为了学习带修莫队的。
有别于普通的莫队,带修莫队多了一维对时间的要求,每个询问可以表示为[l,r,t],表示区间需要我们维护到区间[l,r]在时刻t的存在情况。方法是,对于l,r还按照普通莫队的对l分组方式分组,组内对r排序,转移的时候暴力调整时间即可。貌似为了达到最优秀的复杂度,分组时的大小需要调整一下,但这题显然没有那么苛刻啦,o(* ̄▽ ̄*)ブ
1 #include <bits/stdc++.h> 2 3 const int siz = 10000000; 4 5 char buf[siz], *bit = buf; 6 7 inline int nextInt (void) { 8 register int ret = 0; 9 register int neg = 0; 10 11 while (*bit < ‘0‘) 12 if (*bit++ == ‘-‘) 13 neg ^= true; 14 15 while (*bit >= ‘0‘) 16 ret = ret*10 + *bit++ - ‘0‘; 17 18 return neg ? -ret : ret; 19 } 20 21 inline char nextChar (void) { 22 while (*bit != ‘Q‘ && *bit != ‘R‘)++bit; 23 return *bit ++; 24 } 25 26 const int maxn = 200000 + 5; 27 const int maxm = 2000000 + 5; 28 29 int n, m; 30 int l, r; 31 int s, t; 32 int answer; 33 int col[maxn]; 34 int vis[maxm]; 35 36 struct query { 37 int l, r, id, ans; 38 }qry[maxn]; int q; 39 40 struct change { 41 int p, a, b, id; 42 }chg[maxn]; int c; 43 44 inline bool cmp_lr (const query &A, const query &B) { 45 if (A.l / s != B.l / s) 46 return A.l < B.l; 47 else 48 return A.r < B.r; 49 } 50 51 inline bool cmp_id (const query &A, const query &B) { 52 return A.id < B.id; 53 } 54 55 inline void removeColor (int c) { 56 if (--vis[c] == 0)--answer; 57 } 58 59 inline void insertColor (int c) { 60 if (++vis[c] == 1)++answer; 61 } 62 63 inline void removeChange (change &c) { 64 col[c.p] = c.a; 65 if (c.p >= l && c.p <= r) { 66 removeColor(c.b); 67 insertColor(c.a); 68 } 69 } 70 71 inline void insertChange (change &c) { 72 col[c.p] = c.b; 73 if (c.p >= l && c.p <= r) { 74 removeColor(c.a); 75 insertColor(c.b); 76 } 77 } 78 79 inline void solve (query &q) { 80 while (t > 0 && chg[t].id > q.id) 81 removeChange (chg[t--]); 82 while (t < c && chg[t + 1].id < q.id) 83 insertChange (chg[++t]); 84 85 86 87 while (l < q.l)removeColor (col[l++]); 88 while (l > q.l)insertColor (col[--l]); 89 while (r < q.r)insertColor (col[++r]); 90 while (r > q.r)removeColor (col[r--]); 91 92 q.ans = answer; 93 } 94 95 signed main (void) { 96 fread (buf, 1, siz, stdin); 97 98 n = nextInt (); 99 m = nextInt ();100 101 for (register int i = 1; i <= n; ++i)102 col[i] = nextInt ();103 104 for (register int i = 1; i <= m; ++i) {105 switch (nextChar ()) {106 case ‘R‘ : {107 chg[++c].id = i;108 chg[c].p = nextInt ();109 chg[c].b = nextInt ();110 chg[c].a = col[chg[c].p];111 col[chg[c].p] = chg[c].b;112 break;113 }114 case ‘Q‘ : {115 qry[++q].id = i;116 qry[q].l = nextInt ();117 qry[q].r = nextInt ();118 }119 }120 }121 122 s = sqrt (n); t = c; answer = 0; l = 1; r = 0;123 124 std::sort (qry + 1, qry + 1 + q, cmp_lr);125 126 for (register int i = 1; i <= q; ++i)solve(qry[i]);127 128 std::sort (qry + 1, qry + 1 + q, cmp_id);129 130 for (register int i = 1; i <= q; ++i)131 printf ("%d\n", qry[i].ans);132 }
@Author: YouSiki
BZOJ 2120: 数颜色