QOJ.ac

QOJ

ID题目提交者结果用时内存语言文件大小提交时间测评时间
#935086#10226. Tree Flip中国入民小学附属信息I队 (Zicheng Wang, Shubang Liu, Minkai Shao)#AC ✓239ms21892kbC++148.1kb2025-03-15 12:32:072025-03-15 12:32:14

Judging History

This is the latest submission verdict.

  • [2025-03-15 12:32:14]
  • Judged
  • Verdict: AC
  • Time: 239ms
  • Memory: 21892kb
  • [2025-03-15 12:32:07]
  • Submitted

answer

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <numeric>
#include <cstring>
#include <queue>
#include <set>
#include <map>
#include <bitset>
#include <random>
#include <ctime>
#include <chrono>
#include <iomanip>
#include <array>
#include <cassert>

typedef long long ll;
typedef double lf;

// #define DEBUG 1
struct IO
{
    #define MAXSIZE (1 << 20)
    #define isdigit(x) (x >= '0' && x <= '9')
    char buf[MAXSIZE], *p1, *p2;
    char pbuf[MAXSIZE], *pp;
    #if DEBUG
    #else
    IO() : p1(buf), p2(buf), pp(pbuf) {}
    ~IO() {fwrite(pbuf, 1, pp - pbuf, stdout);}
    #endif
    #define gc() (p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, MAXSIZE, stdin), p1 == p2) ? ' ' : *p1++)
    #define blank(x) (x == ' ' || x == '\n' || x == '\r' || x == '\t')

    template <typename T>
    void Read(T &x)
    {
        #if DEBUG
        std::cin >> x;
        #else
        bool sign = 0; char ch = gc(); x = 0;
        for (; !isdigit(ch); ch = gc())
            if (ch == '-') sign = 1;
        for (; isdigit(ch); ch = gc()) x = x * 10 + (ch ^ 48);
        if (sign) x = -x;
        #endif
    }
    void Read(char *s)
    {
        #if DEBUG
        std::cin >> s;
        #else
        char ch = gc();
        for (; blank(ch); ch = gc());
        for (; !blank(ch); ch = gc()) *s++ = ch;
        *s = 0;
        #endif
    }
    void Read(char &c) {for (c = gc(); blank(c); c = gc());}

    void Push(const char &c)
    {
        #if DEBUG
        putchar(c);
        #else
        if (pp - pbuf == MAXSIZE) fwrite(pbuf, 1, MAXSIZE, stdout), pp = pbuf;
        *pp++ = c;
        #endif
    }
    template <typename T>
    void Write(T x)
    {
        if (x < 0) x = -x, Push('-');
        static T sta[35];
        int top = 0;
        do sta[top++] = x % 10, x /= 10; while (x);
        while (top) Push(sta[--top] ^ 48);
    }
    template <typename T>
    void Write(T x, char lst) {Write(x), Push(lst);}
} IO;
#define Read(x) IO.Read(x)
#define Write(x, y) IO.Write(x, y)
#define Put(x) IO.Push(x)

using namespace std;

const int MAXN = 1e5 + 10;

int n, q;
bool a[MAXN];
vector <int> e[MAXN];

int dep[MAXN], fa[MAXN], sz[MAXN], son[MAXN];
int dfn[MAXN], idx[MAXN], top[MAXN], clk, btm[MAXN];
void DFS1(int u, int t)
{
    dep[u] = dep[t] + 1, fa[u] = t, sz[u] = 1, son[u] = 0;
    for (auto v : e[u])
        if (v != t) DFS1(v, u), (sz[son[u]] < sz[v]) && (son[u] = v), sz[u] += sz[v];
}
void DFS2(int u, int t)
{
    dfn[u] = ++clk, idx[clk] = u, top[u] = t, btm[u] = u;
    if (son[u]) DFS2(son[u], t), btm[u] = btm[son[u]];
    for (auto v : e[u])
        if (v != fa[u] && v != son[u]) DFS2(v, v);
}

struct BIT
{
    bool bit[MAXN];
    inline void add(int x) { for (; x <= n; x += (x & -x)) bit[x] ^= 1; }
    inline void flip(int x) { add(dfn[x]), add(dfn[x] + sz[x]); }
    inline bool ask(int x) { bool res = 0; for (; x; x -= (x & -x)) res ^= bit[x]; return res; }
    inline bool ask(int u, int v) { if (dfn[v] < dfn[u]) return 0; return ask(dfn[v]) ^ ask(dfn[fa[u]]); }
}B;

