QOJ.ac

QOJ

ID题目提交者结果用时内存语言文件大小提交时间测评时间
#390153#6842. Popcount Words_LiMLE_TL 0ms30292kbC++145.3kb2024-04-15 08:54:572024-04-15 08:54:58

Judging History

你现在查看的是最新测评结果

  • [2024-04-15 08:54:58]
  • 评测
  • 测评结果:TL
  • 用时:0ms
  • 内存:30292kb
  • [2024-04-15 08:54:57]
  • 提交

answer

#include<bits/stdc++.h>
#define int long long
#define IL inline
#define FOR(i, s, t) for(int i = (s); i <= (t); ++i)
#define ROF(i, s, t) for(int i = (s); i >= (t); --i)
#define pb push_back
#define fi first
#define se second
#define pii pair<int, int>
#define mkp make_pair

using namespace std;

const int N = 5e5 + 5;

int n, Q;
int L[N], R[N];
int F[N * 20], zt[N * 20], ct = 0, ids[N];
vector<pii> G, V;

IL void pre(int l, int r, int k, int op) {
    V.clear();
    while(l <= r) {
        while(k > 0 && (r - (1 << k) + 1) < l) k--, op ^= 1;
        int pos = r - (1 << k) + 1;
        V.pb(mkp(k, op)); r = pos - 1; op ^= 1;
    }
    reverse(V.begin(), V.end());
    return;
}

IL void suf(int l, int r, int k, int op) {
    V.clear(); 
    while(l <= r) {
        while(k > 0 && (l + (1 << k) - 1) > r) k--;
        int pos = l + (1 << k) - 1;
        V.pb(mkp(k, op)); l = pos + 1; op ^= 1;
    }
    return;
}

IL void ad() {
    for(pii v : V) G.pb(v); V.clear();
    return;
}

struct ACAM {
    int Trie[N][2], fail[N], tot = 0;
    int f[N][31][2], cc[N][31][2], cnt[N];
    vector<pii> V[N]; 

    IL int insert(string S) {
        int len = S.length(), cur = 0;
        FOR(i, 1, len) {
            int it = (S[i - 1] - '0');
            if(!Trie[cur][it]) Trie[cur][it] = ++tot, V[cur].pb(mkp(Trie[cur][it], it));
            cur = Trie[cur][it];
        }
        return cur;
    }

    IL void build() {
        queue<int> q;
        FOR(i, 0, 1) if(Trie[0][i]) q.push(Trie[0][i]);
        while(!q.empty()) {
            int x = q.front(); q.pop();
            FOR(i, 0, 1) {
                if(!Trie[x][i]) Trie[x][i] = Trie[fail[x]][i];
                else fail[Trie[x][i]] = Trie[fail[x]][i], q.push(Trie[x][i]);
            }
        }
        return;
    }

    IL void buildf() {
        FOR(i, 0, tot) f[i][0][0] = Trie[i][1], f[i][0][1] = Trie[i][0]; 
        FOR(k, 1, 29) {
            FOR(i, 0, tot) {
                f[i][k][0] = f[f[i][k - 1][0]][k - 1][1];
                f[i][k][1] = f[f[i][k - 1][1]][k - 1][0];
            }   
        }
        return;
    }

    IL void print(int x) {
        cout << x << ": ";
        for(pii p : V[x]) cout << p.fi << ' ' << p.se << '\n';
        cout << "fail: " << fail[x] << '\n';
        for(pii p : V[x]) print(p.fi);
        return;
    }

