首页 > 其他分享 >CF231E 题解

CF231E 题解

时间:2024-02-07 10:45:25浏览次数:39  
标签:int 题解 pntcnt vising CF231E low treesum push

本文采用 CC BY-NC-SA 4.0 协议发布。

前言

提供一个圆方树做法。

孩子圆方树学傻了,忘了还有缩点这回事。

正文

建圆方树。

考虑一条圆方树上的路径,哪些点对答案有贡献:

  • 方点,这表示路径经过一个环,方案数 \(\times 2\).

  • 旁边有方点的圆点。这表示走到这时可以选择在环上绕一圈,方案数 \(\times 2\).

于是设“旁边有方点的圆点”权值为 \(2\),方点权值为 \(\dfrac{1}{2}\),其他点权值 \(1\),路径上的点权值乘起来就是方案数。取对数后转化为路径权值和问题。

代码

#include <iostream>
#include <cassert>
#include <vector>
#include <stack>
#define UP(i,s,e) for(auto i=s; i<e; ++i)
using std::cin; using std::cout;
namespace m{ // }{{{
constexpr int N = 1e5, M = 1e5, lgN = 17, MOD = 1e9+7;
int in, im;
std::vector<int> tos0[N], tos1[N*2];
int pntcnt, low[N], dfn[N]; //nearsq[N];
extern int treesum[];
std::stack<int> vising;
void tarjan_dfs(int x, int f){
    static int dfstim = 0;
    dfstim++;
    low[x] = dfn[x] = dfstim;
    vising.push(x);
    for(int i:tos0[x]){
        if(i == f) continue;
        if(dfn[i]){
            low[x] = std::min(low[x], dfn[i]);
            continue;
        }
        tarjan_dfs(i, x);
        low[x] = std::min(low[x], low[i]);
        if(low[i] > dfn[x]){
            assert(vising.top() == i);
            tos1[i].push_back(x);
            tos1[x].push_back(i);
            //while(vising.top() != i){
            //    vising.pop();
            //}
            vising.pop();
        } else if(low[i] == dfn[x]){
            while(1){
                int t = vising.top(); vising.pop();
                tos1[pntcnt].push_back(t);
                tos1[t].push_back(pntcnt);
                treesum[t] = 1;
                if(t == i) break;
            }
            tos1[pntcnt].push_back(x);
            tos1[x].push_back(pntcnt);
            treesum[x] = 1;
            //nearsq[x] = pntcnt;
            pntcnt++;
        }
    }
}
int dfn1[N*2], fa[N*2], treesum[N*2], st[lgN][N*2];
void dfs_mark(int x, int f){
    static int dfstim = -1; dfstim++;
    fa[x] = f; treesum[x] += (x==f ? 0 : treesum[f]) - (x>=in);
    dfn1[x] = dfstim;
    st[0][dfn1[x]] = fa[x];
    for(int i:tos1[x]){
        if(i == f) continue;
        dfs_mark(i, x);
    }
}
int dfnmin(int x, int y){ return dfn1[x] < dfn1[y] ? x : y; }
int pow2[N];
void prepare(){
    cin >> in >> im;
    pntcnt = in;
    UP(i, 0, im){
        int ia, ib; cin >> ia >> ib; ia--, ib--;
        tos0[ia].push_back(ib);
        tos0[ib].push_back(ia);
    }
    tarjan_dfs(0, 0);
    dfs_mark(0, 0);
    UP(layer, 1, lgN){
        UP(i, 0, pntcnt-(1<<layer)+1){
            st[layer][i] = dfnmin(st[layer-1][i], st[layer-1][i+(1<<layer>>1)]);
        }
    }
    pow2[0] = 1;
    UP(i, 1, pntcnt-in+1){
        pow2[i] = pow2[i-1] * 2 % MOD;
    }
}
int getlca(int x, int y){
    if(x == y) return x;
    x = dfn1[x]; y = dfn1[y];
    if(x > y) std::swap(x, y);
    int delta = std::__lg(y-x);
    return dfnmin(st[delta][x+1], st[delta][y-(1<<delta)+1]);
}
void work(){
    int ix, iy; cin >> ix >> iy; ix--, iy--;
    //if(nearsq[ix]) ix = nearsq[ix];
    //if(nearsq[iy]) iy = nearsq[iy];
    int lca = getlca(ix, iy);
    int sum = treesum[ix] + treesum[iy] - treesum[lca] - (lca ? treesum[fa[lca]] : 0);
    cout << pow2[sum] << '\n';
}
} // {}}}
int main(){ 
    m::prepare();
    int iq;
    cin >> iq;
    while(iq--) m::work();
}

