根號算法

分塊與莫隊

1.分塊

大概就是把序列分紅若干塊,預處理出一些東西,整塊打標記,邊角暴力。html

固然不必定是對着序列分塊。可能對着任何東西分塊,好比哈希衝突...ide

複雜度隨着題目相應變化,通常用均值不等式來肯定塊的大小。spa

例題首先是hzwer的分塊9題
指針

說兩個神奇的分塊思想。code

1.當暴力複雜度是n²m,且每一個元素貢獻獨立的時候,能夠把這些元素分紅n0.5塊,分別暴力。htm

總複雜度是n1.5m(有用嗎...)blog

2.對時間/操做分塊。排序

把操做分紅若干塊,塊大小是T。以後對於每一塊,預處理以前的影響,暴力查塊內的影響。隊列

總複雜度是O(預處理複雜度 * T + m² / T * 暴力查詢複雜度)字符串

2.莫隊

先%%%莫濤。

兩個指針跳來跳去......

普通莫隊:就是離線,把詢問排序,使得挪動指針的總次數不大於n1.5

例題:小Z的襪子 小B的詢問  大爺的字符串  小清新人渣的本願 

帶修莫隊:三個維度排序。時間做爲第三關鍵字。

至關於三個指針跳來跳去,第三個指針在時間上。這樣就懂了吧。

例題:洛谷P1903 [國家集訓隊]數顏色 / 維護隊列

樹上莫隊:用括號序列轉成序列莫隊。

先講一下括號序列,DFS序,歐拉序的區別。

  • DFS序就是DFS時,第一次進入點時將其加入序列,長度爲n。樹剖就是用的一種DFS序。
  • 括號序列:出來的時候額外加入一次,長度爲2n。
  • 歐拉序:每次切換節點時,將當前節點加入序列。長度爲2n - 1,由於每一個非根節點都會本身加入一次,讓父親加入一次。

把括號序列搞出來。咱們統計其中出現一次的點。

對於路徑x - y的詢問,先讓first[x] < first[y]

求lca:z。若是z == x,那麼直接用first[x] - first[y]

不然就用last[x] - first[y],而後查詢時加上z。

此處的first和last表示第一,二次在序列中出現的位置。

例題:SP10707 Count on a tree II

  1 #include <cstdio>
  2 #include <algorithm>
  3 #include <cmath>
  4 
  5 const int N = 200010;
  6 
  7 struct Edge {
  8     int nex, v;
  9 }edge[N << 1]; int top;
 10 
 11 int fr[N], e[N], a[N], first[N], last[N], num, val[N], X[N], fa[N][20], pw[N], d[N], n, ans, bin[N];
 12 bool vis[N];
 13 
 14 struct ASK {
 15     int l, r, t, ans, ex;
 16     inline bool operator <(const ASK &w) const {
 17         if(fr[l] != fr[w.l]) {
 18             return l < w.l;
 19         }
 20         return r < w.r;
 21     }
 22 }ask[N];
 23 
 24 inline bool cmp(const ASK &A, const ASK &B) {
 25     return A.t < B.t;
 26 }
 27 
 28 inline void add(int x, int y) {
 29     top++;
 30     edge[top].v = y;
 31     edge[top].nex = e[x];
 32     e[x] = top;
 33     return;
 34 }
 35 
 36 void DFS(int x, int f) {
 37     a[++num] = x;
 38     first[x] = num;
 39     fa[x][0] = f;
 40     d[x] = d[f] + 1;
 41     for(int i = e[x]; i; i = edge[i].nex) {
 42         int y = edge[i].v;
 43         if(y == f) {
 44             continue;
 45         }
 46         DFS(y, x);
 47     }
 48     a[++num] = x;
 49     last[x] = num;
 50     return;
 51 }
 52 
 53 inline int lca(int x, int y) {
 54     if(d[x] > d[y]) {
 55         std::swap(x, y);
 56     }
 57     int t = pw[n];
 58     while(t >= 0 && d[x] != d[y]) {
 59         if(d[fa[y][t]] >= d[x]) {
 60             y = fa[y][t];
 61         }
 62         t--;
 63     }
 64     if(x == y) {
 65         return x;
 66     }
 67     t = pw[n];
 68     while(t >= 0 && fa[x][0] != fa[y][0]) {
 69         if(fa[x][t] != fa[y][t]) {
 70             x = fa[x][t];
 71             y = fa[y][t];
 72         }
 73         t--;
 74     }
 75     return fa[x][0];
 76 }
 77 
 78 inline void add(int x) {
 79     if(!bin[val[x]]) {
 80         ans++;
 81     }
 82     bin[val[x]]++;
 83     return;
 84 }
 85 
 86 inline void del(int x) {
 87     bin[val[x]]--;
 88     if(!bin[val[x]]) {
 89         ans--;
 90     }
 91     return;
 92 }
 93 
 94 inline void work(int p) {
 95     int x = a[p];
 96     if(vis[x]) {
 97         del(x);
 98     }
 99     else {
100         add(x);
101     }
102     vis[x] ^= 1;
103     return;
104 }
105 
106 int main() {
107     int m;
108     scanf("%d%d", &n, &m);
109     for(int i = 2; i <= n; i++) {
110         pw[i] = pw[i >> 1] + 1;
111     }
112     int T = sqrt(n << 1);
113     for(int i = 1; i <= n; i++) {
114         scanf("%d", &val[i]);
115         X[i] = val[i];
116     }
117     std::sort(X + 1, X + n + 1);
118     int temp = std::unique(X + 1, X + n + 1) - X - 1;
119     for(int i = 1; i <= n; i++) {
120         val[i] = std::lower_bound(X + 1, X + temp + 1, val[i]) - X;
121     }
122     for(int i = 1, x, y; i < n; i++) {
123         scanf("%d%d", &x, &y);
124         add(x, y);
125         add(y, x);
126     }
127     DFS(1, 0);
128     for(int j = 1; j <= pw[n]; j++) {
129         for(int i = 1; i <= n; i++) {
130             fa[i][j] = fa[fa[i][j - 1]][j - 1];
131         }
132     }
133     for(int i = 1; i <= n * 2; i++) {
134         fr[i] = (i - 1) / T + 1;
135     }
136     for(int i = 1, x, y; i <= m; i++) {
137         scanf("%d%d", &x, &y);
138         if(first[x] > first[y]) {
139             std::swap(x, y);
140         }
141         int z = lca(x, y);
142         if(z == x) {
143             ask[i].l = first[x];
144             ask[i].r = first[y];
145         }
146         else {
147             ask[i].l = last[x];
148             ask[i].r = first[y];
149             ask[i].ex = z;
150         }
151         ask[i].t = i;
152     }
153 
154     std::sort(ask + 1, ask + m + 1);
155     int l = 1, r = 1;
156     work(1);
157 
158     for(int i = 1; i <= m; i++) {
159         while(l > ask[i].l) {
160             l--;
161             work(l);
162         }
163         while(r < ask[i].r) {
164             r++;
165             work(r);
166         }
167         while(l < ask[i].l) {
168             work(l);
169             l++;
170         }
171         while(r > ask[i].r) {
172             work(r);
173             r--;
174         }
175         if(ask[i].ex) {
176             add(ask[i].ex);
177         }
178         ask[i].ans = ans;
179         if(ask[i].ex) {
180             del(ask[i].ex);
181         }
182     }
183 
184     std::sort(ask + 1, ask + m + 1, cmp);
185     for(int i = 1; i <= m; i++) {
186         printf("%d\n", ask[i].ans);
187     }
188     return 0;
189 }
AC代碼

