diff --git a/README.md b/README.md index bc5c4477da48e64e5c9a0411599ec0e2340fb974..06182a23c2c12efef9498f23b21598abc711a9de 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,4 @@ -# Implementações de algoritmos e resoluções de problemas; estudo, treino e consulta para programação competitiva (maratona de programação) +# Implementações de algoritmos e resoluções de problemas -Os algoritmos estão classificados por area de estudo e todos eles estão listados nas Issues. +Estudo, treino e consulta para programação competitiva (maratona de programação) +Os algoritmos estão classificados por categorias e todos eles estão listados nas Issues. diff --git a/geometry/convex_hull.cpp b/algorithms/geometry/convex_hull.cpp similarity index 99% rename from geometry/convex_hull.cpp rename to algorithms/geometry/convex_hull.cpp index e32b285c836b5914b3eeeb92fea2cbadc7c7778c..f1c1109b7872d69e404cba8ac15db3fa63c0483a 100644 --- a/geometry/convex_hull.cpp +++ b/algorithms/geometry/convex_hull.cpp @@ -7,6 +7,7 @@ typedef pair<double,double> dd; + // The three points are a counter-clockwise turn if cross > 0, clockwise if // cross < 0, and collinear if cross = 0 double cross(dd a, dd b, dd c) { diff --git a/graph/articulations_bridges.cpp b/algorithms/graph/articulations_bridges.cpp similarity index 91% rename from graph/articulations_bridges.cpp rename to algorithms/graph/articulations_bridges.cpp index 7d2d4041df9a2e14816cf09a3952ec447c4244a8..62b086868b2ce5e02e1a7ca3c36b19ef0a766b45 100644 --- a/graph/articulations_bridges.cpp +++ b/algorithms/graph/articulations_bridges.cpp @@ -9,9 +9,14 @@ vector<int> graph[MAX]; int cont[MAX], parent[MAX]; int low[MAX], L[MAX]; + +// Answer vector with bridges (edges) vector<ii> bridges; + +// Answer vector with articulations (vertices) vector<int> articulations; + // Finds all articulations and bridges in the graph void dfs(int x) { int child = 0; diff --git a/graph/bfs.cpp b/algorithms/graph/bfs.cpp similarity index 100% rename from graph/bfs.cpp rename to algorithms/graph/bfs.cpp diff --git a/graph/bipartite_match.cpp b/algorithms/graph/bipartite_match.cpp similarity index 100% rename from graph/bipartite_match.cpp rename to algorithms/graph/bipartite_match.cpp diff --git a/graph/dfs.cpp b/algorithms/graph/dfs.cpp similarity index 100% rename from graph/dfs.cpp rename to algorithms/graph/dfs.cpp diff --git a/graph/dijkstra.cpp b/algorithms/graph/dijkstra.cpp similarity index 100% rename from graph/dijkstra.cpp rename to algorithms/graph/dijkstra.cpp diff --git a/graph/floyd_warshall.cpp b/algorithms/graph/floyd_warshall.cpp similarity index 100% rename from graph/floyd_warshall.cpp rename to algorithms/graph/floyd_warshall.cpp diff --git a/graph/kosaraju.cpp b/algorithms/graph/kosaraju.cpp similarity index 100% rename from graph/kosaraju.cpp rename to algorithms/graph/kosaraju.cpp diff --git a/graph/kruskal.cpp b/algorithms/graph/kruskal.cpp similarity index 100% rename from graph/kruskal.cpp rename to algorithms/graph/kruskal.cpp diff --git a/graph/lca.cpp b/algorithms/graph/lca.cpp similarity index 100% rename from graph/lca.cpp rename to algorithms/graph/lca.cpp diff --git a/graph/tarjan.cpp b/algorithms/graph/tarjan.cpp similarity index 100% rename from graph/tarjan.cpp rename to algorithms/graph/tarjan.cpp diff --git a/graph/topological_sort.cpp b/algorithms/graph/topological_sort.cpp similarity index 100% rename from graph/topological_sort.cpp rename to algorithms/graph/topological_sort.cpp diff --git a/math/fast_matrix_pow.cpp b/algorithms/math/fast_matrix_pow.cpp similarity index 100% rename from math/fast_matrix_pow.cpp rename to algorithms/math/fast_matrix_pow.cpp diff --git a/math/modular_multiplicative_inverse.cpp b/algorithms/math/modular_multiplicative_inverse.cpp similarity index 100% rename from math/modular_multiplicative_inverse.cpp rename to algorithms/math/modular_multiplicative_inverse.cpp diff --git a/math/sieve_of_eratosthenes.cpp b/algorithms/math/sieve_of_eratosthenes.cpp similarity index 99% rename from math/sieve_of_eratosthenes.cpp rename to algorithms/math/sieve_of_eratosthenes.cpp index 6bf6c3e3a2b5345d4bcdfc226adb9b91b17c0102..dd3069809d5de7fcc817bc6ac8fc13c7d8588385 100644 --- a/math/sieve_of_eratosthenes.cpp +++ b/algorithms/math/sieve_of_eratosthenes.cpp @@ -5,7 +5,6 @@ * Complexity (Space): O(n) */ - // Returns vector of primes less than or equal to n vector<int> sieve(int n) { vector<int> primes; diff --git a/paradigm/edit_distance.cpp b/algorithms/paradigm/edit_distance.cpp similarity index 100% rename from paradigm/edit_distance.cpp rename to algorithms/paradigm/edit_distance.cpp diff --git a/paradigm/kadane.cpp b/algorithms/paradigm/kadane.cpp similarity index 100% rename from paradigm/kadane.cpp rename to algorithms/paradigm/kadane.cpp diff --git a/paradigm/lis.cpp b/algorithms/paradigm/lis.cpp similarity index 100% rename from paradigm/lis.cpp rename to algorithms/paradigm/lis.cpp diff --git a/paradigm/ternary_search.cpp b/algorithms/paradigm/ternary_search.cpp similarity index 100% rename from paradigm/ternary_search.cpp rename to algorithms/paradigm/ternary_search.cpp diff --git a/string/kmp.cpp b/algorithms/string/kmp.cpp similarity index 100% rename from string/kmp.cpp rename to algorithms/string/kmp.cpp diff --git a/structure/avl.cpp b/algorithms/structure/avl.cpp similarity index 100% rename from structure/avl.cpp rename to algorithms/structure/avl.cpp diff --git a/structure/ball_tree.cpp b/algorithms/structure/ball_tree.cpp similarity index 100% rename from structure/ball_tree.cpp rename to algorithms/structure/ball_tree.cpp diff --git a/structure/bit.cpp b/algorithms/structure/bit.cpp similarity index 100% rename from structure/bit.cpp rename to algorithms/structure/bit.cpp diff --git a/structure/bit2d.cpp b/algorithms/structure/bit2d.cpp similarity index 100% rename from structure/bit2d.cpp rename to algorithms/structure/bit2d.cpp diff --git a/structure/bitmask.cpp b/algorithms/structure/bitmask.cpp similarity index 100% rename from structure/bitmask.cpp rename to algorithms/structure/bitmask.cpp diff --git a/structure/disjoint_set.cpp b/algorithms/structure/disjoint_set.cpp similarity index 100% rename from structure/disjoint_set.cpp rename to algorithms/structure/disjoint_set.cpp diff --git a/structure/lazy_segment_tree.cpp b/algorithms/structure/lazy_segment_tree.cpp similarity index 100% rename from structure/lazy_segment_tree.cpp rename to algorithms/structure/lazy_segment_tree.cpp diff --git a/structure/policy_tree.cpp b/algorithms/structure/policy_tree.cpp similarity index 100% rename from structure/policy_tree.cpp rename to algorithms/structure/policy_tree.cpp diff --git a/structure/segment_tree.cpp b/algorithms/structure/segment_tree.cpp similarity index 100% rename from structure/segment_tree.cpp rename to algorithms/structure/segment_tree.cpp diff --git a/algorithms/structure/trie.cpp b/algorithms/structure/trie.cpp new file mode 100644 index 0000000000000000000000000000000000000000..b1bd70ddbfab7e96ba4ea29542fef21c7e2c7ff4 --- /dev/null +++ b/algorithms/structure/trie.cpp @@ -0,0 +1,82 @@ +/** + * Trie + * + * Complexity (Time): + * Insert -> O(m) + * Search -> O(m) + * Complexity (Space): O(alphabet_size * N) + */ + +// ========== Trie for String ========== +int states = 0; +int data[MAX][26]; + +// ending[i] is true when the node i represents the last +// character of a string contained in the Trie +bool ending[MAX]; + +// Insert word into the Trie +void insert(string word) { + int node = 0; + + for (int i = 0; i < word.size(); ++i) { + int b = word[i] - 'a'; + + if (data[node][b] == -1) + data[node][b] = states++; + node = data[node][b]; + } + + ending[node] = true; +} + +// Return true if word is in the Trie +bool search(string word) { + int node = 0; + + for (int i = 0; i < word.size(); ++i) { + node = data[node][word[i] - 'a']; + if (node == -1) + return false; + } + + return ending[node]; +} + + +// ========== Trie for Integer ========== +int states = 0; +int data[MAX][2]; + +// ending[i] is true when the node i represents the last +// bit of an integer contained in the Trie +bool ending[MAX]; + +// Insert x into the Trie +void insert(int x) { + int node = 0; + + for (int i = 0; i < 32; ++i) { + int b = x >> (31 - i) & 1; + + if (data[node][b] == -1) + data[node][b] = states++; + node = data[node][b]; + } + + ending[node] = true; +} + +// Return true if x is in the Trie +bool search(int x) { + int node = 0; + + for (int i = 0; i < 32; ++i) { + int b = x >> (31 - i) & 1; + node = data[node][b]; + if (node == -1) + return false; + } + + return ending[node]; +} diff --git a/template.cpp b/misc/template.cpp similarity index 97% rename from template.cpp rename to misc/template.cpp index d7a0c748e9a6e4791f676ac6d4357991ff5d3e09..fce5f0bf34f62d12f5cfcce16ba1aad0844f995c 100644 --- a/template.cpp +++ b/misc/template.cpp @@ -1,6 +1,5 @@ #include <bits/stdc++.h> -#define MAX 0 #define MOD 1000000007 #define EPS 1e-6 #define inf 0x3f3f3f3f diff --git a/problems/crise_hidrica.cpp b/problems/crise_hidrica.cpp new file mode 100644 index 0000000000000000000000000000000000000000..1d062589f6679998c9b9d1712d0f2069c813ebaf --- /dev/null +++ b/problems/crise_hidrica.cpp @@ -0,0 +1,141 @@ +#include <bits/stdc++.h> + +#define MAX 5010 +#define MAXLOG 20 +#define MOD 1000000007 +#define EPS 1e-6 +#define inf 0x3f3f3f3f +#define llinf 0x3f3f3f3f3f3f3f3f + +#define fi first +#define se second +#define sz size() +#define pb push_back +#define ende '\n' + +#define all(x) (x).begin(), (x).end() +#define rall(x) (x).rbegin(), (x).rend() +#define mset(x, y) memset(&x, (y), sizeof(x)) + +using namespace std; + +typedef long long ll; +typedef pair<int,int> ii; + +int found; +ll dp[MAX][1010]; +ll v[MAX]; +vector<int> graph[MAX]; + +int h[MAX]; +int par[MAX][MAXLOG]; +ll upd[MAX]; + +void dfs(int va, int p = -1) { + par[va][0] = p; + + if (p + 1) + h[va] = h[p] + 1; + + for (int i = 1; i < MAXLOG; ++i) + if (par[va][i - 1] + 1) + par[va][i] = par[par[va][i - 1]][i - 1]; + + for (auto u : graph[va]) + if (p != u) + dfs(u, va); +} + + +void preprocess(int va) { + memset(par, -1, sizeof par); + dfs(va); +} + + +int query(int p, int q) { + if (h[p] < h[q]) + swap(p, q); + + for (int i = MAXLOG - 1; i >= 0; --i) + if (par[p][i] + 1 && h[par[p][i]] >= h[q]) { + p = par[p][i]; + } + + if (p == q) + return p; + + for (int i = MAXLOG - 1; i >= 0; --i) + if (par[p][i] + 1 && par[p][i] != par[q][i]) { + p = par[p][i]; + q = par[q][i]; + } + + return par[p][0]; +} + + +ll fill(int x, int pare) { + ll ans = 0; + + for (auto i : graph[x]) + if (i != pare) + ans += fill(i, x); + + return v[x] = ans + upd[x]; +} + + +int main() { + ios::sync_with_stdio(0); + cin.tie(0); + + int n, d, a, b; + cin >> n >> d; + for (int i = 0; i < n - 1; ++i) { + cin >> a >> b; + graph[a].pb(b); + graph[b].pb(a); + } + + int m, vv, c; + cin >> m; + vector<int> w(m+1), avail; + for (int i = 0; i < m; ++i) { + cin >> c >> vv; + w[i+1] = vv; + avail.pb(c); + } + + preprocess(1); + + ll q, l; + int x, y; + cin >> q; + for (int i = 0; i < q; ++i) { + cin >> x >> y >> l; + int lca = query(x, y); + + upd[lca] -= l; + upd[x] += l; + upd[y] += l; + if (lca != 1) + upd[par[lca][0]] -= l; + } + + fill(1, -1); + + vector<ll> vvv(m+1); + for (int i = 0; i < m; ++i) + vvv[i+1] = v[avail[i]]; + + for (int i = 1; i <= m; ++i) + for (int j = 0; j <= d; ++j) + if (j >= w[i]) + dp[i][j] = max(dp[i-1][j], dp[i-1][j - w[i]] + vvv[i]); + else + dp[i][j] = dp[i-1][j]; + + cout << dp[m][d] << ende; + return 0; +} diff --git a/problems/escalacao.cpp b/problems/escalacao.cpp new file mode 100644 index 0000000000000000000000000000000000000000..f046333d724ee3e494f34d6c3ae44d449f671375 --- /dev/null +++ b/problems/escalacao.cpp @@ -0,0 +1,107 @@ +#include <bits/stdc++.h> + +#define MAX 1010101 +#define MOD 1000000007 +#define EPS 1e-6 +#define inf 0x3f3f3f3f +#define llinf 0x3f3f3f3f3f3f3f3f + +#define fi first +#define se second +#define sz size() +#define pb push_back +#define ende '\n' + +#define all(x) (x).begin(), (x).end() +#define rall(x) (x).rbegin(), (x).rend() +#define mset(x, y) memset(&x, (y), sizeof(x)) + +using namespace std; + +typedef long long ll; +typedef pair<int,int> ii; + +ll v[MAX]; +ll n, k, r; +vector<ll> tree[MAX]; +vector<ll> res; + +void merge(vector<ll> a, vector<ll> b, vector<ll> &ans) { + int i = 0, j = 0, cnt = 0; + for ( ; cnt < k && i < a.size() && j < b.size(); ++cnt) + if (a[i] > b[j]) + ans.push_back(a[i++]); + else + ans.push_back(b[j++]); + + if (cnt < k) { + if (i < a.size()) + for (; cnt < k && i < a.size(); ++i, ++cnt) + ans.push_back(a[i]); + else + for (; cnt < k && j < b.size(); ++j, ++cnt) + ans.push_back(b[j]); + } +} + + +void build(int node = 1, int a = 0, int b = n - 1) { + if (a > b) + return; + if (a == b) { + tree[node].push_back(v[a]); + return; + } + + build(node * 2, a, (a + b) / 2); + build(node * 2 + 1, 1 + (a + b) / 2, b); + merge(tree[node * 2], tree[node * 2 + 1], tree[node]); +} + + +void query(int i, int j, int node = 1, int a = 0, int b = n - 1) { + if (a > b || a > j || b < i) + return; + if (a >= i && b <= j) { + res.insert(res.end(), tree[node].begin(), tree[node].end()); + return; + } + + query(i, j, node * 2, a, (a + b) / 2); + query(i, j, 1 + node * 2, 1 + (a + b) / 2, b); +} + + +int main() { + ios::sync_with_stdio(0); + cin.tie(0); + + cin >> n >> k >> r; + for (int i = 0; i < n; ++i) + cin >> v[i]; + build(); + + for (int i = 0; i < r; ++i) { + int ans = 1; + res.clear(); + ll a, b; cin >> a >> b; + query(a - 1, b - 1); + + sort(res.begin(), res.end()); + + int cnt = 0; + for (int j = res.size() - 1; j >= 0; --j) { + if (cnt >= k || res[j] == 0) { + if (res[j] == 0 && j == res.size() - 1) + ans = 0; + break; + } + ans = (ans * res[j]) % MOD; + cnt++; + } + + cout << ans << ende; + } + + return 0; +}