結果

問題 No.801 エレベーター
ユーザー kappybarkappybar
提出日時 2020-05-01 12:16:13
言語 C++14
(gcc 12.3.0 + boost 1.83.0)
結果
TLE  
実行時間 -
コード長 2,638 bytes
コンパイル時間 1,994 ms
コンパイル使用メモリ 180,796 KB
実行使用メモリ 13,756 KB
最終ジャッジ日時 2024-06-06 05:55:29
合計ジャッジ時間 6,314 ms
ジャッジサーバーID
(参考情報)
judge4 / judge5
このコードへのチャレンジ
(要ログイン)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 AC 2 ms
13,756 KB
testcase_01 AC 2 ms
6,944 KB
testcase_02 AC 2 ms
6,940 KB
testcase_03 AC 52 ms
6,944 KB
testcase_04 AC 52 ms
6,940 KB
testcase_05 AC 52 ms
6,940 KB
testcase_06 AC 52 ms
6,944 KB
testcase_07 AC 53 ms
6,940 KB
testcase_08 AC 55 ms
6,940 KB
testcase_09 AC 52 ms
6,940 KB
testcase_10 AC 52 ms
6,944 KB
testcase_11 AC 52 ms
6,940 KB
testcase_12 AC 53 ms
6,944 KB
testcase_13 TLE -
testcase_14 -- -
testcase_15 -- -
testcase_16 -- -
testcase_17 -- -
testcase_18 -- -
testcase_19 -- -
testcase_20 -- -
testcase_21 -- -
testcase_22 -- -
testcase_23 -- -
testcase_24 -- -
testcase_25 -- -
testcase_26 -- -
testcase_27 -- -
testcase_28 -- -
権限があれば一括ダウンロードができます

ソースコード

diff #

#include <bits/stdc++.h>
#define rep(i,n) for(int i=0;i<n;i++)
using namespace std;
using ll =  long long ;
using P = pair<int,int> ;
using pll = pair<ll,ll>;
const int INF = 1e9;
const ll LINF = 1e18;
const int MOD = 1000000007;


struct LazySegmentTree{
        int n;
        vector<long long> node;
        vector<long long> lazy;
        ll inv2 = MOD -  (MOD / 2) % MOD;
        
        LazySegmentTree(vector<long long> v){
                int sz = v.size();
                n = 1;
                while(n < sz) n *= 2;
                node.resize(2*n-1,0);
                lazy.resize(2*n-1,0);
                for(int i=0;i<sz;i++) node[i+n-1] = v[i];
                for(int i=n-2;i>=0;i--) node[i] = (node[2*i+1] + node[2*i+2])%MOD;
        }

        void eval(int k){
            if(lazy[k] == 0) return;
            if(k < n - 1){
                lazy[2*k+1] = (lazy[2*k+1] + (lazy[k] * inv2)%MOD)%MOD;
                lazy[2*k+2] = (lazy[2*k+2] + (lazy[k] * inv2)%MOD)%MOD;
            }
            node[k] = (node[k] + lazy[k])%MOD;
            lazy[k] = 0;
        }
        
        //[a,b) の区間全てにxを足す。
        void update(int a,int b,long long x,int k=0,int l=0,int r=-1){
            eval(k);
            if(r < 0) r = n;
            if(a <= l && r <= b){
                lazy[k] = (lazy[k] + ((r - l) * x)%MOD)%MOD;
                eval(k);
            }else if(a < r && l < b){
                update(a,b,x,2*k+1,l,(l+r)/2);
                update(a,b,x,2*k+2,(l+r)/2,r);
                node[k] = (node[2*k+1] + node[2*k+2])%MOD;
            }
        }
        
        //[a,b) の区間の総和を求める。
        long long query(int a,int b,int k=0,int l=0,int r=-1){
            eval(k);
            if(r < 0) r = n;
            if(r <= a || b <= l) return 0;
            if(a <= l && r <= b) return node[k];
            long long vl = query(a,b,2*k+1,l,(l+r)/2);
            long long vr = query(a,b,2*k+2,(l+r)/2,r);
            return (vl + vr)%MOD;
        }
};

int main(){
    int n,m,k;
    cin >> n >> m >> k;
    vector<P> lr(m);
    rep(i,m){
        cin >> lr[i].first >> lr[i].second;
        --lr[i].first; --lr[i].second;
    }
    vector<ll> a(n,0);
    a[0] = 1;
    LazySegmentTree seg_prev(a);
    LazySegmentTree seg(vector<ll>(n,0));
    rep(i,k){
        rep(j,m){
            ll plus = seg_prev.query(lr[j].first,lr[j].second+1);
            seg.update(lr[j].first,lr[j].second+1,plus);
        }
        swap(seg,seg_prev);
        LazySegmentTree empty(vector<ll>(n,0));
        swap(empty,seg);
    }
    cout << seg_prev.query(n-1,n) << endl;
    return 0;
}
0