    IL void getres() {
        int cur = 0;
        FOR(i, 1, ct) {
            int t1 = F[i], t2 = zt[i];
            cc[cur][t1][t2]++; cur = f[cur][t1][t2];
            // cout << t1 << ' ' << cur << '\n';
        }
        cnt[cur]++;
        ROF(k, 4, 1) {
            FOR(i, 0, tot) {
                FOR(zt, 0, 1) {
                    // if(i == 7 && k == 1 && zt == 0) cout << cc[i][k][zt] << ' ' << f[i][k - 1][zt] << " ?!!\n";
                    // if(f[i][k - 1][zt] == 6) cout << i << ' ' << k << ' ' << zt << ' ' << cc[i][k][zt] << '\n';
                    cc[i][k - 1][zt] += cc[i][k][zt];
                    cc[f[i][k - 1][zt]][k - 1][zt ^ 1] += cc[i][k][zt];
                }
            }
        }
        // cout << cc[6][0][0] << ' ' << cc[6][0][1] << '\n';
        FOR(i, 0, tot) cnt[i] += cc[i][0][0] + cc[i][0][1];
        // FOR(i, 1, tot) cout << cnt[i] << ' ';
        // cout << '\n';
        ROF(i, tot, 1) cnt[fail[i]] += cnt[i];
        return;
    }
}A;

signed main() {
    // freopen("test.in", "r", stdin);
    // freopen("test.out", "w", stdout);
    ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
    cin >> n >> Q;
    FOR(i, 1, n) {
        cin >> L[i] >> R[i];
        int k = 0, op = 0;
        while(true) {
            int l = (1 << k), r = (1 << (k + 1)) - 1;
            if(l < L[i] && r > R[i]) {
                while(true) {
                    k--; int mid = l + (1 << k) - 1;
                    if(R[i] < mid) r = mid;
                    else if(L[i] > mid + 1) l = mid + 1, op ^= 1;
                    else {
                        pre(L[i], mid, k, op); ad();
                        suf(mid + 1, R[i], k, op ^ 1); ad();
                    } 
                }
                break;
            }
            if(r < L[i]) k++;
            else if(l >= L[i] && l <= R[i] && r > R[i]) {
                pre(L[i], l - 1, k - 1, op); ad();
                suf(l, R[i], k, op); ad();
                break;
            }
            else if(l < L[i] && r >= L[i] && r <= R[i]) {
                pre(L[i], r, k, op); ad();
                suf(r + 1, R[i], k + 1, op); ad();
                break;
            }
            else {
                pre(L[i], l - 1, k - 1, op); ad();
                while(r <= R[i]) {
                    G.pb(mkp(k, 0)); k++;
                    if((1 << (k + 1)) - 1 > R[i]) {
                        suf(r + 1, R[i], k, op); ad();
                        break;
                    }
                    l = (1 << k), r = (1 << (k + 1)) - 1;
                } 
                break;
            }
        }
    }
    for(pii p : G) F[++ct] = p.fi, zt[ct] = p.se;
    FOR(i, 1, Q) {
        string S; cin >> S;
        ids[i] = A.insert(S);
    }
    A.build(); A.buildf(); A.getres();
    FOR(i, 1, Q) cout << A.cnt[ids[i]] << '\n';
    return 0;
}

/*
    Author: _LiMLE_
    start coding at: 20:17
    finish debugging at: 
*/

详细

Test #1:

score: 100
Accepted
time: 0ms
memory: 30220kb

input:

3 5
2 6
1 3
4 8
0
1
11
101
0011010

output:

6
7
2
2
1

result:

ok 5 lines

Test #2:

score: 0
Accepted
time: 0ms
memory: 30292kb

input:

3 10
2 6
1 3
4 8
0
1
11
101
0011010
0
1
11
101
0011010

output:

6
7
2
2
1
6
7
2
2
1

result:

ok 10 lines

Test #3:

score: -100
Time Limit Exceeded

input:

100000 37701
605224571 605224571
681034454 681034463
468837041 468837048
323235128 323235135
400367345 400367345
394938241 394938241
221026001 221026007
183872918 183872926
881878131 881878138
374491962 374491967
588030380 588030381
109143398 109143401
727016146 727016149
857189138 857189138
1940312...

output:


result: