日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

H - Hello Ms. Ze(树状数组套主席树,线段树上二分)

發布時間:2023/12/4 编程问答 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 H - Hello Ms. Ze(树状数组套主席树,线段树上二分) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

H - Hello Ms. Ze

給定nnn種不同的材料,第iii種材料有aia_iai?個,有mmm個操作,操作分為兩類:

  • 把第xxx種材料修改為yyy個,
  • 只用[l,r][l, r][l,r]區間的材料制作衣服,每件衣服要用kkk個不同的材料,最多能做多少件,

先不考慮修改操作,對于區間[l,r][l, r][l,r]中的原料,我們可以做最多多少衣服,如何求解,

考慮二分答案,假設我們可以最多做xxx件衣服,顯然對于個數大于等于xxx的材料,在每件衣服中我們只能使用一次,

假設我們當前二分的區間為[l,r][l, r][l,r],設材料數量大于midmidmid的有aaa種,材料數量小于等于midmidmid的數量和為sumsumsum

a+summid+1≥ka + \frac{sum}{mid + 1} \geq ka+mid+1sum?k,說明答案在右區間,否則答案在左區間,其實這整個過程我們可以在線段樹上直接二分執行。

考慮帶修,樹狀數組套主席樹,然后主席樹上二分即可,由于答案值域在[1,105×106][1, 10 ^ 5 \times 10 ^ 6][1,105×106],所以我直接動態開點建了一顆值域在[1,1<<37][1, 1 << 37][1,1<<37]的主席樹。

樹套樹

#include <bits/stdc++.h>using namespace std;const int N = 1e5 + 10;const long long maxn = 1ll << 37;int a[N], n, m;int root[N], ls[N << 9], rs[N << 9], tot[N << 9], num;long long sum[N << 9];void update(int &rt, long long l, long long r, int x, int v) {if (!rt) {rt = ++num;}sum[rt] += x * v, tot[rt] += v;if (l == r) {return ;}long long mid = l + r >> 1;if (x <= mid) {update(ls[rt], l, mid, x, v);}else {update(rs[rt], mid + 1, r, x, v);} }inline int lowbit(int x) {return x & (-x); }void update(int pos, int x, int v) {while (pos <= n) {update(root[pos], 1, maxn, x, v);pos += lowbit(pos);} }int A[50], B[50], cnt1, cnt2;long long query(long long l, long long r, long long S, int cnt, int k) {if (l == r) {return l;}long long mid = l + r >> 1;long long ans = 0, res = 0;for (int i = 1; i <= cnt1; i++) {ans -= sum[ls[A[i]]];res -= tot[ls[A[i]]];}for (int i = 1; i <= cnt2; i++) {ans += sum[ls[B[i]]];res += tot[ls[B[i]]];}if ((S + ans) / (mid + 1) + cnt - res >= k) {S += ans, cnt -= res;for (int i = 1; i <= cnt1; i++) {A[i] = rs[A[i]];}for (int i = 1; i <= cnt2; i++) {B[i] = rs[B[i]];}return query(mid + 1, r, S, cnt, k);}else {for (int i = 1; i <= cnt1; i++) {A[i] = ls[A[i]];}for (int i = 1; i <= cnt2; i++) {B[i] = ls[B[i]];}return query(l, mid, S, cnt, k);} }int main() {// freopen("in.txt", "r", stdin);// freopen("out.txt", "w", stdout);scanf("%d %d", &n, &m);for (int i = 1; i <= n; i++) {scanf("%d", &a[i]);update(i, a[i], 1);}for (int i = 1, op, l, r, k; i <= m; i++) {scanf("%d %d %d", &op, &l, &r);if (op == 2) {update(l, a[l], -1);a[l] = r;update(l, a[l], 1);}else {scanf("%d", &k);cnt1 = cnt2 = 0;for (int j = l - 1; j; j -= lowbit(j)) {A[++cnt1] = root[j];}for (int j = r; j; j -= lowbit(j)) {B[++cnt2] = root[j];}printf("%lld\n", query(1, maxn, 0, r - l + 1, k));}}return 0; }

總結

以上是生活随笔為你收集整理的H - Hello Ms. Ze(树状数组套主席树,线段树上二分)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。