제출 #652979

#제출 시각아이디문제언어결과실행 시간메모리
652979vladutpieleBigger segments (IZhO19_segments)C++17
0 / 100
1 ms212 KiB
#include <bits/stdc++.h> #define int long long using namespace std; const int nmax = 500000; int n; int v[nmax + 5], sume[nmax + 5]; struct elem { int maxSeg; /// in cate segmente impart prefixul [1 ... i] int minSum; /// suma minima pentru ultimul segment int maxPrv; /// cel mai mare capat dreapta al segmentului dp[i].maxSeg - 1 ///daca capatul stanga al ultimului segment e j, maxPrv este j - 1 }; elem dp[nmax + 5]; signed main() { cin >> n; for(int i = 1; i <= n; i ++) { cin >> v[i]; sume[i] = sume[i - 1] + v[i]; } /// initializare for(int i = 1; i <= n; i ++) { dp[i] = {1, sume[i], 1}; } /// cum pot sa calculez cel mai usor dp[i].maxPrv? /// segmentul [a .... b] este un segment care imbunatateste solutia /// daca sume[b] - sume[a] >= sume[a] - sume[dp[a].maxPrv] /// relatie echivalenta cu : sume[b] >= 2 * sume[a] - sume[dp[a].maxPrv] /// astfel pot sa caut binar valoarea a (cel mai mare a) for(int i = 2; i <= n; i ++) { if(dp[i - 1].maxSeg > dp[i].maxSeg) { dp[i] = dp[i - 1]; dp[i].minSum = dp[i - 1].minSum + v[i]; } int st = 1, dr = i; int maxPrv = 0; while(st <= dr) { int mid = (st + dr) >> 1; if(sume[i] >= 2 * sume[mid] - sume[dp[mid].maxPrv - 1]) { maxPrv = mid; st = mid + 1; } else { dr = mid - 1; } } dp[i].maxSeg = 1 + dp[maxPrv].maxSeg; dp[i].minSum = sume[i] - sume[maxPrv]; dp[i].maxPrv = maxPrv; } cout << dp[n].maxSeg << '\n'; return 0; }
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...