typedef array <int, 2> arr;
struct SGT
{
    #define ls (p << 1)
    #define rs (p << 1 | 1)
    bool tag[MAXN << 2];
    arr tr[MAXN << 2];
    inline void Pushup(int p) { tr[p][0] = tr[ls][0] + tr[rs][0], tr[p][1] = tr[ls][1] + tr[rs][1]; }
    inline void Calc(int p) { swap(tr[p][0], tr[p][1]), tag[p] ^= 1; }
    inline void Pushdown(int p) { if (!tag[p]) return; Calc(ls), Calc(rs), tag[p] = 0; }
    void Build(int p, int l, int r)
    {
        tr[p][0] = tr[p][1] = tag[p] = 0;
        if (l == r) return;
        int mid = l + r >> 1;
        Build(ls, l, mid), Build(rs, mid + 1, r);
    }
    void Update(int p, int l, int r, int x, int y)
    {
        if (x > y) return;
        if (x <= l && r <= y) return Calc(p);
        Pushdown(p); int mid = l + r >> 1;
        if (x <= mid) Update(ls, l, mid, x, y);
        if (y > mid) Update(rs, mid + 1, r, x, y);
        Pushup(p);
    }
    void Modify(int p, int l, int r, int x, int k0, int k1)
    {
        if (l == r) return tr[p][0] += k0, tr[p][1] += k1, void();
        Pushdown(p); int mid = l + r >> 1;
        if (x <= mid) Modify(ls, l, mid, x, k0, k1);
        else Modify(rs, mid + 1, r, x, k0, k1);
        Pushup(p);
    }
    arr Query(int p, int l, int r, int x, int y)
    {
        if (x > y) return arr{0, 0};
        if (x <= l && r <= y) return tr[p];
        Pushdown(p); int mid = l + r >> 1; arr ret{0, 0}, cur;
        if (x <= mid) ret = Query(ls, l, mid, x, y);
        if (y > mid) cur = Query(rs, mid + 1, r, x, y), ret[0] += cur[0], ret[1] += cur[1];
        return ret;
    }
}T1, T2;

arr f[MAXN];

void Init(int u)
{
    bool c = B.ask(top[u], u);
    // cout << "In " << u << ' ' << c << '\n';
    T1.Modify(1, 1, n, dfn[u], c == 0, c == 1);
    T2.Modify(1, 1, n, dfn[u], (c ^ a[u]) == 0, (c ^ a[u]) == 1);
    for (auto v : e[u]) if (v != fa[u]) Init(v);
    if (top[u] == u && fa[u])
    {
        f[u] = T1.Query(1, 1, n, dfn[u], dfn[btm[u]]);
        c = B.ask(top[fa[u]], fa[u]);
        // cout << "# " << u << " " << f[u][0] << ' ' << f[u][1] << ' ' << c << '\n';
        T1.Modify(1, 1, n, dfn[fa[u]], f[u][c], f[u][!c]);
        c ^= a[fa[u]];
        T2.Modify(1, 1, n, dfn[fa[u]], f[u][c], f[u][!c]);
    }
    // for (int i = 1; i <= n; i++) cout << T1.Query(1, 1, n, dfn[i], dfn[i])[0] << ' '; cout << '\n';
}

void Update(int u)
{
    T1.Update(1, 1, n, dfn[u], dfn[btm[u]]), T2.Update(1, 1, n, dfn[u] + 1, dfn[btm[u]]);
    B.flip(u), a[u] ^= 1;
    while (fa[u = top[u]])
    {
        arr t = T1.Query(1, 1, n, dfn[u], dfn[btm[u]]);
        bool c = B.ask(top[fa[u]], fa[u]);
        T1.Modify(1, 1, n, dfn[fa[u]], t[c] - f[u][c], t[!c] - f[u][!c]);
        c ^= a[fa[u]];
        T2.Modify(1, 1, n, dfn[fa[u]], t[c] - f[u][c], t[!c] - f[u][!c]);
        f[u] = t, u = fa[u];
    }
}

