Using a persistent data structure or rollbacking, you are able to simulate deleting from a data structure while only using insertion operations.
DSU With Rollback
DSU with Rollback is an extension to DSU that saves merges and can undo the previous merges.
| Status | Source | Problem Name | Difficulty | Tags | ||
|---|---|---|---|---|---|---|
| YS | Easy | Show TagsDSUrb | ||||
| MMCC | Easy | Show TagsDSUrb | ||||
Warning: Watch Out!
Because path compression is amortized, it does not guarantee runtime. You must use merging by rank. An explanation is given here.
Implementation
Adding on to the usual DSU, we can store the parent and sizes of nodes that are being merged before each merge. This allows us revert each node to their parents before the merge so that the rollback function can use the information to undo the merges.
We can store each state of the DSU using an integer, captured by the snapshot function which returns the number of old merges that has not been rolled back. It's similar to taking a picture of something, and years later going back into your photo album and scrolling up until you find this picture.
For example, if history array stores , this means that before our most recent unite, the representative element for component is , and before that the representative element for component is . If we want to roll back two unites, we would pop the last two elements of the history array and update the representative elements in order.
Furthermore, we can extend this array to roll back component sizes or anything else our DSU may track!
C++
class DSU {private:vector<int> p, sz;// stores previous unitesvector<pair<int &, int>> history;public:DSU(int n) : p(n), sz(n, 1) { iota(p.begin(), p.end(), 0); }int get(int x) { return x == p[x] ? x : get(p[x]); }
Python
class DSU:def __init__(self, n: int):self.p = list(range(n))self.sz = [1] * nself.history = [] # stores all history info related to mergesdef get(self, x) -> int:if self.p[x] == x:return xreturn self.get(self.p[x])
Dynamic Connectivity
Dynamic Connectivity is the most common problem using the deleting trick. These types of problems involve determining whether pairs of nodes are in the same connected component while edges are being inserted and removed.
| Resources | |||||
|---|---|---|---|---|---|
| CP-Algorithms | |||||
| GCP | |||||
| CF | |||||
| Vivek Gupta | |||||
Focus Problem – try your best to solve this problem before continuing!
Solution - Vertex Add Component Sum
For dynamic connectivity problems, we say for every query, there's an interval where it's active. Obviously, for each edge add/remove query, (the index of the query which adds the edge), and (the index of the query which removes the edge) . If an edge never gets removed, then . Note that we assign intervals such that for queries outside the interval, they are completely unaffected by this query. We can use similar reasoning to construct intervals for type and queries.
We can now construct a query tree. If our interval is encapsulated by the the tree's interval, then we can add our query to the node corresponding to the interval. When answering queries, as we enter the interval, we can process all the operations inside the interval. As we exit the interval, we need to undo them. If we are at a leaf, we can answer type queries since we have processed all queries outside this interval . Since we are processing intervals by halves each time, the depth is at most , similar to divide and conquer.
See the code below for implementation details. Note that similar to unite operations, we can also perform and undo operations of type !
C++
#include <bits/stdc++.h>using namespace std;using ll = long long;Code Snippet: DSU (Click to expand)const int MAXN = 3e5;DSU dsu(MAXN);
Python
MAXN = 300000tree = [[] for _ in range(4 * MAXN)]Code Snippet: DSU (Click to expand)class Query:def __init__(self, t: int, u: int, v=None, x=None):self.t = tself.u = u
| Status | Source | Problem Name | Difficulty | Tags | ||
|---|---|---|---|---|---|---|
| CF | Easy | Show TagsDynacon | ||||
| CF | Normal | Show TagsDSUrb | ||||
| CF | Normal | Show TagsDSUrb | ||||
| CF | Normal | Show TagsDSUrb | ||||
| CF | Hard | Show TagsDynacon | ||||
| CF | Hard | Show TagsDynacon | ||||
| Baltic OI | Very Hard | Show TagsD&C, DSUrb | ||||
Module Progress:
Join the USACO Forum!
Stuck on a problem, or don't understand a module? Join the USACO Forum and get help from other competitive programmers!