# CSES Sort & Search [Link contest](https://lqdoj.edu.vn/contest/csessortandsearch). Cần join group CSES trước. ## Checkpoint 1: Sort, Binary Search, Two Pointer --- ## Distinct sort lại hoặc dùng map, hoặc set: `cout << s.size()` ## Apartments: Sắp xếp $a,b$ tăng dần rồi ghép (chính là 2-pointer). ## Ferris Sắp tăng dần và ghép đứa nhẹ nhất với đứa nặng nhất. ## Sum of Two Values TKNP hoặc hai con trỏ. (Binary search or two-pointer). ## Sum of Three Xem bài BOSOTG Chạy `for` để cố định một số thì quy về bài trên. ## Stick Lengths Cho dãy $a$. Tìm số $x$ để biến đổi toàn dãy về giá trị này, để chi phí $\sum (a_i-x)$ đạt $\min$. Đáp án: Lấy $x$ là trung vị của dãy. Tức phần tử đứng giữa (vị trí $n/2$) trong dãy $a$ đã sắp xếp. Trường hợp $n$ chẵn thì lấy nằm giữa hai giá trị này là được. VD $a = \{1,2,3,7,8,11\}$ thì $x \in [3,7]$ ## Missing Coin Sum Giả sử đã tạo được mọi tổng từ $1$ tới $X$. Nếu cho thêm một xu $x_i$, chắc chắn tạo thêm được đoạn $1 + x_i$ tới $X + x_i$ - TH 1: $X + 1 \ge 1+x_i$. Kết hợp đoạn cũ $[1,X]$ với đoạn mới thì sẽ bao phủ được $[1, X + x_i]$ - TH 2: Sẽ bị "lủng" ở giữa, ít nhất là tổng $X+1$ không tạo được. Lúc mới đầu, ta có $X = 0$. Có thể sort dãy $x$ tăng dần và xét như trên. ## Đếm phân phối `bool mark[N]`. Gán` mark[v] = true` nếu giá trị $v$ đã xuất hiện, ngược lại `false` `int cnt[N]`. Ý nghĩa `cnt[v]` là số lần `v` đã xuất hiện. Gán `cnt[v]++` với mỗi giá trị $v$ ta gặp. Bất kì thông tin gì liên quan, được lưu theo giá trị. ## Collecting Numbers Đề bài: Lần lượt lấy các số $1,2,3,\dots,n$. Mỗi lượt chỉ được đi từ trái sang phải. Cần bao nhiêu lượt? pos[v]: vt xuất hiện. pos[v] < pos[v+1] ## Collecting Numbers 2 Xét sự thay đổi của đáp án. https://cses.fi/paste/cb8b308657ebd68617d3f6/ ## Playlist Hai con trỏ, giải bài toán: "Tìm đoạn dài nhất/ ngắn nhất thỏa mãn tính chất $\alpha$ bất kì". Bài này kết hợp thêm đếm phân phối. ## Học Two pointer, Binary Search ở đâu? ![](https://i.imgur.com/xIkJ5Sz.png) Vào codeforces.com, mục Edu và đăng ký "ITMO Academy: pilot course". Sau đó đọc theory và làm practice ở hai phần này. ![](https://i.imgur.com/42R55iw.png) ## Checkpoint 2: C++ STL (set, multiset, map) & Greedy --- ## Concert ticket ### Cách dùng `multiset` ```cpp= #include<bits/stdc++.h> using namespace std; void print(multiset<int> s) { for (auto val : s) cout << val << ' '; cout << '\n'; } int main() { multiset<int> s({2,2,2,3,3,5,1,4,6,0}); print(s); s.erase(3); print(s); //iterator s.erase(s.find(2)); print(s); //s.erase(s.find(-1)); //RTE if (s.count(-1)) s.erase(s.find(-1)); //get min cout << "Min: " << (*s.begin()) << '\n'; //get max cout << "Max: " << (*s.rbegin()) << '\n'; } ``` Kết quả: ![](https://i.imgur.com/PP4y3s1.png) ### TKNP trên set/multiset ```cpp auto it = s.upper_bound(x); //phần tử nhỏ nhất > x if (it != s.end()) cout << (*it); //lấy ra giá trị lower_bound() thì khác là tìm >= thôi. ``` ### Sol Bài này dùng TKNP trên multiset và erase dần đi những vé đã được mua. ## Towers Bài này dùng chiến thuật "Tham lam" (Greedy). Dùng một multiset lưu kích thước của khối trên cùng mỗi tháp. Đầu tiên là sort giảm dần. Xét từng khối từ lớn tới bé. TKNP trên multiset, xem thử có tháp nào chứa được nó (lower_bound). Nếu không có thì tạo tháp mới. ## Restaurant ### Cách dùng `map` ```cpp= #include<bits/stdc++.h> using namespace std; int main() { map<int,int> mp; mp[1] = 5; mp[2] = 9; mp[5] = 6; for (auto info : mp) cout << info.first << ' ' << info.second << '\n'; } ``` Kết quả: ![](https://i.imgur.com/Haoe7Qt.png) Một cách để hình dung `map` là có thể dùng nó để đếm phân phối với `v` rất lớn ($10^9$) ### Sol Nén số: Không quan tâm chính xác là ai đã đến hoặc đi vào thời điểm nào, chỉ cần giữ đúng thứ tự trước sau thôi. Thêm nữa, cần dùng prefix-sum để tính số khách tại một thời điểm bất kì (Mẹo tăng đoạn $[L,R]$ nhanh: `s[L]++, s[R]--`. Sau cùng chạy for và `s[sau] += s[trước]`) ## Maximum Subarray Sum ### Kĩ thuật Prefix Sum Đặt $s_i$ là tổng $i$ số đầu tiên. Tính bằng công thức $s_i = s_{i-1} + a_i$. Công thức tính tổng đoạn liên tiếp $[l,r]: s[r] - s[l-1]$ Giải thích: Tổng $r$ số đầu tiên, loại đi $l-1$ số ở đầu. Xem thêm: https://vnoi.info/wiki/algo/data-structures/prefix-sum-and-difference-array.md ### Sol Giả sử đang xét đoạn kết thúc tại $i$. Cần tìm $j: 0\le j < i$ để $s_i - s_j$ (tổng của đoạn $[j+1,j+2,\dots,i]$) đạt $\max \rightarrow$ quy về tìm $s_j \min$. ## Movie Festival ###### tags: `QHĐ` Sort các bộ phim theo $b$ tăng dần. Đặt `f[i]` là số phim nhiều nhất xem được nếu phim cuối cùng ta xem là phim thứ $i$ (trong mảng đã sort). TKNP để có được $j$ lớn nhất mà $b_j \le a_i$ với mỗi $i$. Có `f[i] = max(f[1..j]) + 1`. Có thể áp dụng prefix-max? Solution rất giống bài 2 KC, OLP MTTN 2023: https://lqdoj.edu.vn/problem/olp4ck1b2a/editorial ## Các bài khó hơn ## Josephus 2 Cần mô phỏng lại trò chơi. Tại mỗi thời điểm, cần: - tính xem người bị loại tiếp theo đứng thứ mấy trong mảng còn lại? - xóa một người ra khỏi hàng. Dùng Segment Tree lưu `cnt` và TKNP trên segment tree (walk) ## Sliding Median và Cost Tìm trung vị của đoạn "trượt". Nhận thấy mọi thời điểm ta chỉ "xóa một, thêm một". Hướng 1: Nén số, sau đó dùng segment tree trên tập các giá trị. - Cần TKNP - Với bài cost, cần tính tổng --> lưu thêm thông tin vào mỗi nút Hướng 2: Dùng hai cái `multiset` đại diện cho tập lớn và tập bé. - Luôn đảm bảo $\max small \le \min big$ - Kích thước hai tập ngang nhau.