标签:int,题解,pntcnt,vising,CF231E,low,treesum,push
From: https://www.cnblogs.com/x383494/p/18010714

相关文章

  • 洛谷P10136 暨 USACOJan2024S T3 题解
    题意简述原题已经很简了,没有什么简述的必要了。思维路径请注意本题解可以保证正确性但不保证如果有极端的Hack数据能够通过。拿到这道题上来的暴力想必是很容易的,即枚举每个\(L\)判断是否合法。接着我们就考虑优化,减少需要枚举的\(L\)的量。题目中要求余数最多有\(3\)......
  • CF1408F Two Different 题解
    解题思路先考虑如何将一堆数变为相同的。显然,这里有一个条件\(n=2^k,k\in\mathbbZ\),证明如下:每次操作只能将两个数变为相同的,那么一个数在使得其他数变为相同数的操作中(我们不妨将所有数进行这种操作称为一轮操作),一个数最多被使用一次;按照错位操作,即第一轮\(1\)和\(2\)......
  • CF1428D Bouncing Boomerangs 题解
    解题思路很简单的贪心。观察发现以下性质:当\(a_i=2\)时,这一行一定只有两个目标,且第二个目标一定位于一个\(a_j=1\)的格子内;当\(a_i=3\),那么当前列右边某一列发生转向的地方,\(a_j\not=0\);那么这道题就基本已经做出来了。因为\(a_i=3\)的格子转向格可以在任意非\(0\)......
  • CF1401E Divide Square 题解
    解题思路其实多看几组也能发现块数等于交点的数量加上两个端点都在边上的线段数量再加一。证明如下(图见样例):对于两条只有一个端点位于边上的线段,因为保证有一个端点位于边上,那么这两条线段的交点一定会和已存在的点、边构成一个新的矩形;对于其中有一条为两个端点均位于边上的......
  • CF1401F Reverse and Swap 题解
    解题思路巧妙的数据结构题,非常巧妙的利用的线段树的奇妙性质。操作逐条维护:Replace:直接线段树上单点修改;Sum:线段树查询区间和;Reverse:考虑线段树的形态。线段树第\(i\)层(除最后一层)有\(2^{i-1}\)个节点,那么将所有\(i\ge1\)的区间\([(i-1)\times2^k,i\times2^k]\)......
  • CF1408E Avoid Rainbow Cycles 题解
    解题思路第一眼看过去感觉不是很可做……但是我们可以发现,如果有两个点在不同的集合中出现过,那么一定会存在彩虹环,那么两个点最多出现一次。同时我们考虑将题意转化一下,变成求最大能选取的点,使得不出现彩虹环。根据刚刚的性质,我们可以考虑每个点向它所在的集合连一条边权为\(a_......
  • CF1338C Perfect Triples 题解
    解题思路没什么好说的,就是打表找规律……表在这里不难发现,三元组中第一个数的最后两位按照\(00\to01\to10\to11\)的顺序变化,其他位也一样,同样,第二个数和第三个数中每两位分别按照\(00\to10\to11\to01\)和\(00\to11\to01\to10\)的顺序变化,且与第一个数对应变化......
  • CF1379C Choosing flowers 题解
    解题思路不是那么显然的,当只选一种\(b_i\)或全选\(a_i\)时最优。那么我们可以先对\(a_i\)从大到小排序,枚举每一个\(b_i\),然后二分找到第一个大于等于\(b_i\)的\(a_j\),判断\(a_1\sima_j\)中是否包含\(a_i\),如果包含,当前的答案为\(\displaystyle\left(\sum_{k=1}^{......
  • CF1385F Removing Leaves 题解
    解题思路简单贪心,优先选择叶子节点最多的,这样能够保证一定能把所有能删的都删了。因为要建一个可删除的图,所以我们可以使用set来存边,不然就需要维护一堆东西……那么我们肯定是从有叶子节点的点向父亲更新的,那么我们每次选择叶子节点最多的点,然后删除\(k\)个叶子,判断一下删......
  • CF1415E New Game Plus! 题解
    解题思路简单贪心题,我们可以把整个序列看作\(k+1\)个可重集,首先可以得到一个显然的结论:较大的数一定比较小的数先放入一个集合中。同样,由于每一轮\(ans\getsans+\maxsum_i\),其中\(sum_i\)表示第\(i\)个集合的元素和,那么,我们一定会将当前的元素放入当前和最大的哪个集合......