Submission #1108038

#TimeUsernameProblemLanguageResultExecution timeMemory
1108038matuCat in a tree (BOI17_catinatree)C++17
0 / 100
1 ms504 KiB
#include <bits/stdc++.h> // #include <tr2/dynamic_bitset> using namespace std; // using namespace tr2; #pragma GCC optimize("O3,unroll-loops,Ofast") //#pragma GCC target("avx2,bmi,bmi2,lzcnt,popcnt") const long double eps = 1e-9; mt19937 rng(chrono::steady_clock::now().time_since_epoch().count()); int randgen(int left, int right) { return uniform_int_distribution<int>(left, right)(rng); } long long randgenll(long long left, long long right) { return uniform_int_distribution<long long>(left, right)(rng); } typedef long long ll; typedef long double ld; const int mod = 1e9 + 7; struct Mint { int val; Mint(int x = 0) { val = x % mod; } Mint(long long x) { val = x % mod; } Mint operator+(Mint oth) { return val + oth.val; } Mint operator*(Mint oth) { return 1LL * val * oth.val; } Mint operator-(Mint oth) { return val - oth.val + mod; } Mint fp(Mint a, long long n){ Mint p = 1; while(n){ if(n & 1){ p = p * a; } a = a * a; n /= 2; } return p; } Mint operator/(Mint oth){ Mint invers = fp(oth, mod - 2); return 1LL * val * invers.val; } friend ostream& operator << (ostream& os, const Mint& lol){ os << lol.val; return os; } }; struct AINT{ vector<pair<ll,Mint>> aint; vector<pair<ll,Mint>> lazy; pair<ll,Mint> b; AINT(int n){ b = {1e18,Mint(0)}; aint.assign(n * 4 + 1,make_pair(1e18,Mint(0))); lazy.assign(n * 4 + 1, make_pair(1e18, Mint(0))); } void update(int v, int tl, int tr, int l, int r, pair<ll,Mint> val){ if(lazy[v].first != 1e18){ if(aint[v].first == lazy[v].first){ aint[v].second = aint[v].second + lazy[v].second; }else if(aint[v].first > lazy[v].first) aint[v] = lazy[v]; if(tl != tr){ if(lazy[v * 2].first == lazy[v].first){ lazy[v * 2].second = lazy[v * 2].second + lazy[v].second; }else if(lazy[v * 2].first > lazy[v].first) lazy[v * 2] = lazy[v]; if(lazy[v * 2 + 1].first == lazy[v].first){ lazy[v * 2 + 1].second = lazy[v * 2 + 1].second + lazy[v].second; }else if(lazy[v * 2 + 1].first > lazy[v].first){ lazy[v * 2 + 1] = lazy[v]; } } lazy[v] = b; } if(l <= tl && r >= tr){ if(aint[v].first == val.first){ aint[v].second = aint[v].second + val.second; }else if(aint[v].first > val.first){ aint[v] = val; } if(tl != tr){ if(lazy[v * 2].first == val.first){ lazy[v * 2].second = lazy[v * 2].second + val.second; }else if(lazy[v * 2].first > val.first){ lazy[v * 2] = val; } if(lazy[v * 2 + 1].first == val.first){ lazy[v * 2 + 1].second = lazy[v * 2 + 1].second + val.second; }else if(lazy[v * 2 + 1].first > val.first){ lazy[v * 2 + 1] = val; } } return; } if(l > tr || r < tl) return; int tm = (tl + tr) / 2; update(v * 2, tl, tm, l,r , val); update(v * 2 + 1, tm + 1, tr, l, r, val); if(aint[v * 2].first == aint[v * 2 + 1].first){ aint[v] = aint[v * 2]; aint[v].second = aint[v].second + aint[v * 2 + 1].second; }else{ if(aint[v * 2].first < aint[v * 2 + 1].first) aint[v] = aint[v * 2]; else aint[v] = aint[v * 2 + 1]; } } pair<ll,Mint> query(int v, int tl, int tr, int l, int r){ if(lazy[v].first != 1e18){ if(aint[v].first == lazy[v].first){ aint[v].second = aint[v].second + lazy[v].second; }else if(aint[v].first > lazy[v].first) aint[v] = lazy[v]; if(tl != tr){ if(lazy[v * 2].first == lazy[v].first){ lazy[v * 2].second = lazy[v * 2].second + lazy[v].second; }else if(lazy[v * 2].first > lazy[v].first) lazy[v * 2] = lazy[v]; if(lazy[v * 2 + 1].first == lazy[v].first){ lazy[v * 2 + 1].second = lazy[v * 2 + 1].second + lazy[v].second; }else if(lazy[v * 2 + 1].first > lazy[v].first){ lazy[v * 2 + 1] = lazy[v]; } } lazy[v] = b; } if(l <= tl && r >= tr) return aint[v]; if(l > tr || r < tl) return make_pair(1e18,Mint(0)); int tm = (tl + tr) / 2; pair<ll,Mint> p1 = query(v * 2, tl ,tm, l, r); pair<ll,Mint> p2 = query(v * 2 + 1, tm + 1, tr, l,r); if(p1.first == p2.first){ p1.second = p1.second + p2.second; return p1; }else if(p1.first < p2.first) return p1; return p2; // int tm = (tl + tr) / 2; // if(l <= tl && r >= tr) return aint[v]; // if(l > tr || r < tl) return 0; // return max(query(v * 2, tl, tm,l,r), query(v * 2 + 1, tm + 1, tr, l,r)); } }; struct AIB{ vector<int> aib; AIB(int n){ aib.assign(n + 1,0); } void update(int pos, int val){ for(; pos < aib.size(); pos += (pos & -pos)){ aib[pos] += val; } } int query(int pos){ int pl = 0; for(; pos > 0; pos -= (pos & -pos)) pl += aib[pos]; return pl; } }; int red = 0; int ______________________________________________________________________________________________________________________________________________; void solve(){ int n, d; cin >> n >> d; vector<vector<int>> liste(n + 1); vector<deque<int>> dp(n + 1),maxp(n + 1); for(int i = 2; i <= n; i++){ int x; cin >> x; x++; liste[i].push_back(x); liste[x].push_back(i); } int ans = 0; function<void(int,int)> dfs = [&](int nod, int p){ int maxim = 0, nd = 0; for(auto i : liste[nod]){ if(i == p) continue; dfs(i,nod); if(maxim < dp[i].size()){ maxim = dp[i].size(); nd = i; } } if(liste[nod].size() == 1 && nod != 1){ dp[nod].push_back(1); maxp[nod].push_back(1); return; } int crt = 1 + (dp[nd].size() <= d - 1 ? 0 : maxp[nd][d - 1]); swap(dp[nod],dp[nd]); swap(maxp[nd],maxp[nod]); dp[nod].push_front(crt); int lvl = maxp[nod][0]; maxp[nod].push_front(max(crt,lvl)); for(auto i : liste[nod]){ if(i == p || i == nd) continue; int crt = 1 + (dp[i].size() <= d - 1 ? 0 : maxp[i][d - 1]); dp[i].push_front(crt); int lvl = maxp[i][0]; maxp[i].push_front(max(crt,lvl)); // if(nod == 1 && i == 3){ // for(auto j : dp[i]){ // cout << j << " "; // } // } deque<int> new_dp; new_dp = dp[nod]; for(int j = 0; j < dp[i].size(); j++){ for(int l = 0; l < dp[nod].size(); l++){ if(l + j >= d) new_dp[min(l,j)] = max(new_dp[min(l,j)], dp[nod][l] + dp[i][j]); } } dp[nod] = new_dp; } for(auto i : dp[nod]){ ans = max(ans, i); } }; dfs(1,1); cout << ans; } int32_t main(){ cin.tie(0)->sync_with_stdio(0); int t = 1; if(red) cin >> t; while(t--){ solve(); } } /* 4 1 0 0 1 j + 1 + p >= d p >= d - j - 1 i + j >= d j >= d - i; */

