首页 > 其他分享 >Bingbong的回文路径

Bingbong的回文路径

时间:2024-04-20 16:23:04浏览次数:24  
标签:int 路径 Bingbong fa 哈希 字符串 回文 节点 mathrm

Bingbong的回文路径

题目描述

Bingbong 有一棵有根树,根节点为 $1$,总共有 $n$ 个节点。树中的节点通过 $n−1$ 条无向边相连,每条边的权重为 $1$。

树中的每个节点包含一个小写字母。Bingbong 将选择从节点 $u$ 开始,并在选择最短路径的情况下到达节点 $v$。他想知道他所走路径形成的字符串是否是一个回文串。由于他会进行多次行走,您需要回答多个查询。

输入描述:

第一行包含一个整数 $n(1 \leq n \leq 10^5)$,表示节点的数量。

第二行包含 $n$ 个小写字母,其中第 $i$ 个字母表示第 $i$ 个节点上的小写字母。

第三行包含 $n$ 个整数,其中第 $i$ 个数字 $a_i​(1 \leq a_​i \leq i−1)$ 表示第 $i$ 个节点的父节点,对于根节点 $a_i​=0$。

第四行包含一个整数 $q(1 \leq q \leq 10^5)$,表示查询的数量。

接下来是 $q$ 行,每行包含两个整数 $u$ 和 $v$,表示一个查询的起点和终点。

输出描述:

输出 $q$ 行,每行包含一个字符串。如果查询的起点到终点的路径连接形成一个回文串,则输出 "YES",否则输出 "NO"。

示例1

输入

5
bbaaa
0 1 1 2 2
3
2 3
4 3
5 3

输出

NO
YES
YES

 

解题思路

  假设 $u$ 和 $v$ 的最近公共祖先是 $p$,从 $u$ 走到 $p$ 再走到 $v$ 的字符串表示为 $s_{upv}$,同理从 $v$ 走到 $p$ 再走到 $u$ 的字符串表示为 $s_{vpu}$。当 $s_{upv} = s_{vpu}$ 时说明 $u$ 到 $v$ 路径上的字符串是回文串。判断回文串可以用字符串哈希(这里基数 $P$ 取 $13331$,自然溢出取模),由于是树上的问题我们可以通过倍增来维护树链的信息。

  定义 $\mathrm{fa}[u][i]$ 表示从 $u$ 往上走 $2^i$ 步后达到的节点,$h_1[u][i]$ 表示从 $u$ 往上走 $2^i$ 步构成字符串的哈希值(不含 $u$ 上的字符),$h_2[u][i]$ 表示从 $\mathrm{fa}[u][i]$ 往下走 $2^i$ 步构成字符串的哈希值(含 $\mathrm{fa}[u][i]$ 上的字符)。以下图为例:

  从 $u$ 往上走 $2^2$ 步构成的字符串为 acbc,从 $\mathrm{fa}[u][2]$ 往下走 $2^2$ 步构成的字符串为 cbca

  考虑如何维护这些信息。对于 $\mathrm{fa}[u][i]$,我们先从 $u$ 往上走 $2^{i-1}$ 步到达 $\mathrm{fa}[u][i-1]$,再从 $\mathrm{fa}[u][i-1]$ 往上走 $2^{i-1}$ 步走到 $\mathrm{fa}[u][i]$,因此有 $\mathrm{fa}[u][i] = \mathrm{fa}[\mathrm{fa}[u][i-1]][i-1]$。

  对于两个字符串 $s_1$ 和 $s_2$,哈希值分别为 $h_1$ 和 $h_2$,那么拼接后的 $s_1 \, s_2$ 的哈希值就是 $h_1 \cdot P^{|s_2|} + h_2$。因此 $h_1[u][i] = h_1[u][i-1] \cdot P^{2^{i-1}} + h_1[\mathrm{fa}[u][i-1]][i-1]$,$h_2[u][i] = h_2[\mathrm{fa}[u][i-1]][i-1] \cdot P^{2^{i-1}} + h_2[u][i-1]$。

  对于询问的 $(u,v)$,我们可以仿照求最近公共祖先的做法,分别求出 $u \to v$ 和 $v \to u$ 对应字符串的哈希值。在 $u$ 往上跳到 $p$ 的过程中,维护 $hu_1$ 表示从 $u$ 往上跳的过程中构成字符串的哈希值,和 $hu_2$ 表示往下跳到 $u$ 的过程中构成字符串的哈希值。同理维护 $v$ 的 $hv_1$ 和 $hv_2$。最后让 $u$ 跳到 $p$,$v$ 跳到 $p$ 的下一个节点,判断是否满足 $hu_1 \cdot P^{d2} + hv_2 = hv_1 \cdot P^{d1} + hu_2$ 即可,其中 $d_1$ 表示 $u$ 往上跳经过的节点个数,$d_2$ 表示 $v$ 往上跳经过的节点个数。

  AC 代码如下,时间复杂度为 $O(n \log{n} + q \log{n})$:

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

