首页 > 其他分享 >ABC351G题解

ABC351G题解

时间:2024-04-27 22:58:44浏览次数:25  
标签:g2 int 题解 son fa dfn zz ABC351G

考虑动态 dp 的套路,先树剖一下,令 \(son(x)\) 为点 \(x\) 的重儿子。 \(g_x=\prod\limits_{u\in C(x)\land u\neq son(x)}f_u\)。

于是有 \(f_x=f_{son(x)}g_x+a_x\)。

于是 \(\begin{bmatrix}f_x&1\end{bmatrix}=\begin{bmatrix}f_{son(x)}&1\end{bmatrix}\begin{bmatrix}g_x&0\\ a_x&1\end{bmatrix}\)。

然后直接线段树维护矩阵乘法就行了。

不懂的可以做一下 P4719

代码:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

#define int ll

const int N = 2e5 + 5, p = 998244353;
int dfn[N], top[N], dn[N], fa[N], son[N], sz[N], a[N], c[N], f[N], g[N], g2[N], ts, n;
bool zz[N];

vector<int> e[N];
struct M
{
    int a[2][2];
    M operator*(M x)
    {
        M y; memset(y.a, 0, sizeof y.a);
        for(int i = 0; i < 2; i ++) for(int j = 0; j < 2; j ++) for(int k = 0; k < 2; k ++)
            (y.a[i][k] += a[i][j] * x.a[j][k]) %= p;
        return y;
    }
};

struct sgt
{
    M a[N << 2];
    void pu(int x) {a[x] = a[x << 1 | 1] * a[x << 1];}
    void upd(int q, int l, int r, int x, M v)
    {
        if(l == r) return a[x] = v, void();
        int mid = l + r >> 1;
        if(mid >= q) upd(q, l, mid, x << 1, v);
        else upd(q, mid + 1, r, x << 1 | 1, v);
        pu(x);
    }
    M qry(int ql, int qr, int l, int r, int x)
    {
        if(ql <= l && r <= qr) return a[x];
        int mid = l + r >> 1;
        if(mid < ql) return qry(ql, qr, mid + 1, r, x << 1 | 1);
        if(mid >= qr) return qry(ql, qr, l, mid, x << 1);
        return qry(ql, qr, mid + 1, r, x << 1 | 1) * qry(ql, qr, l, mid, x << 1);
    }
}t;

void dfs1(int x, int fa)
{
    sz[x] = 1;
    ::fa[x] = fa;
    f[x] = 1;
    for(int i : e[x])
    {
        dfs1(i, x);
        f[x] = f[x] * f[i] % p;
        sz[x] += sz[i];
        if(sz[i] >= sz[son[x]]) son[x] = i;
    }
    if(e[x].empty()) f[x] = 0;
    f[x] = (f[x] + a[x]) % p;
    zz[x] = f[x] == 0;
}

void dfs2(int x, int tp)
{
    dfn[x] = ++ts;
    top[x] = tp;
    dn[tp] = x;
    if(son[x]) dfs2(son[x], tp);
    g[x] = g2[x] = 1;
    for(int i : e[x])
        if(i != son[x])
        {
            dfs2(i, i);
            c[x] += !f[i];
            g[x] = g[x] * f[i] % p;
            if(f[i]) g2[x] = g2[x] * f[i] % p;
        }
    if(e[x].empty()) g[x] = g2[x] = 0;
    t.upd(dfn[x], 1, n, 1, {g[x], 0, a[x], 1});
}

int qpow(int a, int b)
{
    if(!b) return 1;
    return ((b & 1) ? a : 1ll) * qpow(a * a % p, b >> 1) % p;
}

void upd(int u, int v)
{
    M x = t.qry(dfn[u], dfn[u], 1, n, 1);
    x.a[1][0] = v;
    t.upd(dfn[u], 1, n, 1, x);
    while(fa[top[u]])
    {
        u = top[u];
        M y = t.qry(dfn[u], dfn[dn[u]], 1, n, 1);
        int nf = y.a[1][0];
        if(nf && zz[u]) c[fa[u]] --, zz[u] = 0;
        else if(nf == 0 && !zz[u]) c[fa[u]] ++, zz[u] = 1;
        M z = t.qry(dfn[fa[u]], dfn[fa[u]], 1, n, 1);
        if(f[u]) g2[fa[u]] = qpow(f[u], p - 2) * g2[fa[u]] % p;
        if(c[fa[u]]) g[fa[u]] = 0;
        else
        {
            g2[fa[u]] = g2[fa[u]] * nf % p;
            g[fa[u]] = g2[fa[u]];
        }
        f[u] = nf;
        z.a[0][0] = g[fa[u]];
        t.upd(dfn[fa[u]], 1, n, 1, z);
        u = fa[u];
    }
}