Compilation message (stderr)

catinatree.cpp: In member function 'void AIB::update(int, int)':
catinatree.cpp:176:19: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  176 |         for(; pos < aib.size(); pos += (pos & -pos)){
      |               ~~~~^~~~~~~~~~~~
catinatree.cpp: In lambda function:
catinatree.cpp:211:22: warning: comparison of integer expressions of different signedness: 'int' and 'std::deque<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  211 |             if(maxim < dp[i].size()){
      |                ~~~~~~^~~~~~~~~~~~~~
catinatree.cpp:223:38: warning: comparison of integer expressions of different signedness: 'std::deque<int>::size_type' {aka 'long unsigned int'} and 'int' [-Wsign-compare]
  223 |         int crt = 1 + (dp[nd].size() <= d - 1 ? 0 : maxp[nd][d - 1]);
      |                        ~~~~~~~~~~~~~~^~~~~~~~
catinatree.cpp:235:41: warning: comparison of integer expressions of different signedness: 'std::deque<int>::size_type' {aka 'long unsigned int'} and 'int' [-Wsign-compare]
  235 |             int crt = 1 + (dp[i].size() <= d - 1 ? 0 : maxp[i][d - 1]);
      |                            ~~~~~~~~~~~~~^~~~~~~~
catinatree.cpp:247:30: warning: comparison of integer expressions of different signedness: 'int' and 'std::deque<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  247 |             for(int j = 0; j < dp[i].size(); j++){
      |                            ~~^~~~~~~~~~~~~~
catinatree.cpp:248:34: warning: comparison of integer expressions of different signedness: 'int' and 'std::deque<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  248 |                 for(int l = 0; l < dp[nod].size(); l++){
      |                                ~~^~~~~~~~~~~~~~~~
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...