能夠發現上樹以後代碼量大了不少.....

樹上帶修莫隊:前面兩種嵌套一下就好了。

經典例題是糖果公園,碼量有點大,不過主要是卡常。事實證實O2好用的一批......

  1 #include <cstdio>
  2 #include <algorithm>
  3 #include <cmath>
  4 
  5 typedef long long LL;
  6 const int N = 400010;
  7 
  8 template<class T> inline void read(T &x) {
  9     x = 0;
 10     char c = getchar();
 11     while(c < '0' || c > '9') {
 12         c = getchar();
 13     }
 14     while(c >= '0' && c <= '9') {
 15         x = (x << 3) + (x << 1) + c - 48;
 16         c = getchar();
 17     }
 18     return;
 19 }
 20 
 21 struct Edge {
 22     int nex, v;
 23 }edge[N << 1]; int top;
 24 
 25 int fr[N], e[N], a[N], bin[N], first[N], last[N], num, fa[N][20], d[N], n, col[N], pw[N];
 26 LL ans, W[N], val[N];
 27 bool vis[N];
 28 
 29 struct ASK {
 30     int l, r, t, ex;
 31     LL ans;
 32     inline bool operator <(const ASK &w) const {
 33         if(fr[l] != fr[w.l]) {
 34             return l < w.l;
 35         }
 36         if(r != w.r) {
 37             return r < w.r;
 38         }
 39         return t < w.t;
 40     }
 41 }ask[N]; int topa;
 42 
 43 struct Change {
 44     int p, t, x, last;
 45 }change[N]; int topc;
 46 
 47 inline bool cmp(const ASK &A, const ASK &B) {
 48     return A.t < B.t;
 49 }
 50 
 51 inline void add(int x, int y) {
 52     top++;
 53     edge[top].v = y;
 54     edge[top].nex = e[x];
 55     e[x] = top;
 56     return;
 57 }
 58 
 59 void DFS(int x, int f) {
 60     a[++num] = x;
 61     first[x] = num;
 62     fa[x][0] = f;
 63     d[x] = d[f] + 1;
 64     for(int i = e[x]; i; i = edge[i].nex) {
 65         int y = edge[i].v;
 66         if(y == f) {
 67             continue;
 68         }
 69         DFS(y, x);
 70     }
 71     a[++num] = x;
 72     last[x] = num;
 73     return;
 74 }
 75 
 76 inline int lca(int x, int y) {
 77     if(d[x] > d[y]) {
 78         std::swap(x, y);
 79     }
 80     int t = pw[n];
 81     while(t >= 0 && d[y] > d[x]) {
 82         if(d[fa[y][t]] >= d[x]) {
 83             y = fa[y][t];
 84         }
 85         t--;
 86     }
 87     if(x == y) {
 88         return x;
 89     }
 90     t = pw[n];
 91     while(t >= 0 && fa[x][0] != fa[y][0]) {
 92         if(fa[x][t] != fa[y][t]) {
 93             x = fa[x][t];
 94             y = fa[y][t];
 95         }
 96         t--;
 97     }
 98     return fa[x][0];
 99 }
