結果
問題 | No.1548 [Cherry 2nd Tune B] 貴方と私とサイクルとモーメント |
ユーザー | tokusakurai |
提出日時 | 2021-06-11 23:33:09 |
言語 | C++17 (gcc 12.3.0 + boost 1.83.0) |
結果 |
AC
|
実行時間 | 550 ms / 4,500 ms |
コード長 | 8,655 bytes |
コンパイル時間 | 2,717 ms |
コンパイル使用メモリ | 225,940 KB |
実行使用メモリ | 129,648 KB |
最終ジャッジ日時 | 2024-05-08 20:00:51 |
合計ジャッジ時間 | 14,752 ms |
ジャッジサーバーID (参考情報) |
judge4 / judge3 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 2 ms
6,812 KB |
testcase_01 | AC | 2 ms
6,940 KB |
testcase_02 | AC | 293 ms
65,880 KB |
testcase_03 | AC | 87 ms
35,216 KB |
testcase_04 | AC | 261 ms
126,404 KB |
testcase_05 | AC | 24 ms
10,968 KB |
testcase_06 | AC | 209 ms
63,912 KB |
testcase_07 | AC | 219 ms
129,164 KB |
testcase_08 | AC | 283 ms
66,368 KB |
testcase_09 | AC | 198 ms
35,920 KB |
testcase_10 | AC | 188 ms
125,240 KB |
testcase_11 | AC | 221 ms
126,132 KB |
testcase_12 | AC | 129 ms
19,168 KB |
testcase_13 | AC | 132 ms
125,688 KB |
testcase_14 | AC | 202 ms
34,024 KB |
testcase_15 | AC | 127 ms
129,304 KB |
testcase_16 | AC | 203 ms
67,588 KB |
testcase_17 | AC | 291 ms
124,780 KB |
testcase_18 | AC | 216 ms
128,264 KB |
testcase_19 | AC | 106 ms
34,428 KB |
testcase_20 | AC | 164 ms
126,720 KB |
testcase_21 | AC | 186 ms
64,284 KB |
testcase_22 | AC | 330 ms
129,340 KB |
testcase_23 | AC | 350 ms
129,300 KB |
testcase_24 | AC | 356 ms
129,500 KB |
testcase_25 | AC | 325 ms
129,632 KB |
testcase_26 | AC | 353 ms
129,480 KB |
testcase_27 | AC | 350 ms
129,372 KB |
testcase_28 | AC | 333 ms
129,560 KB |
testcase_29 | AC | 358 ms
129,384 KB |
testcase_30 | AC | 355 ms
129,488 KB |
testcase_31 | AC | 354 ms
129,380 KB |
testcase_32 | AC | 211 ms
129,444 KB |
testcase_33 | AC | 216 ms
129,344 KB |
testcase_34 | AC | 216 ms
129,424 KB |
testcase_35 | AC | 214 ms
129,648 KB |
testcase_36 | AC | 193 ms
129,484 KB |
testcase_37 | AC | 550 ms
129,364 KB |
testcase_38 | AC | 189 ms
129,416 KB |
testcase_39 | AC | 190 ms
129,528 KB |
testcase_40 | AC | 176 ms
129,580 KB |
testcase_41 | AC | 191 ms
129,416 KB |
ソースコード
#include <bits/stdc++.h> using namespace std; #define rep(i, n) for(int i = 0; i < n; i++) #define rep2(i, x, n) for(int i = x; i <= n; i++) #define rep3(i, x, n) for(int i = x; i >= n; i--) #define each(e, v) for(auto &e: v) #define pb push_back #define eb emplace_back #define all(x) x.begin(), x.end() #define rall(x) x.rbegin(), x.rend() #define sz(x) (int)x.size() using ll = long long; using pii = pair<int, int>; using pil = pair<int, ll>; using pli = pair<ll, int>; using pll = pair<ll, ll>; //const int MOD = 1000000007; const int MOD = 998244353; const int inf = (1<<30)-1; const ll INF = (1LL<<60)-1; template<typename T> bool chmax(T &x, const T &y) {return (x < y)? (x = y, true) : false;}; template<typename T> bool chmin(T &x, const T &y) {return (x > y)? (x = y, true) : false;}; struct io_setup{ io_setup(){ ios_base::sync_with_stdio(false); cin.tie(NULL); cout << fixed << setprecision(15); } } io_setup; template<int mod> struct Mod_Int{ int x; Mod_Int() : x(0) {} Mod_Int(long long y) : x(y >= 0 ? y % mod : (mod - (-y) % mod) % mod) {} Mod_Int &operator += (const Mod_Int &p){ if((x += p.x) >= mod) x -= mod; return *this; } Mod_Int &operator -= (const Mod_Int &p){ if((x += mod - p.x) >= mod) x -= mod; return *this; } Mod_Int &operator *= (const Mod_Int &p){ x = (int) (1LL * x * p.x % mod); return *this; } Mod_Int &operator /= (const Mod_Int &p){ *this *= p.inverse(); return *this; } Mod_Int &operator ++ () {return *this += Mod_Int(1);} Mod_Int operator ++ (int){ Mod_Int tmp = *this; ++*this; return tmp; } Mod_Int &operator -- () {return *this -= Mod_Int(1);} Mod_Int operator -- (int){ Mod_Int tmp = *this; --*this; return tmp; } Mod_Int operator - () const {return Mod_Int(-x);} Mod_Int operator + (const Mod_Int &p) const {return Mod_Int(*this) += p;} Mod_Int operator - (const Mod_Int &p) const {return Mod_Int(*this) -= p;} Mod_Int operator * (const Mod_Int &p) const {return Mod_Int(*this) *= p;} Mod_Int operator / (const Mod_Int &p) const {return Mod_Int(*this) /= p;} bool operator == (const Mod_Int &p) const {return x == p.x;} bool operator != (const Mod_Int &p) const {return x != p.x;} Mod_Int inverse() const{ assert(*this != Mod_Int(0)); return pow(mod-2); } Mod_Int pow(long long k) const{ Mod_Int now = *this, ret = 1; for(; k > 0; k >>= 1, now *= now){ if(k&1) ret *= now; } return ret; } friend ostream &operator << (ostream &os, const Mod_Int &p){ return os << p.x; } friend istream &operator >> (istream &is, Mod_Int &p){ long long a; is >> a; p = Mod_Int<mod>(a); return is; } }; using mint = Mod_Int<MOD>; template<typename Monoid, typename Operator_Monoid> struct Lazy_Segment_Tree{ using F = function<Monoid(Monoid, Monoid)>; using G = function<Monoid(Monoid, Operator_Monoid)>; using H = function<Operator_Monoid(Operator_Monoid, Operator_Monoid)>; int n, height; vector<Monoid> seg; vector<Operator_Monoid> lazy; const F f; const G g; const H h; const Monoid e1; const Operator_Monoid e2; Lazy_Segment_Tree(const vector<Monoid> &v, const F &f, const G &g, const H &h, const Monoid &e1, const Operator_Monoid &e2) : f(f), g(g), h(h), e1(e1), e2(e2){ int m = v.size(); n = 1, height = 0; while(n < m) n <<= 1, height++; seg.assign(2*n, e1), lazy.assign(2*n, e2); copy(begin(v), end(v), seg.begin()+n); for(int i = n-1; i > 0; i--) seg[i] = f(seg[2*i], seg[2*i+1]); } inline Monoid reflect(int i) const{ return (lazy[i] == e2? seg[i] : g(seg[i], lazy[i])); } inline void recalc(int i){ while(i >>= 1) seg[i] = f(reflect(2*i), reflect(2*i+1)); } inline void eval(int i){ if(i < n && lazy[i] != e2){ lazy[2*i] = h(lazy[2*i] ,lazy[i]); lazy[2*i+1] = h(lazy[2*i+1], lazy[i]); seg[i] = reflect(i), lazy[i] = e2; } } inline void thrust(int i){ for(int j = height; j > 0; j--) eval(i>>j); } void apply(int l, int r, const Operator_Monoid &x){ l = max(l, 0), r = min(r, n); if(l >= r) return; l += n, r += n; thrust(l), thrust(r-1); int a = l, b = r; while(l < r){ if(l&1) lazy[l] = h(lazy[l], x), l++; if(r&1) r--, lazy[r] = h(lazy[r], x); l >>= 1, r >>= 1; } recalc(a), recalc(b-1); } Monoid query(int l, int r){ l = max(l, 0), r = min(r, n); if(l >= r) return e1; l += n, r += n; thrust(l), thrust(r-1); Monoid L = e1, R = e1; while(l < r){ if(l&1) L = f(L, reflect(l++)); if(r&1) R = f(reflect(--r), R); l >>= 1, r >>= 1; } return f(L, R); } Monoid operator [] (int i) {return query(i, i+1);} template<typename C> int find_subtree(int i, const C &check, const Monoid &x, Monoid &M, bool type){ while(i < n){ eval(i); Monoid nxt = type? f(reflect(2*i+type), M) : f(M, reflect(2*i+type)); if(check(nxt, x)) i = 2*i+type; else M = nxt, i = 2*i+(type^1); } return i-n; } template<typename C> int find_first(int l, const C &check, const Monoid &x){ Monoid L = e1; int a = l+n, b = n+n; thrust(a); while(a < b){ if(a&1){ Monoid nxt = f(L, reflect(a)); if(check(nxt, x)) return find_subtree(a, check, x, L, false); L = nxt, a++; } a >>= 1, b >>= 1; } return n; } template<typename C> int find_last(int r, const C &check, const Monoid &x){ Monoid R = e1; int a = n, b = r+n; thrust(b-1); while(a < b){ if(b&1 || a == 1){ Monoid nxt = f(reflect(--b), R); if(check(nxt, x)) return find_subtree(b, check, x, R, true); R = nxt; } a >>= 1, b >>= 1; } return -1; } }; int main(){ int N; cin >> N; vector<vector<pll>> a(5, vector<pll>(N, pll(1, 1))); rep(i, N){ ll x; cin >> x; rep(j, 4){ a[j+1][i].first = (a[j][i].first*x)%MOD; } } auto f = [](pll a, pll b) {return pll((a.first+b.first)%MOD, a.second+b.second);}; auto g = [](pll a, ll b) {return (b == -1? a : pll(b*a.second%MOD, a.second));}; auto h = [](ll a, ll b) {return (b == -1? a : b);}; vector<Lazy_Segment_Tree<pll, ll>> bit; rep(i, 5) bit.eb(a[i], f, g, h, pll(0, 0), -1); int Q; cin >> Q; while(Q--){ int t, u, v, w; cin >> t >> u >> v >> w; if(u > v) swap(u, v); bool flag = !(u < w && w < v); if(flag) v--; else u--; //cout << u << ' ' << v << ' ' << flag << '\n'; //cout << bit[0].query(0, N).first << '\n'; vector<mint> b(5, 0); rep(i, 5){ b[i] = bit[i].query(u, v).first; if(flag) b[i] = -b[i]+bit[i].query(0, N).first; } mint L = b[0].inverse(); mint M = b[1]*L; //rep(i, 5) cout << b[i] << ' '; cout << '\n'; if(t == 0){ vector<ll> c(5, 1); ll x; cin >> x; rep(i, 4){ c[i+1] = (c[i]*x)%MOD; } if(!flag){ rep(i, 5){ bit[i].apply(u, v, c[i]); } } else{ rep(i, 5){ bit[i].apply(0, u, c[i]); bit[i].apply(v, N, c[i]); } } continue; } mint ans = 0; if(t == 1){ ans -= M*b[0]; ans += b[1]; } if(t == 2){ ans += M*M*b[0]; ans -= M*2*b[1]; ans += b[2]; } if(t == 3){ ans -= M*M*M*b[0]; ans += M*M*3*b[1]; ans -= M*3*b[2]; ans += b[3]; } if(t == 4){ ans += M*M*M*M*b[0]; ans -= M*M*M*4*b[1]; ans += M*M*6*b[2]; ans -= M*4*b[3]; ans += b[4]; } cout << ans*L << '\n'; } }