「Codeforces」245H Queries for Number of Palindromes (区间dp)

题意:原题在这

You‘ve got a string s = s1s2... s|s| of length |s|, consisting of lowercase English letters. There also are q queries, each query is described by two integers li, ri (1 ≤ li ≤ ri ≤ |s|). The answer to the query is the number of substrings of string s[li... ri], which are palindromes.

String s[l... r] = slsl + 1... sr (1 ≤ l ≤ r ≤ |s|) is a substring of string s = s1s2... s|s|.

String t is called a palindrome, if it reads the same from left to right and from right to left. Formally, if t = t1t2... t|t| = t|t|t|t| - 1... t1.

给定一个字符串,q个询问,每个询问这个区间内有多少个回文串

思路:

定义vis[i][j]表示是不是回文串;
dp[i][j]为i到j的回文串总数

初始化一个:

    1.自己是回文串
    2.相邻的是回文串

再用can(x,y)递归判断vis[x][y]其他情况:

bool can(int x,int y)
{
    if (vis[x][y]) return true;
    else return vis[x][y]=(s[x]==s[y])&&(can(x+1,y-1));
}

代码:

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define inf 99999999999
#define maxn 1000005
using namespace std;
string s;
int q,cnt,len;
bool vis[5005][5005];//vis[i][j]表示是否为回文串
int dp[5005][5005];//dp[i][j]表示回文子串数量

struct description
{
    int l,r;
}f[maxn];

bool can(int x,int y)
{
    if (vis[x][y]) return true;
    else return vis[x][y]=(s[x]==s[y])&&(can(x+1,y-1));
}

int main()
{
    cin>>s;
    len=s.size();
    //初始化
    memset(dp,0,sizeof(dp));
    memset(vis,0,sizeof(vis));
    for(int i=0;i<=len;i++)//自己是回文串
    {
        vis[i][i]=1;
        dp[i][i]=1;
    }
    for(int i=0;i<len;i++)//相邻的也是
    {
        if(s[i]==s[i+1])
        {
            vis[i][i+1]=1;
            dp[i][i+1]=3;
        }
        else dp[i][i+1]=2;
    }

    for(int k=2;k<len;k++)
    {
        int l=0;
        for(int r=k;r<len;r++)
        {
            dp[l][r]=dp[l+1][r]+dp[l][r-1]-dp[l+1][r-1];
            if (can(l,r)) dp[l][r]++;
            l++;
        }
    }

    cin>>q;
    for(int i=1;i<=q;i++)
    {
        scanf("%d%d",&f[i].l,&f[i].r);
        // cin>>f[i].l>>f[i].r;
        // cout<<dp[f[i].l-1][f[i].r-1]<<endl;
        printf("%d\n",dp[f[i].l-1][f[i].r-1]);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/LocaEtric/p/9615062.html

时间: 2024-10-18 17:27:10

「Codeforces」245H Queries for Number of Palindromes (区间dp)的相关文章

codeforces 245H H. Queries for Number of Palindromes(区间dp)

题目链接: codeforces 245H 题目大意: 给出一个字符串,询问任意区间内的回文子串的个数. 题目分析: 定义isPar[i][j]表示区间字符串[i,j]是否是回文,可以通过isPar[i+1][j-1]递推得到. 定义dp[i][j]表示及区间[i,j]内的回文子串的个数,转移方程如下: dp[i][j]=dp[i+1][j]+dp[i][j?1]?dp[i+1][j?1]+isPar[i][j] 用到了一点容斥的思想. AC代码: #include <iostream> #i

dp --- Codeforces 245H :Queries for Number of Palindromes

Queries for Number of Palindromes Problem's Link:   http://codeforces.com/problemset/problem/245/H Mean: 给你一个字符串,然后q个询问:从i到j这段字符串中存在多少个回文串. analyse: dp[i][j]表示i~j这段的回文串数. 首先判断i~j是否为回文,是则dp[i][j]=1,否则dp[i][j]=0; 那么dp[i][j]=dp[i][j]+dp[i][j-1]+dp[i+1[j

Codeforces 245H Queries for Number of Palindromes

题意:给你一个字符串,再给你一个q(询问个数1000000),问你这个区间内回文串的个数. 解题思路: 1)dp,先把 i 到j 是回文串的真值赋值给 dp[i][j] ,然后从后往前dp    dp[i][j] += dp[i+1][j] + dp[i][j-1]  -dp[i+1][j-1]; 解题代码: 1 // File Name: 245h.dp.cpp 2 // Author: darkdream 3 // Created Time: 2015年03月30日 星期一 15时27分10

Codeforces 245H Queries for Number of Palindromes:区间dp

题目链接:http://codeforces.com/problemset/problem/245/H 题意: 给你一个字符串s. 然后有t个询问,每个询问给出x,y,问你区间[x,y]中的回文子串的个数. 题解: 表示状态: dp[x][y] = numbers 表示区间[x,y]中的回文子串个数. 找出答案: 每次询问:ans = dp[x][y] 如何转移: dp[x][y] = dp[x][y-1] + dp[x+1][y] - dp[x+1][y-1] + pal[x][y] 用到了容

[CF245H] Queries for Number of Palindromes (容斥原理dp计数)

题目链接:http://codeforces.com/problemset/problem/245/H 题目大意:给你一个字符串s,对于每次查询,输入为一个数对(i,j),输出s[i..j]之间回文串的个数. 容斥原理: dp[i][j] = dp[i+1][j]+dp[i][j-1]-dp[i+1][j-1]; if( str[i]==str[j] 并且 str[i+1..j-1]是回文串 ) dp[i][j]++; 代码: 1 #include <cstdio> 2 #include &l

「Codeforces」758D(贪心细节/dp)

题意:原题在这 Alexander is learning how to convert numbers from the decimal system to any other, however, he doesn't know English letters, so he writes any number only as a decimal number, it means that instead of the letter A he will write the number 10.

CF245H Queries for Number of Palindromes(回文树)

题意翻译 题目描述 给你一个字符串s由小写字母组成,有q组询问,每组询问给你两个数,l和r,问在字符串区间l到r的字串中,包含多少回文串. 输入格式 第1行,给出s,s的长度小于5000 第2行给出q(1<=q<=10^6) 第2至2+q行 给出每组询问的l和r 输出格式 输出每组询问所问的数量. 题目描述 You've got a string s=\(s_{1}\)\(s_{2}\)...\(s_{|s|}\) of length |s| , consisting of lowercase

「模板」线段树静态开点(单点+区间修改)、动态开点

相关讲解资料: 树状数组:https://blog.csdn.net/qq_34374664/article/details/52787481 (线段树预备) 线段树讲解: 初学版:https://blog.csdn.net/zearot/article/details/52280189 进阶完整版:https://www.cnblogs.com/AC-King/p/7789013.html 代码: 完整注释模板一张,参(chao)考(xi)楼上的博客 #include<iostream> #

「JSOI2019」神经网络(容斥+组合计数+背包dp)

Address luogu5333 loj3102 Solution 容易发现,一条哈密顿回路本质上就是:把每棵树都拆成若干条有向路径,再把所有的有向路径连接成环,环上的相邻两条有向路径不可以来自同一棵树. 先求出 \(g_{i,j}\) 表示把第 \(i\) 棵树拆成 \(j\) 条有向路径的方案数. 考虑 \(\text{dp}\),记 \(f_{u,i,0/1/2/3}\) 分别表示:\(u\) 的子树拆成 \(i\) 条路径,\(u\) 是路径起点,是路径终点,单点成路径,既不是路径起点