100 
101 inline void add(int x) {
102     //bin[col[x]]++;
103     (*(bin + (*(col + x))))++;
104     //ans += W[bin[col[x]]] * val[col[x]];
105     ans += (*(W + (*(bin + (*(col + x)))))) * (*(val + (*(col + x))));
106     return;
107 }
108 
109 inline void del(int x) {
110     //ans -= W[bin[col[x]]] * val[col[x]];
111     ans -= (*(W + (*(bin + (*(col + x)))))) * (*(val + (*(col + x))));
112     //bin[col[x]]--;
113     (*(bin + (*(col + x))))--;
114     return;
115 }
116 
117 inline void work(int p) {
118     int x = *(a + p);
119     (*(vis + x)) ? del(x) : add(x);
120     vis[x] ^= 1;
121     return;
122 }
123 
124 int main() {
125     int m, q;
126     read(n);
127     read(m);
128     read(q);
129     int T = pow(n << 1, 2.0 / 3.0);
130     for(int i = 1; i <= m; i++) {
131         read(val[i]);
132     }
133     for(int i = 1; i <= n; i++) {
134         read(W[i]);
135     }
136     for(int i = 1, x, y; i < n; i++) {
137         read(x);
138         read(y);
139         add(x, y);
140         add(y, x);
141     }
142     for(int i = 1; i <= n; i++) {
143         read(col[i]);
144     }
145 
146     DFS(1, 0);
147     for(int i = 2; i <= n; i++) {
148         pw[i] = pw[i >> 1] + 1;
149     }
150     for(int j = 1; j <= pw[n]; j++) {
151         for(int i = 1; i <= n; i++) {
152             fa[i][j] = fa[fa[i][j - 1]][j - 1];
153         }
154     }
155 
156     for(int i = 1, f, x, y; i <= q; i++) {
157         read(f);
158         read(x);
159         read(y);
160         if(!f) { // change
161             ++topc;
162             change[topc].p = x;
163             change[topc].x = y;
164             change[topc].t = i;
165         }
166         else { // ask
167             ++topa;
168             if(first[x] > first[y]) {
169                 std::swap(x, y);
170             }
171             int z = lca(x, y);
172             if(z == x) {
173                 ask[topa].l = first[x];
174                 ask[topa].r = first[y];
175             }
176             else {
177                 ask[topa].l = last[x];
178                 ask[topa].r = first[y];
179                 ask[topa].ex = z;
180             }
181             ask[topa].t = i;
182         }
183     }
184     for(int i = 1; i <= n * 2; i++) {
185         fr[i] = (i - 1) / T + 1;
186     }
187     std::sort(ask + 1, ask + topa + 1);
188 
189     int l = 1, r = 1, time = 0;
190     work(1);
191     for(int i = 1; i <= topa; i++) {
192         while(ask[i].l < l) {
193             work(--l);
194         }
195         while(r < ask[i].r) {
196             work(++r);
197         }
198         while(l < ask[i].l) {
199             work(l++);
200         }
201         while(ask[i].r < r) {
202             work(r--);
203         }
204         while(change[time].t > ask[i].t) {
205             bool f = *(vis + change[time].p);
206             if(f) {
207                 del(change[time].p);
208             }
209             *(col + change[time].p) = change[time].last;
210             if(f) {
211                 add(change[time].p);
212             }
213             time--;
214         }
215         while(time < topc && change[time + 1].t <= ask[i].t) {
216             time++;
217             bool f = *(vis + change[time].p);
218             change[time].last = *(col + change[time].p);
219             if(f) {
220                 del(change[time].p);
221             }
222             *(col + change[time].p) = change[time].x;
223             if(f) {
224                 add(change[time].p);
225             }
226         }
227         if(ask[i].ex) {
228             add(ask[i].ex);
229         }
230         ask[i].ans = ans;
231         if(ask[i].ex) {
232             del(ask[i].ex);
233         }
234     }
235 
236     std::sort(ask + 1, ask + topa + 1, cmp);
237     for(int i = 1; i <= topa; i++) {
238         printf("%lld\n", ask[i].ans);
239     }
240     return 0;
241 }
AC代碼

回滾莫隊:解決很差刪除,只能增長的莫隊。

例:歴史の研究 

相關文章
相關標籤/搜索