제출 #537773

#제출 시각아이디문제언어결과실행 시간메모리
537773Lobo구슬과 끈 (APIO14_beads)C++17
57 / 100
135 ms131072 KiB
#include<bits/stdc++.h> using namespace std; const long long inf = (long long) 1e15 + 10; const int inf1 = (int) 1e9 + 10; #define int long long #define dbl long double #define endl '\n' #define sc second #define fr first #define mp make_pair #define pb push_back #define all(x) x.begin(), x.end() #define maxn 220000 int n, a[maxn], b[maxn], c[maxn]; vector<int> dp[maxn][5][2], mark[maxn][5][2]; vector<pair<int,int>> g[maxn]; /* u = vertice p = como esta 0 = liga pro pai ou pra algum filho com azul e o outro azul que vai ligar para esse segundo 3 = liga pro pai ou pra algum filho com azul e esse cara que vai ligar para o primeiro 1 = liga pro pai com red ou nao pode mais ligar pra ninguem / ja fez sua ligacao 2 = pode ligar para alguem com red ou para 2 com azul / ainda nao ligou pra ninguem f = pode nao usar nada 1 = tem que nao usar 0 = tem que usar */ int sol(int u, int p, int f, int id, int ant) { if(mark[u][p][f][id]) return dp[u][p][f][id]; mark[u][p][f][id] = 1; if(p == 0) { //tem que ligar para alguem com azul e nao se preocupa //se ja tiver passado por todo mundo é -inf if(id == g[u].size()) return dp[u][p][f][id] = -inf; int v = g[u][id].fr; int w = g[u][id].sc; if(v == ant) return dp[u][p][f][id] = sol(u,p,f,id+1,ant); int ans = -inf; //se o outro azul é v ans = max(ans, w + sol(v,2,0,0,u) + sol(u,1,f,id+1,ant)); ans = max(ans, w + sol(v,2,f,0,u) + sol(u,1,0,id+1,ant)); //só passa para o próximo podendo ligar para ele com azul //(e o outro azul que vai estar ocupado) ou red ans = max(ans, max(sol(v,1,0,0,u),w+sol(v,3,0,0,u)) + sol(u,0,f,id+1,ant)); ans = max(ans, max(sol(v,1,f,0,u),w+sol(v,3,f,0,u)) + sol(u,0,0,id+1,ant)); // cout << u << " ja ligou azul p = 0 || f = " << f << " atual no " << v << " id " << id << " = " << " = " << ans << endl; return dp[u][p][f][id] = ans; } else if(p == 3) { //tem que ligar para alguem com azul e o filho azul nao pode ligar para mais ninguem //se ja tiver passado por todo mundo é -inf if(id == g[u].size()) return dp[u][p][f][id] = -inf; int v = g[u][id].fr; int w = g[u][id].sc; if(v == ant) return dp[u][p][f][id] = sol(u,p,f,id+1,ant); int ans = -inf; //se o outro azul é v ans = max(ans, w + sol(v,1,0,0,u) + sol(u,1,f,id+1,ant)); ans = max(ans, w + sol(v,1,f,0,u) + sol(u,1,0,id+1,ant)); //só passa para o próximo podendo ligar para ele com azul //(e o outro azul que vai estar ocupado) ou red ans = max(ans, max(sol(v,1,0,0,u),w+sol(v,3,0,0,u)) + sol(u,3,f,id+1,ant)); ans = max(ans, max(sol(v,1,f,0,u),w+sol(v,3,f,0,u)) + sol(u,3,0,id+1,ant)); // cout << u << " ja ligou azul p = 3 || f = " << f << " atual no " << v << " id " << id << " = " << " = " << ans << endl; return dp[u][p][f][id] = ans; } else if(p == 1) { //ele ja fez a ligacao entao só pode passar pro próximo //o v pode ligar para ele com red ou azul if(id == g[u].size()) { if(f == 1) return dp[u][p][f][id] = -inf; else return dp[u][p][f][id] = 0; } int v = g[u][id].fr; int w = g[u][id].sc; if(v == ant) return dp[u][p][f][id] = sol(u,p,f,id+1,ant); int ans = -inf; //só passa para o próximo podendo ligar para ele com azul //(e o outro azul que vai estar ocupado) ou red ans = max(ans, max(sol(v,1,0,0,u),w+sol(v,3,0,0,u)) + sol(u,1,f,id+1,ant)); ans = max(ans, max(sol(v,1,f,0,u),w+sol(v,3,f,0,u)) + sol(u,1,0,id+1,ant)); // cout << u << " ja ligou para todos || f = " << f << " atual no " << v << " = " << ans << endl; return dp[u][p][f][id] = ans; } else if(p == 2) { //ele nao ligou para ninguem ainda e se f = 1 pode continuear nao ligando if(id == g[u].size()) { if(f == 1) return dp[u][p][f][id] = 0; //ele fez a opcao de nao ligar para niguem else return dp[u][p][f][id] = -inf; //ele nao pode nao ligar para ninguem e mesmo assim fez } int v = g[u][id].fr; int w = g[u][id].sc; if(v == ant) return dp[u][p][f][id] = sol(u,p,f,id+1,ant); int ans = -inf; //só passa para o próximo podendo ligar para ele com azul //(e o outro azul que vai estar ocupado) ou red ans = max(ans, max(sol(v,1,0,0,u),w+sol(v,3,0,0,u)) + sol(u,2,f,id+1,ant)); ans = max(ans, max(sol(v,1,f,0,u),w+sol(v,3,f,0,u)) + sol(u,2,0,id+1,ant)); //ele vai fazer o papel da aresta red que é quebrada pelo v ans = max(ans, w+sol(v,0,0,0,u) + sol(u,1,f,id+1,ant)); ans = max(ans, w+sol(v,0,f,0,u) + sol(u,1,0,id+1,ant)); //liga para o proximo com um uma aresta red ans = max(ans, sol(v,2,0,0,u) + sol(u,1,f,id+1,ant)); ans = max(ans, sol(v,2,f,0,u) + sol(u,1,0,id+1,ant)); //liga para o proximo com uma aresta azul e v que vai fazer a aresta red ans = max(ans, w+sol(v,1,0,0,u) + sol(u,0,f,id+1,ant)); ans = max(ans, w+sol(v,1,f,0,u) + sol(u,0,0,id+1,ant)); //liga para o proximo com uma aresta azul e o outro que vai fazer a aresta red ans = max(ans, w+sol(v,2,0,0,u) + sol(u,3,f,id+1,ant)); ans = max(ans, w+sol(v,2,f,0,u) + sol(u,3,0,id+1,ant)); // cout << u << " nao liga pra ninguem e || f = " << f << " atual no " << v << " = " << ans << endl; return dp[u][p][f][id] = ans; } return 0; } void solve() { cin >> n; for(int i = 1; i < n; i++) { cin >> a[i] >> b[i] >> c[i]; g[a[i]].pb(mp(b[i],c[i])); g[b[i]].pb(mp(a[i],c[i])); } int rt; for(int i = 1; i <= n; i++) { if(g[i].size() == 1) { rt = i; // break; } } for(int i = 1; i <= n; i++) { for(int j = 0; j <= g[i].size()+5; j++) { for(int k = 0; k <= 3; k++) { dp[i][k][0].pb(0); dp[i][k][1].pb(0); mark[i][k][0].pb(0); mark[i][k][1].pb(0); } } } cout << sol(rt,2,1,0,0) << endl; } int32_t main() { ios::sync_with_stdio(false); cin.tie(0); // freopen("in.in", "r", stdin); // freopen("out.out", "w", stdout); int tt = 1; // cin >> tt; while(tt--) solve(); }