typedef unsigned long long ULL;

const int N = 1e5 + 5, M = N * 2, P = 13331;

char s[N];
int h[N], e[M], ne[M], idx;
int fa[N][17], dep[N];
ULL h1[N][17], h2[N][17], pp[N];

void add(int u, int v) {
    e[idx] = v, ne[idx] = h[u], h[u] = idx++;
}

void dfs(int u, int p) {
    dep[u] = dep[p] + 1;
    fa[u][0] = p;
    h1[u][0] = h2[u][0] = s[p];
    for (int i = 1; i <= 16; i++) {
        fa[u][i] = fa[fa[u][i - 1]][i - 1];
        h1[u][i] = h1[u][i - 1] * pp[1 << i - 1] + h1[fa[u][i - 1]][i - 1];
        h2[u][i] = h2[fa[u][i - 1]][i - 1] * pp[1 << i - 1] + h2[u][i - 1];
    }
    for (int i = h[u]; i != -1; i = ne[i]) {
        int v = e[i];
        if (v == p) continue;
        dfs(v, u);
    }
}

bool query(int a, int b) {
    if (dep[a] < dep[b]) swap(a, b);
    ULL ha1 = s[a], ha2 = s[a], hb1 = s[b], hb2 = s[b];
    int d1 = 1, d2 = 1;
    for (int i = 16; i >= 0; i--) {
        if (dep[fa[a][i]] >= dep[b]) {
            ha1 = ha1 * pp[1 << i] + h1[a][i];
            ha2 = h2[a][i] * pp[d1] + ha2;
            d1 += 1 << i;
            a = fa[a][i];
        }
    }
    if (a == b) return ha1 == ha2;
    for (int i = 16; i >= 0; i--) {
        if (fa[a][i] != fa[b][i]) {
            ha1 = ha1 * pp[1 << i] + h1[a][i];
            ha2 = h2[a][i] * pp[d1] + ha2;
            d1 += 1 << i;
            a = fa[a][i];
            hb1 = hb1 * pp[1 << i] + h1[b][i];
            hb2 = h2[b][i] * pp[d2] + hb2;
            d2 += 1 << i;
            b = fa[b][i];
        }
    }
    // 当前a和b往上跳1步就会同时到达最近公共祖先,选择让a跳
    ha1 = ha1 * pp[1] + h1[a][0];
    ha2 = h2[a][0] * pp[d1++] + ha2;
    return ha1 * pp[d2] + hb2 == hb1 * pp[d1] + ha2;
}

int main() {
    int n, m;
    scanf("%d %s", &n, s + 1);
    memset(h, -1, sizeof(h));
    for (int i = 1; i <= n; i++) {
        int x;
        scanf("%d", &x);
        add(i, x), add(x, i);
    }
    pp[0] = 1;
    for (int i = 1; i <= n; i++) {
        pp[i] = pp[i - 1] * P;
    }
    dfs(1, 0);
    scanf("%d", &m);
    while (m--) {
        int a, b;
        scanf("%d %d", &a, &b);
        printf("%s\n", query(a, b) ? "YES" : "NO");
    }
    
    return 0;
}

 