void Solve(int u)
{
    int ans = 0;
    arr t = T1.Query(1, 1, n, dfn[u], dfn[btm[u]]); bool c = B.ask(top[u], u);
    ans += t[c ^ a[u] ^ 1];
    t = T2.Query(1, 1, n, dfn[top[u]], dfn[u] - 1);
    ans += t[c ^ 1];
    while (fa[u = top[u]])
    {
        ans -= f[u][c ^ a[fa[u]] ^ 1];
        c ^= B.ask(top[fa[u]], fa[u]);
        t = T1.Query(1, 1, n, dfn[fa[u]], dfn[btm[fa[u]]]);
        ans += t[c ^ a[fa[u]] ^ 1];
        t = T2.Query(1, 1, n, dfn[top[fa[u]]], dfn[fa[u]] - 1);
        ans += t[c ^ 1];
        u = fa[u];
    }
    cout << ans << '\n';
}

int main()
{
    // freopen(".in", "r", stdin);
    // freopen(".out", "w", stdout);
    #if DEBUG
    #else
    ios::sync_with_stdio(0), cin.tie(0);
    #endif
    int T;
    Read(T);
    while (T--)
    {
        Read(n), Read(q);
        for (int i = 1; i <= n; i++)
        {
            e[i].clear();
            B.bit[i] = 0;
            dep[i] = fa[i] = sz[i] = son[i] = 0;
            dfn[i] = idx[i] = top[i] = btm[i] = 0;
            f[i][0] = f[i][1] = 0;
        }
        clk = 0;
        for (int i = 1; i <= n; i++) Read(a[i]);
        for (int i = 1, u, v; i < n; i++) Read(u), Read(v), e[u].push_back(v), e[v].push_back(u);
        
        DFS1(1, 0), DFS2(1, 1);
        for (int i = 1; i <= n; i++) if (a[i]) B.flip(i);
        T1.Build(1, 1, n), T2.Build(1, 1, n);
        Init(1);
        
        // for (int i = 1; i <= n; i++) cout << dfn[i] << ' '; cout << "\n";
        // for (int i = 1; i <= n; i++) cout << T1.Query(1, 1, n, dfn[i], dfn[i])[0] << ' '; cout << '\n';
        // for (int i = 1; i <= n; i++) cout << T1.Query(1, 1, n, dfn[i], dfn[i])[1] << ' '; cout << '\n';
        // for (int i = 1; i <= n; i++) cout << T2.Query(1, 1, n, dfn[i], dfn[i])[0] << ' '; cout << '\n';
        // for (int i = 1; i <= n; i++) cout << T2.Query(1, 1, n, dfn[i], dfn[i])[1] << ' '; cout << '\n';

        int op, x, rt = 1;
        while (q--)
        {
            Read(op), Read(x);
            if (op == 1) Update(x);
            else rt = x;
            Solve(rt);
            // for (int i = 1; i <= n; i++) cout << T1.Query(1, 1, n, dfn[i], dfn[i])[0] << ' '; cout << '\n';
        }
    }
    return 0;
}

这程序好像有点Bug,我给组数据试试?

詳細信息

Test #1:

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

input:

1
3 3
0 0 1
1 2
3 1
1 1
2 2
1 1

output:

2
1
1

result:

ok 3 lines

Test #2:

score: 0
Accepted
time: 239ms
memory: 21892kb

input:

1
100000 100000
0 0 1 1 0 0 0 1 1 1 0 0 0 1 1 1 1 1 0 1 1 1 0 1 0 1 1 0 0 0 1 0 1 0 0 1 1 1 1 1 1 1 0 0 0 0 0 0 1 1 0 0 1 1 1 0 0 0 1 1 1 1 0 0 0 0 0 0 1 1 1 1 0 1 1 0 0 0 0 1 0 0 0 0 1 0 1 1 1 1 0 0 0 0 1 1 1 1 0 1 0 1 1 0 1 0 1 1 0 0 1 1 1 1 1 1 1 1 1 0 1 0 0 1 0 0 1 0 1 0 0 1 1 1 0 1 1 1 1 0 1 1 ...

output:

49874
50073
49944
49944
49917
49916
49984
49947
49945
49870
50089
50088
50024
50025
49980
49862
49984
49982
49983
49990
49949
49951
49950
50195
50196
50196
49955
50072
50071
49993
50021
50134
49985
49917
49886
49885
50134
49818
49819
49952
49954
49955
49986
50046
50018
50021
50020
50017
50130
50132
...

result:

ok 100000 lines

Test #3:

score: 0
Accepted
time: 134ms
memory: 18476kb

input:

10
9141 9858
1 1 1 0 0 1 1 1 0 0 1 1 0 1 1 1 0 1 0 0 1 0 0 0 1 1 0 1 1 0 0 1 1 0 0 1 0 1 1 1 0 1 1 0 1 1 1 0 0 0 1 1 1 0 1 0 1 1 1 0 0 1 1 1 0 1 1 1 1 0 1 0 1 0 0 0 1 0 1 0 0 1 1 0 0 0 0 1 0 1 0 1 1 0 0 0 0 1 1 1 1 1 0 1 0 0 0 0 1 1 1 0 1 0 1 1 1 0 1 1 1 1 1 1 1 0 1 0 0 1 0 0 1 0 1 0 0 1 0 1 1 0 1 1...

output:

4598
4606
4568
4553
4529
4529
4601
4600
4534
4546
4548
4621
4597
4515
4586
4586
4587
4536
4552
4553
4555
4556
4618
4580
4555
4561
4549
4547
4547
4517
4594
4593
4556
4556
4554
4554
4553
4557
4603
4603
4602
4602
4600
4516
4523
4593
4592
4589
4590
4591
4584
4543
4542
4532
4537
4562
4611
4558
4556
4591
...

result:

ok 95405 lines

Test #4:

score: 0
Accepted
time: 97ms
memory: 14340kb

input:

10
9997 9996
1 1 1 0 0 0 1 1 0 1 0 0 0 0 0 0 0 1 0 0 0 1 1 0 1 0 0 1 0 1 0 0 0 1 0 1 0 0 0 1 0 1 1 0 0 0 0 1 1 1 0 0 1 1 1 1 0 0 0 1 1 1 0 1 0 0 0 1 1 1 0 0 1 1 0 1 0 1 1 0 0 0 1 1 1 1 0 0 0 1 1 0 0 1 0 0 1 0 1 1 0 0 0 1 0 0 0 1 0 1 1 0 0 0 1 1 0 0 1 1 0 0 1 1 0 1 1 1 0 1 0 1 1 0 0 0 0 0 1 1 1 1 1 1...

output:

4997
5001
4990
5018
5024
5011
5042
5066
4955
4971
4971
4937
4947
4983
5050
4971
5038
5038
4960
5032
4966
4924
4999
4957
5041
5041
4957
4973
5000
5025
4998
5015
5016
5020
5006
5020
5015
5036
5008
5008
4975
4996
4957
4986
5051
4998
5004
4990
4990
4965
4995
4963
4987
5009
4989
5007
5007
4985
4998
5007
...

result:

ok 99942 lines

Test #5:

score: 0
Accepted
time: 17ms
memory: 16104kb

input:

10000
10 10
0 1 0 1 1 0 0 0 0 1
3 5
5 10
5 4
2 10
2 7
9 4
2 1
3 8
6 7
2 5
1 7
1 5
2 1
1 7
1 9
2 1
2 7
1 8
2 4
4 10
1 1 0 1
1 4
1 2
2 3
1 4
1 4
2 4
1 3
2 3
2 3
1 2
2 2
2 1
2 4
10 10
0 1 0 1 1 1 0 0 1 0
10 7
7 3
10 5
10 1
1 8
5 6
10 9
2 9
4 3
2 1
2 6
1 1
2 7
1 7
1 5
1 3
1 5
2 7
1 3
4 10
0 1 1 1
1 2
3 ...

output:

7
5
5
3
5
4
4
3
4
7
2
1
3
2
2
2
3
2
2
2
3
3
5
5
5
5
5
5
5
5
2
2
2
2
2
1
3
2
1
1
1
2
2
2
2
2
1
1
1
2
4
5
4
3
5
5
3
4
5
4
4
5
2
3
2
4
5
4
2
3
2
2
2
2
2
2
2
3
3
3
5
2
2
7
2
2
3
2
7
6
4
3
1
1
2
2
3
4
5
4
6
6
4
5
6
4
5
6
6
7
5
2
4
2
4
1
3
2
4
4
3
7
2
6
6
7
6
1
1
1
0
0
1
1
2
1
2
1
2
2
4
2
1
1
2
2
4
3
3
2
...

result:

ok 100000 lines

Extra Test:

score: 0
Extra Test Passed