Submission #467947

#TimeUsernameProblemLanguageResultExecution timeMemory
467947VladithurSafety (NOI18_safety)C++17
100 / 100
554 ms23508 KiB
//#pragma GCC optimize ("Ofast") //#pragma GCC target("avx2") //#pragma GCC target ("sse4") //#pragma GCC optimize("unroll-loops") //#include <x86intrin.h> //#include <ext/pb_ds/assoc_container.hpp> //#include <ext/pb_ds/tree_policy.hpp> #include <bits/stdc++.h> #define PI 3.141592653589793L #define FAST ios::sync_with_stdio(false); cin.tie(NULL); // Use for file I/O; #define FIN string _fname = "input"; \ string _is = _fname + ".in", _os = _fname + ".out"; \ freopen(_is.c_str(), "r", stdin); \ freopen(_os.c_str(), "w", stdout); #define FIN2 freopen("input.txt", "r", stdin); freopen("output.txt", "w", stdout); #define RINIT mt19937 rng(chrono::steady_clock::now().time_since_epoch().count()); #define all(x) (x).begin(), (x).end() #define allr(x) (x).rbegin(), (x).rend() #define gsize(x) (int)((x).size()) using namespace std; //using namespace __gnu_pbds; struct chash { static inline uint64_t splitmix64(uint64_t x) { // http://xorshift.di.unimi.it/splitmix64.c x += 0x9e3779b97f4a7c15; x = (x ^ (x >> 30)) * 0xbf58476d1ce4e5b9; x = (x ^ (x >> 27)) * 0x94d049bb133111eb; return x ^ (x >> 31); } inline size_t operator()(uint64_t x) const { static const uint64_t FIXED_RANDOM = chrono::steady_clock::now().time_since_epoch().count(); return splitmix64(x + FIXED_RANDOM); } inline size_t operator()(const pair<uint64_t, uint64_t> &v) const { chash int_hasher; return int_hasher(v.first) * 31 + int_hasher(v.second); } }; //template <class K> //using hash_set = gp_hash_table<K, null_type, chash>; const char nl = '\n'; typedef long long ll; typedef long double ld; // typedef unsigned int uint; typedef unsigned long long ull; const ll inf = 1e9 + 10; const ll inf2 = 1e18 + 99LL; const ld inf3 = 1e14; const ll mod = 1e9+7, mod2 = 998244353; const ld eps = 1e-9; const bool local = false; //RINIT; const int logn = 21, maxn = 200001, maxm = 1, maxn2 = 3; struct node { ll v, prop; int size, pr; node *l, *r; node (int x, node * tl = nullptr, node * tr = nullptr) { size = 1; v = x; prop = 0; pr = rand(); l = tl; r = tr; } }; int size(node * t) { if (t) return t->size; return 0; } void recalc(node * t) { if (!t) return; t->size = 1 + size(t->l) + size(t->r); } void prop(node * t) { if (!t || !t->prop) return; t->v += t->prop; if (t->l) t->l->prop += t->prop; if (t->r) t->r->prop += t->prop; t->prop = 0; } // We want to split of the first 'n' ellements array<node *, 2> split(node * t, int n) { if (!t) return {nullptr, nullptr}; prop(t); if (size(t->l) >= n) { auto res = split(t->l, n); t->l = res[1]; recalc(t); return {res[0], t}; } else { auto res = split(t->r, n - size(t->l) - 1); t->r = res[0]; recalc(t); return {t, res[1]}; } } // Merge two treaps node * merge(node *a, node *b) { if (!a) return b; if (!b) return a; prop(a); prop(b); if (a->pr < b->pr) { a->r = merge(a->r, b); recalc(a); return a; } else { b->l = merge(a, b->l); recalc(b); return b; } } // Add 'x' on [l; r) node *update(node * t, int l, int r, int x) { if (l >= r) return t; auto a = split(t, l); auto b = split(a[1], r - l); //cout << "U: " << size(a[0]) << ' ' << size(b[0]) << ' ' << size(b[1]) << endl; b[0]->prop += x; return merge(a[0], merge(b[0], b[1])); } ll h; node * f(node * t, int ti) { auto a = split(t, ti + 1); a[0]->prop -= h; a[1]->prop += h; return merge(a[0], a[1]); //t = update(t, 0, ti + 1, -h); //t = update(t, ti + 1, i * 2, +h); } // Get first index with value > x int tres = 0; void find(node * t, ll x, int ti = 0) { if (!t) return; prop(t); if (t->v > x) { tres = min(tres, size(t->l) + ti); find(t->l, x, ti); } else { find(t->r, x, ti + size(t->l) + 1); } } void print(node * t) { if (!t) return; prop(t); print(t->l); cout << t->v << ' '; print(t->r); } vector<ll> ta; void toArray(node * t) { if (!t) return; prop(t); toArray(t->l); ta.push_back(t->v); toArray(t->r); delete t; } int main() { FAST; srand(0); int n; cin >> n >> h; __int128 k = -1, b; ll ttta; cin >> ttta; b = ttta; node * t = new node(b); t = merge(t, new node(b)); for (int i = 1; i < n; i++) { ll a; cin >> a; //print(t); //cout << nl; int ti = - 1 - k; //cout << 0 << ' ' << ti + 1 << ' ' << ti + 1 << ' ' << i * 2 << endl; t = f(t, ti); //print(t); //cout << nl; tres = inf; find(t, a); // cout << tres << nl; node * t2 = new node(a); t2 = merge(t2, new node(a)); if (tres == inf) { t = merge(t, t2); } else { auto c = split(t, tres); c[0] = merge(c[0], t2); t = merge(c[0], c[1]); } b += k * h; k--; b += a; //print(t); //cout << nl; //cout << "F: " << k << ' ' << b << nl << nl; } __int128 ans = inf2; toArray(t); for (ll x: ta) { ans = min(ans, x * k + b); k++; b -= x; } cout << (ll)(ans) << nl; } // © Benq /* stuff you should look for * int overflow, array bounds * special cases (n=1?) * do smth instead of nothing and stay organized * WRITE STUFF DOWN * DON'T GET STUCK ON ONE APPROACH */
#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...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...