参考资料

  牛客小白月赛91文字版题解:https://ac.nowcoder.com/discuss/1294108

标签:int,路径,Bingbong,fa,哈希,字符串,回文,节点,mathrm
From: https://www.cnblogs.com/onlyblues/p/18147822

相关文章

  • leetcode回文数
    给你一个整数x,如果x是一个回文整数,返回true;否则,返回false。回文数是指正序(从左向右)和倒序(从右向左)读都是一样的整数。例如,121是回文,而123不是。示例1:输入:x=121输出:true示例2:输入:x=-121输出:false解释:从左向右读,为-121。从右向左读,为121-。因此......
  • blog.admin net8发布二级目录,及图片上传路径处理
    1、发布二级目录,修改以下配置,及对应的二级目录名跟配置的一致 2、图片上传a、修改后台api代码imgController.cspublicasyncTask<MessageModel<string>>InsertPicture([FromForm]UploadFileDtodto){if(dto.file==null||!dto.file.Any())returnFail......
  • NL2SQL技术方案系列(1):NL2API、NL2SQL技术路径选择;LLM选型与Prompt工程技巧,揭秘项目落
    NL2SQL技术方案系列(1):NL2API、NL2SQL技术路径选择;LLM选型与Prompt工程技巧,揭秘项目落地优化之道NL2SQL基础系列(1):业界顶尖排行榜、权威测评数据集及LLM大模型(SpidervsBIRD)全面对比优劣分析[Text2SQL、Text2DSL]NL2SQL基础系列(2):主流大模型与微调方法精选集,Text2SQL经典算法......
  • 小心启动路径带来的小坑
    最近在做一个需求,需要调用同级目录的第三方程序进行数据处理。快速的薅了一个实现Processprocess=newProcess();process.StartInfo=newProcessStartInfo(){};process.StartInfo.FileName="ProgramName.exe";process.StartInfo.CreateNoWindow=true;process.Star......
  • 马拉车Mananer(求最长回文子序列)
    直接上板子直接输出最长回文子序列的长度#include<bits/stdc++.h>#defineintlonglong#defineIOSios::sync_with_stdio(0);cin.tie(0);cout.tie(0);usingnamespacestd;chars[32000005],st[32000005];intp[32000005];intchange(){intlen=strlen(st);......
  • Python案例:输出公元后到目前为止全部回文日期
    一、回文日期一个日期,如果顺读和倒读都一样,那么就称之为回文日期,比如今天:20211202,就是一个神奇的回文日期。二、提出任务输出公元后的全部回文日期要求每行输出五个回文日期统计总共有多少个回文日期三、完成任务(一)涉及知识点1、time模块time模块有两个常用函数time()......
  • 用log4net写入不同路径的日志文件
      用log4net写入不同路径的日志文件///<summary>///根据_jobName路径写入不同日志///</summary>publicclassNLogger{privatestaticDictionary<string,ILog>Loggers=newDictionary<string,ILog>();privatestr......
  • VBS遍历文件或文件夹路径输入文件的所有绝对路径(附源码)
    <p>源码如下:</p>FunctionlistFilesPath(filepath)t1=Timer()Debug.WriteLine"****现在开始执行计数,用时:"+CStr(t1)Setfso=CreateObject("scripting.filesystemobject")Setmyfolder=fso.GetFolder(filepath)......
  • 27天【代码随想录算法训练营34期】第七章 回溯算法part03(● 39. 组合总和 ● 40.组合
    39.组合总和怎么才能避免重复?比现在数小的数就别append到path里面了,之前肯定都试过了classSolution:defcombinationSum(self,candidates:List[int],target:int)->List[List[int]]:result=[]candidates.sort()self.backtracking(cand......
  • 2XC3 最短路径算法
    计算机科学:最终项目此项目将包括最终报告和您的代码。您的最终报告将包括以下内容。你会正在为此最终项目提交.py(NOT*.ipynb)文件。•标题页•目录•图表表•一份执行摘要,强调你的实验/分析的一些主要收获•向TA解释如何导航代码的附录。对于每个实验,在你的实验室报告中包括一个与......