[國家集訓隊]middle

嘟嘟嘟

有誰能想到這題會用到主席樹呢?(不愧是WJMZBMR出的題)

首先考慮如果區間是固定的話,中位數該怎么求。
沒錯,二分。如果大于當前二分值\(mid\)的數比小于\(mid\)的數多,說明\(mid\)還可以再變大,向右二分;否則向左二分。
如果我們把小于\(mid\)的數都標記成\(-1\),大于的標記成\(1\),那么只用判斷這個區間的和是否\(\geqslant 0\)就行了。

但現在區間不固定。首先\([b + 1, c - 1]\)是一定要選的。對于\([a, b]\)\([c, d]\),因為要讓中位數盡量大,所以應該選\([a, b]\)的最大后綴和以及\([c, d]\)的最大前綴和。

主要思路就是這些,但單次查詢的復雜度是\(O(n \log{n})\)的,過不了。得想辦法優化。

如果對每一個二分的值建一棵區間線段樹(這里的二分在序列中的值中進行就行,而不是\(1\)\(1e9\),所以只用建\(n\)棵),把小于他的都標記成\(1\),大于標記成\(-1\),那么每一次查詢就能達到\(O(\log ^ 2{n})\)了。
但是很顯然這樣空間開不下,而且預處理復雜度過高。所以現在得想辦法減少預處理的時間。
如果把序列中的數排一個序,會發現對于相鄰的兩個不一樣的數(因為數字可能有重),建的線段樹只有一處不一樣,而這一處不一樣只會導致線段樹中的一條鏈改變。所以我們只要單獨把這條鏈提建出來就行了。
然后就會發現這其實就是一棵主席樹呀。

于是這題就寫完了。

#include<cstdio>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cstdlib>
#include<cctype>
#include<vector>
#include<stack>
#include<queue>
using namespace std;
#define enter puts("") 
#define space putchar(' ')
#define Mem(a, x) memset(a, x, sizeof(a))
#define rg register
typedef long long ll;
typedef double db;
const int INF = 0x3f3f3f3f;
const db eps = 1e-8;
const int maxn = 2e4 + 5;
const int maxt = 2e6 + 5;
inline ll read()
{ll ans = 0;char ch = getchar(), last = ' ';while(!isdigit(ch)) {last = ch; ch = getchar();}while(isdigit(ch)) {ans = (ans << 1) + (ans << 3) + ch - '0'; ch = getchar();}if(last == '-') ans = -ans;return ans;
}
inline void write(ll x)
{if(x < 0) x = -x, putchar('-');if(x >= 10) write(x / 10);putchar(x % 10 + '0');
}int n, _n, m, a[maxn], b[maxn], q[4];
vector<int> v[maxn];struct Tree
{int ls, rs;int sum, lmax, rmax;
}t[maxt];
int root[maxn], cnt = 0;
void pushup(int now)
{t[now].sum = t[t[now].ls].sum + t[t[now].rs].sum;t[now].lmax = max(t[t[now].ls].lmax, t[t[now].ls].sum + t[t[now].rs].lmax);t[now].rmax = max(t[t[now].rs].rmax, t[t[now].rs].sum + t[t[now].ls].rmax);
}
void build(int& now, int l, int r)
{if(!now) now = ++cnt;if(l == r) {t[now].sum = t[now].lmax = t[now].rmax = 1; return;}int mid = (l + r) >> 1;build(t[now].ls, l, mid);build(t[now].rs, mid + 1, r);pushup(now);
}
void insert(int old, int& now, int l, int r, int id)
{t[now = ++cnt] = t[old];if(l == r) {t[now].sum = t[now].lmax = t[now].rmax = -1; return;}int mid = (l + r) >> 1;if(id <= mid) insert(t[old].ls, t[now].ls, l, mid, id);else insert(t[old].rs, t[now].rs, mid + 1, r, id);pushup(now);
}
int querySum(int now, int l, int r, int L, int R)
{if(R < L) return 0;if(l == L && r == R) return t[now].sum;int mid = (l + r) >> 1;if(R <= mid) return querySum(t[now].ls, l, mid, L, R);else if(L > mid) return querySum(t[now].rs, mid + 1, r, L, R);else return querySum(t[now].ls, l, mid, L, mid) + querySum(t[now].rs, mid + 1, r, mid + 1, R);
}
int queryL(int now, int l, int r, int L, int R)
{if(l == L && r == R) return t[now].lmax;int mid = (l + r) >> 1;if(R <= mid) return queryL(t[now].ls, l, mid, L, R);else if(L > mid) return queryL(t[now].rs, mid + 1, r, L, R);else{int ret1 = queryL(t[now].ls, l, mid, L, mid);int ret2 = querySum(t[now].ls, l, mid, L, mid) + queryL(t[now].rs, mid + 1, r, mid + 1, R);return max(ret1, ret2);}
}
int queryR(int now, int l, int r, int L, int R)
{if(l == L && r == R) return t[now].rmax;int mid = (l + r) >> 1;if(R <= mid) return queryR(t[now].ls, l, mid, L, R);else if(L > mid) return queryR(t[now].rs, mid + 1, r, L, R);else{int ret1 = queryR(t[now].rs, mid + 1, r, mid + 1, R);int ret2 = querySum(t[now].rs, mid + 1, r, mid + 1, R) + queryR(t[now].ls, l, mid, L, mid);return max(ret1, ret2);}
}bool judge(int x)
{int ans1 = querySum(root[x], 1, n, q[1] + 1, q[2] - 1);int ans2 = queryR(root[x], 1, n, q[0], q[1]);int ans3 = queryL(root[x], 1, n, q[2], q[3]);return ans1 + ans2 + ans3 >= 0;
}
int solve()
{int L = 1, R = _n;while(L < R){int mid = (L + R + 1) >> 1;if(judge(mid)) L = mid;else R = mid - 1;}return L;
}int main()
{n = read();build(root[0], 1, n);for(int i = 1; i <= n; ++i) a[i] = b[i] = read();sort(b + 1, b + n + 1);_n = unique(b + 1, b + n + 1) - b - 1;for(int i = 1; i <= n; ++i){a[i] = lower_bound(b + 1, b + _n + 1, a[i]) - b;v[a[i]].push_back(i);}for(int i = 1; i <= _n; ++i){root[i] = root[i - 1];for(int j = 0; j < (int)v[i - 1].size(); ++j)insert(root[i], root[i], 1, n, v[i - 1][j]);}m = read();for(int i = 1, ans = 0; i <= m; ++i){for(int j = 0; j < 4; ++j) q[j] = read();for(int j = 0; j < 4; ++j) q[j] = (q[j] + ans) % n + 1;sort(q, q + 4);ans = b[solve()];write(ans), enter;}return 0;
}