signed main()
{
    ios::sync_with_stdio(0);cin.tie(0);
    int q;
    cin >> n >> q;
    for(int i = 2; i <= n; i ++)
    {
        int x; cin >> x;
        e[x].push_back(i);
    }
    for(int i = 1; i <= n; i ++) cin >> a[i];
    dfs1(1, 0);
    dfs2(1, 1);
    while(q --)
    {
        int x, v; cin >> x >> v;
        upd(x, v);
        cout << t.qry(dfn[1], dfn[dn[1]], 1, n, 1).a[1][0] << "\n";
    }

    return 0;
}

标签:g2,int,题解,son,fa,dfn,zz,ABC351G
From: https://www.cnblogs.com/adam01/p/18162698

相关文章

  • 【BFS】abc351D Grid and Magnet 题解
    传送门D-GridandMagnet题意:给定h行w列的图表,其中部分位置有磁铁,人物会在磁铁四周((x+1,y),(x-1,y),(x,y+1),(x,y-1))停止,某点的自由度定义为从该点出发最多可到的方块数目可以走重复路前置例题:求连通块大小洛谷P1141思路:由自由度的定义联想到连通块的大小,从而决定用BFS......
  • ABC351_F 题解
    实际上很板。考虑在\(i\)后小于\(val_i\)的数都对答案没贡献,所以我们只需要知道在\(i\)后且大于\(val_i\)的数的和以及有多少个这样的数就可以了。知道了我们要求什么,就可以一眼权值线段树。从后往前扫不断加入数,然后访问对应区间即可,当然因为值域比较大,所以还要离散化......
  • eclipse 题解
    Statement给定一个圆,圆按照顺时针排布着\(2n\)个点,依次编号为\(1\simn\),其中编号为\(1\simn\)的点属于Alice,编号为\((n+1)\sim2n\)的点属于Bob。同时给出两个长度为\(n\)的序列\(A,B\)。你需要确定一个最大的正整数\(K\),使得存在\(K\)个二元组\((x_i,y_i)\)......
  • 题解:洛谷 P1137 旅行计划
    标签:图论,拓扑,dp题意给定一张\(n\)个点\(m\)条边的DAG,对于每个\(i\),求以它为终点最多经过多少个点?思路由于是DAG,求的是终点\(i\)经过的所有点,而刚好拓扑序就满足这个。那么就可以考虑拓扑排序。设\(f_i\)是以\(i\)为终点的最多结点数,那么就有转移方程\(f_v=m......
  • CF1842H Tenzing and Random Real Numbers 题解
    题目链接点击打开链接题目解法实数的概率好反直觉!对\(1\)做限制不是很好做,考虑变成正负性的限制(即对\(0\)做限制)令\(y_i=x_i-0.5\),那么限制就变成了\(y_i+y_j\le0,\;y_i+y_j\ge0\)这里要给出一些实数概率的结论:实数下,\(x=y\)的概率为\(0\),因为\(\frac{1}{\inft......
  • CAUC_CTF 题解
    caucctfwpez_隐写如果计算机是中国人发明的Welcome!easy_rsafromCrypto.Util.numberimport*importgmpy2importlibnumimportrandomimporthashlibn=0x52d483c27cd806550fbe0e37a61af2e7cf5e0efb723dfc81174c918a27627779b21fa3c851e9e94188eaee3d5cd6f7524......
  • [题解]P2015 二叉苹果树
    P2015二叉苹果树树形dp,一般用dfs辅助解决。当我们搜索到\(u\),此时剩下\(cnt\)条边可以用,也就是说\(u\)为根节点的子树最多可以保留\(cnt\)条边。由于上一层的需求,我们显然需要枚举剩余边数\(i\)(\(1\leqi\leqcnt\))。接下来对于每个\(i\),我们考虑剩余的\(u\)条边可以怎么放。......
  • 题解:P10329 [UESTCPC 2024] Add
    Add题意将序列进行一系列的操作,输出对\(a_{1}\)的期望值。题目中操作说的比较明了,再次就不特殊声明了。思路据题意所知,每一个\(n\)应该对应了一个固定的答案。于是我就想到可以打表,就打出了下面的式子。n=1时ans=1n=2时ans=5n=3时ans=14n=4时ans=30n=5时ans=5......
  • vue,js直接导出excel,xlsx的方法,XLSX_STYLE 行高设置失效的问题解决
    1、先安装依赖:xlsx、xlsx-style、file-saver三个包npminstallxlsxxlsx-stylefile-saver2、引入:<script>import*asXLSXfrom'xlsx/xlsx.mjs'importXLSX_STYLEfrom'xlsx-style';import{saveAs}from'file-saver';exportdefau......
  • 题解笔记
    P1196银河英雄传说带权并查集(根搭积木很像):对于每个点,分别记录所属链的头结点、该点到头结点的距离以及它所在集合的大小。每次合并将y接在x的尾部,改变y头的权值和所属链的头结点,同时改变x的尾节点。注意:每次查找的时候也要维护每个节点的权值。每次查询时计算两点的权值差。......