컴파일 시 표준 에러 (stderr) 메시지

beads.cpp: In function 'long long int sol(long long int, long long int, long long int, long long int, long long int)':
beads.cpp:43:15: warning: comparison of integer expressions of different signedness: 'long long int' and 'std::vector<std::pair<long long int, long long int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
   43 |         if(id == g[u].size()) return dp[u][p][f][id] = -inf;
      |            ~~~^~~~~~~~~~~~~~
beads.cpp:66:15: warning: comparison of integer expressions of different signedness: 'long long int' and 'std::vector<std::pair<long long int, long long int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
   66 |         if(id == g[u].size()) return dp[u][p][f][id] = -inf;
      |            ~~~^~~~~~~~~~~~~~
beads.cpp:89:15: warning: comparison of integer expressions of different signedness: 'long long int' and 'std::vector<std::pair<long long int, long long int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
   89 |         if(id == g[u].size()) {
      |            ~~~^~~~~~~~~~~~~~
beads.cpp:110:15: warning: comparison of integer expressions of different signedness: 'long long int' and 'std::vector<std::pair<long long int, long long int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  110 |         if(id == g[u].size()) {
      |            ~~~^~~~~~~~~~~~~~
beads.cpp: In function 'void solve()':
beads.cpp:165:26: warning: comparison of integer expressions of different signedness: 'long long int' and 'std::vector<std::pair<long long int, long long int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  165 |         for(int j = 0; j <= g[i].size()+5; j++) {
      |                        ~~^~~~~~~~~~~~~~~~
beads.cpp:176:27: warning: 'rt' may be used uninitialized in this function [-Wmaybe-uninitialized]
  176 |     cout << sol(rt,2,1,0,0) << endl;
      |                           ^
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...