轉載于:https://www.cnblogs.com/mrclr/p/10074763.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/450444.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/450444.shtml
英文地址,請注明出處:http://en.pswp.cn/news/450444.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Java List<Object>去掉重復對象-java8

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 public List<String> removeStringListDupli(List<String> stringList) {Set<String> set new LinkedHashSet<&g…

Vue + webpack 項目配置化、接口請求統一管理

準備工作 需求由來&#xff1a; 當項目越來越大的時候提高項目運行編譯速度、壓縮代碼體積、項目維護、bug修復......等等成為不得不考慮而且不得不做的問題。 又或者后面其他同事接手你的模塊&#xff0c;或者改你的bug時避免人家看的眼痛以及心里千百句mamaipi...問候。 并且…

Python實現Adaboost

1.Adaboost概念 提升方法的思路是綜合多個分類器&#xff0c;得到更準確的分類結果。 即“三個臭皮匠頂個諸葛亮”。《統計學習方法》稱AdaBoost是提升算法的代表&#xff0c;所謂提升算法&#xff0c;指的是一種常用的統計學習方法&#xff0c;應用廣泛且有效。在分類問題中&a…

Java List<T>去重方法,引用類型集合去重

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 一、實體類中要重寫比較方法equals&#xff0c;最好也重寫hashcode方法 public class WorkWeightDto implements Serializable {privat…

MyBatis知識點

一、MyBatis簡介 1.1 框架概念 框架&#xff0c;就是軟件的半成品&#xff0c;完成了軟件開發過程中的通用操作&#xff0c;程序員只需很少或者不用進行加工就能夠實現特定的功能&#xff0c;從而簡化開發人員在軟件開發中的步驟&#xff0c;提高開發效率。 1.2 常用框架 MVC…

android studio : clang++.exe: error: invalid linker name in argument '-fuse-ld=bfd

公司jenkins上的C編譯器最近換成了clang&#xff0c;今天更新了代碼發現本地的C/C代碼用NDK編譯不過了&#xff0c;提示&#xff1a; “clang.exe: error: invalid linker name in argument -fuse-ldbfd” 解決辦法&#xff1a; 將Android.mk文件中的“LOCAL_LDFLAGS -fuse-ld…

Git知識點

一、Git簡介 1.1 項目的版本管理 在項目開發過程中&#xff0c;項目沒開發到一個節點就會對當前項目進行備份&#xff0c;這個備份就是項目的一個版本&#xff1b;當我們繼續開發一個階段后&#xff0c;再次進行備份&#xff0c;就生成新的版本——多個版本的集合就是項目的版…

(1)初始化項目

2019獨角獸企業重金招聘Python工程師標準>>> &#xff08;1&#xff09;初始化項目 1 使用vue-cli初始化項目 vue init webpack my-renren得到以下輸出&#xff1a; ? Project name my-renren ? Project description A Vue.js project ? Author neumeng <4048…

C語言變量

C語言二進制、八進制、十六進制詳解 什么是二制制? 在數學計算中&#xff0c;二進制計數系統的公分母是最小的&#xff0c;它以2為基數。你還記得在小學或中學時所學的不同的計數系統嗎?筆者在上小學時&#xff0c;曾在一堂數學課中學過以6為基數的計數系統&#xff1b;你先…

Spring Data JPA - 參考文檔 地址

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 Spring Data JPA - 參考文檔 文檔地址

JS內置方法(Array)

concat() 用于連接兩個或多個數組&#xff0c;該方法不會改變現有的數組&#xff0c;而是返回被連接數組的一個副本。join() 把數組中的所有元素放入一個字符串&#xff0c;元素是通過指定的分隔符進行分隔的。若省略了分隔符參數&#xff0c;則使用逗號作為分隔符。push() 向…

模切ERP和免費OA系統是互相結合提高效率

模切ERP和免費OA系統是互相結合提高效率在模切行業中&#xff0c;模切ERP在管理上的作用占了很大的比重&#xff0c;但是免費OA在管理上的地位都不容忽視的。點晴OA的核心問題是如何提高日常的辦公效率問題。因此點晴OA系統里包含的功能是非常全面&#xff0c;如&#xff1a;辦…

maven知識點

一、Maven簡介 1.1 在項目中如何導入jar包&#xff1f; 下載jar包 &#xff08;mvn&#xff09;將下載的jar包拷貝到項目中&#xff08;WEB-INF/lib&#xff09;選擇jar文件–右鍵–Add as Library 1.2 傳統導入jar包的方式存在什么問題&#xff1f; 步驟多&#xff08;相對…

使用SpringBoot yml配置文件

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 1.上一次我們已經使用SpringBoot實現了一個簡單的HelloWord程序&#xff0c;辣么接下來我們簡單的使用一下他的yml格式的配置文件。 2.在…

軟件行業資訊

為什么只有設計師才能發明流行的新語言 先回顧一下知名編程語言的作者和創造時間&#xff1a;Fortran 語言&#xff0c;50年代&#xff0c;IBM 研究員&#xff1b;Lisp 語言&#xff0c;50年代&#xff0c;MIT 的教授和學生&#xff1b;C語言&#xff0c;70年代&#xff0c;貝爾…

spring知識點

一、Spring概述 1.1 web項目開發中的耦合度問題 在Servlet中需要調用service中的方法&#xff0c;則需要在Servlet類中通過new關鍵字創建Service的實例 public interface ProductService{public List<Product> listProducts(); }public class ProductServiceImpl1 imple…

Linux系統下的權限試題測試

不會做的留言&#xff0c;到時在發布答案&#xff01;一、 有兩個參賽團隊team1、team2&#xff0c;兩個團隊各3人, 這兩個團隊互相競爭&#xff0c;各需提交一份報告&#xff0c;每組成員可以修改自己團隊內的所有文件&#xff0c;且不能讓其他團隊的人修改自己的文件內容&…

電子科大軟件系統架構設計——軟件建模詳細設計

文章目錄 軟件建模詳細設計概述軟件建模詳細設計目標軟件建模詳細設計原則開閉原則里氏 (Liskov) 替換原則依賴倒置原則接口分離原則單一職責原則最少知識原則&#xff08;迪米特法則&#xff09;高內聚原則松耦合原則可重用原則 軟件建模詳細設計內容 UML 軟件靜態結構視圖建模…

YAML文件解析

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 YAML是“另一種標記語言”的外語縮寫,YAML 是一種比JSON&#xff08;json多層次{ 與 [ 會被搞暈的&#xff09;更直觀的表現形式&#xf…

120分鐘React快速掃盲教程

在教程開端先說些題外話&#xff0c;我喜歡在學習一門新技術或讀過一本書后&#xff0c;寫一篇教程或總結&#xff0c;既能幫助消化&#xff0c;也能加深印象和發現自己未注意的細節&#xff0c;寫的過程其實仍然是一個學習的過程。有個記錄的話&#xff0c;在未來需要用到相關…