逆序對的定義:
對於一個數列
\[ a_1, a_2, a_3, \cdots, a_n \]
逆序對的個數是:
\[ \sum_{i < j, a_i > a_j} 1 \]node
涵涵有兩盒火柴,每盒裝有n根火柴,每根火柴都有一個高度。 如今將每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 兩列火柴之間的距離定義爲:\(∑(a_i−b_i)^2\)ios
其中\(a_i\)表示第一列火柴中第i個火柴的高度,\(b_i\)表示第二列火柴中第i個火柴的高度。數組
每列火柴中相鄰兩根火柴的位置均可以交換,請你經過交換使得兩列火柴之間的距離最小。請問獲得這個最小的距離,最少須要交換多少次?若是這個數字太大,請輸出這個最小交換次數對 \(99,999,997\)取模的結果。數據結構
能夠證實當兩個數列都是通過排序以後的數列可使得\(\sum(a_i-b_i)^2\)取到最小,優化
根據上面的結論,就是說若是對於數列\(\{a_n\}, \{b_n\}\):
\[ a_1,~a_2,~a_3,~\cdots,~a_{n-1},~a_n \\ b_1,~b_2,~b_3,~\cdots,~b_{n-1},~b_n \]
對它們的\(id\)排序以後,原數組排序的過程當中至關於消除逆序對,可是原本的\(id\)是正序的,對於這個過程是對\(id\)增長逆序對的數量。能夠說就因該是當\(a\)數組對於\(b\)數組想要移動到如出一轍所須要花的時間。可是如何求出這個步驟是一個值得討論的問題:spa
首先定於\(q[a[i]] = b[i]\),最終的目標是\(a[i]=b[i] = t\),即\(q[t] = t\)。能夠發現終極目標就是將\(q\)數組進行排序所須要的次數是多少,即求\(q\)的逆序對的個數。指針
代碼:code
#include <iostream> #include <algorithm> using namespace std; const int maxn = 100005; const int P = 99999997; int n; struct node { int id, val; } a[maxn], b[maxn]; int T[maxn], m[maxn]; int lb(int i) { return i & (-i); } bool cmp(node x, node y) { return x.val < y.val; } void add(int i, int delta) { while (i <= n) { T[i] += delta; i += lb(i); } } int sum(int i) { int rtn = 0; while (i > 0) { rtn += T[i]; i -= lb(i); } return rtn; } int main() { cin >> n; for (int i = 1; i <= n; i ++) cin >> a[i].val; for (int i = 1; i <= n; i ++) { cin >> b[i].val; a[i].id = b[i].id = i; } sort(a + 1, a + 1 + n, cmp); sort(b + 1, b + 1 + n, cmp); for (int i = 1; i <= n; i ++) { m[a[i].id] = b[i].id; } int ans = 0; for (int i = 1; i <= n; i ++) { add(m[i], 1); ans = (ans + i - sum(m[i])) % P; } cout << ans << endl; return 0; }
約翰家的N 頭奶牛正在排隊遊行抗議。一些奶牛情緒激動,約翰測算下來,排在第i 位的奶牛的理智度爲Ai,數字可正可負。排序
約翰但願奶牛在抗議時保持理性,爲此,他打算將這條隊伍分割成幾個小組,每一個抗議小組的理智度之和必須大於或等於零。奶牛的隊伍已經固定了先後順序,因此不能交換它們的位置,因此分在一個小組裏的奶牛必須是連續位置的。除此以外,分組多少組,每組分多少奶牛,都沒有限制。遊戲
約翰想知道有多少種分組的方案,因爲答案可能很大,只要輸出答案除以1000000009 的餘數便可。
方程:
\[ f(i) = \sum_{0≤j<i,~\sum_{t = j + 1}^i a[t] ≥0}f(j), ~f(0)=1 \]
時間複雜度:\(O(n^3)\),使用前綴和優化:
\[ f(i) = \sum_{0≤j<i,~s[i]-s[j] ≥0}f(j), ~f(0)=1 \]
時間複雜度:\(O(n^2)\),使用樹狀數組存\(f\),其中下標爲\(s\)。
對於\(s\)數組須要使用離散化,而且\(0\)要加入離散化的過程,由於\(f(0)=1\)。
代碼:
#include <iostream> #include <algorithm> using namespace std; const int maxn = 100005; const int P = 1000000009; int n, a[maxn],s[maxn]; int T[maxn]; int lb(int i) { return i & (-i); } void add_sum(int i, int d) { // 因爲數組的大小加了1,因此要n + 1 while (i <= n + 1) { T[i] = (T[i] + d) % P; i += lb(i); } } int query_sum(int i) { int ans = 0; while (i > 0) { ans = (ans + T[i]) % P; i -= lb(i); } return ans; } int main() { cin >> n; for (int i = 1; i <= n; i ++) { cin >> a[i]; s[i] = s[i-1] + a[i]; a[i] = s[i]; } // 對s進行離散化 sort(a, a + n + 1); for(int i = 0; i <= n; i ++) s[i]=lower_bound(a, a + n + 1, s[i]) - a + 1; // 設置f[0] = 1, 此處的s[0]表示在原數組中第0號元素在離散化過的數組中的位置 add_sum(s[0], 1); int ans = 0; for (int i = 1; i <= n; i ++) { ans = query_sum(s[i]); add_sum(s[i], ans); } cout << ans << endl; return 0; }
約翰有n個牧場,編號爲1到n。它們之間有n−1條道路,每條道路鏈接兩個牧場,經過這些道路,全部牧場都是連通的。
1號牧場裏有個大牛棚,裏面有n頭奶牛。約翰會把它們放出來散步。奶牛按編號順序出發,首先出發的是第一頭奶牛,等它到達了目的地後,第二頭奶牛纔會出發,以後也以此類推。每頭奶牛的目的地都不一樣,其中第iii頭奶牛的目的地是\(t_i\)號牧場。假如編號較大的奶牛,在通過一座牧場的時候,遇到了一頭編號較小的奶牛停在那裏散步,就要和它打個招呼。請你統計一下,每頭奶牛要和多少編號比它小的奶牛打招呼。
首先這道題能夠這樣理解:
全部的奶牛從\(1\)號到\(n\)號依次離開出發到各自的牧場\(t_i\),在通過的道路上若是遇到編號比本身小的牛打招呼,統計總共打多少次招呼。因爲編號從小到大,因此只要路徑上有牛就確定會打招呼。
至此,題目的要求的就是對於每頭牛,統計在去的路徑上有多少頭牛。
定義一個農場編號與牛編號的映射關係:\(cow[t[i]] = i\)。
在\(dfs\)整張圖的過程當中,咱們會發現
因此有如下的代碼:
#include <iostream> using namespace std; const int maxn = 100005; int n, p[maxn], head[maxn], cow[maxn], T[maxn], ans[maxn]; int lb(int i) { return i & (-i); } void modify(int i, int delta) { while (i <= n) { T[i] += delta; i += lb(i); } } int query(int i) { int ret = 0; while (i > 0) { ret += T[i]; i -= lb(i); } return ret; } struct edge { int to, next; } g[maxn * 2]; int ecnt = 2; void add_edge(int u, int v) { g[ecnt] = (edge) {v, head[u]}; head[u] = ecnt ++; } void dfs(int u, int fa) { int current = cow[u]; // 這個頭牛的答案就是查詢:樹狀數組當中編號比它小的,在路徑上的個數和 ans[current] = query(current); // 剛剛進入這個節點(及其子樹),因此把這個節點加入樹狀數組 modify(current, 1); for (int e = head[u]; e != 0; e = g[e].next) if (g[e].to != fa) dfs(g[e].to, u); // 即將退出該節點,不再會訪問到,因此將其從樹狀數組中刪除 modify(current, -1); } int main() { cin >> n; for (int i = 1; i < n; i ++) { int a, b; cin >> a >> b; add_edge(a, b); add_edge(b, a); } for (int i = 1; i <= n; i ++) { cin >> p[i]; cow[p[i]] = i; } dfs(1, 0); for (int i = 1; i <= n; i ++) { cout << ans[i] << endl; } return 0; }
在一些撲克遊戲裏,如德州撲克,發牌是有講究的。通常稱呼專業的發牌手爲荷官。荷官在發牌前,先要銷牌(burn card)。所謂銷牌,就是把當前在牌庫頂的那一張牌移動到牌庫底,它用來防止玩家猜牌而影響遊戲。
假設一開始,荷官拿出了一副新牌,這副牌有N 張不一樣的牌,編號依次爲1到N。因爲是新牌,因此牌是按照順序排好的,從牌庫頂開始,依次爲1, 2,……直到N,N 號牌在牌庫底。爲了發完全部的牌,荷官會進行N 次發牌操做,在第i 次發牌以前,他會連續進行Ri次銷牌操做, Ri由輸入給定。請問最後玩家拿到這副牌的順序是什麼樣的?
舉個例子,假設N = 4,則一開始的時候,牌庫中牌的構成順序爲{1, 2, 3, 4}。
假設R1=2,則荷官應該連銷兩次牌,將1 和2 放入牌庫底,再將3 發給玩家。目前牌庫中的牌順序爲{4, 1, 2}。
假設R2=0,荷官不須要銷牌,直接將4 發給玩家,目前牌庫中的牌順序爲{1,2}。
假設R3=3,則荷官依次銷去了1, 2, 1,再將2 發給了玩家。目前牌庫僅剩下一張牌1。
假設R4=2,荷官在重複銷去兩次1 以後,仍是將1 發給了玩家,這是由於1 是牌庫中惟一的一張牌。
輸入格式:
第1 行,一個整數N,表示牌的數量。
第2 行到第N + 1 行,在第i + 1 行,有一個整數Ri,0<=Ri<N
輸出格式:
第1 行到第N行:第i 行只有一個整數,表示玩家收到的第i 張牌的編號。
維護一個數組,其中存的是每張牌是否還在牌庫當中,若在,則值爲\(1\),反之,值爲\(0\)。
每次摸牌,先銷牌\(s\)張就是在剩下的\(m\)張牌中日後尋找\(s\)張牌就是了,若是還未找到\(s\)就已經爲原狀態的最後一張了,其實只須要進行對\(m\)的牌數進行取模,其實這個想法很是好理解,由於銷牌的這個過程是滾動的。
因此咱們定義\(r_0\)爲原來的找牌的「指針」,\(r_t\)爲找到牌的指針,會有下式:
\[ r_t = (s + r_0) \mod m \]
如今,咱們來思考一下\(r_t\)的意義,其實它就是說剩下的牌中(牌的前後位置關係始終未變,變的是找牌的指針)第\(r_t\)張,也就是說咱們要找到要維護的數組當中前綴和爲\(r_t\)的那個位置就是第\(i\)張牌的位置,即第\(i\)個答案。
從上述的表述能夠理解:咱們須要維護一個樹狀數組,並二分答案。
可是其實有一個比二分答案更爲簡單的作法,就是模擬\(lb\)經過二分的方法訪問\(T[]\)來獲得位置,時間複雜度僅爲\(O(\log n)\),而不是\(O(\log^2n)\)。
代碼:
#include <iostream> #include <cstdio> #include <stdio.h> using namespace std; const int maxn = 700005; const int maxx = 1 << 20; int n, T[maxn]; inline int lb(int i) { return i & (-i); } int query(int x) { int pos = 0; // 第一次查詢的區間最大,其後每次減半,至關於二分,但這裏訪問T數組的時間複雜度爲O(1),故時間複雜度爲O(log n)而不是O(log^2 n) for (int i = maxx; i > 0; i >>= 1) { // 更新位置 int j = i + pos; // 整個過程至關於在進行lb,因此x表明直到pos的前綴和與原來所求的差值,即距query目標還差的一部分 if (j <= n && T[j] <= x) { pos = j; x -= T[j]; } } return pos + 1; } void modify(int i, int d) { while (i <= n) { T[i] += d; i += lb(i); } } int main() { scanf("%d", &n); // O(n)時間建樹狀數組,緣由是a[i]=1 for (int i = 1; i <= n; i ++) T[i] = lb(i); int r = 0; for (int m = n; m > 0; m --) { int s; scanf("%d", &s); r = (r + s) % m; // 在樹狀數組當中查詢值爲r的位置,時間複雜度爲O(n) int pos = query(r); // 因爲這張牌被髮掉了,因此應該將這張牌從樹狀數組當中移除 modify(pos, -1); // 答案就爲這個位置編號 printf("%d\n", pos); } return 0; }