统计不同回文子字符串--LeetCode730:分析清晰

2024-06-02 15:38

本文主要是介绍统计不同回文子字符串--LeetCode730:分析清晰,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

统计不同回文子字符串–LeetCode730

题目

给定一个字符串 S,找出 S 中不同的非空回文子序列个数,并返回该数字与 10^9 + 7的模。通过从 S 中删除 0 个或多个字符来获得子字符序列。如果一个字符序列与它反转后的字符序列一致,那么它是回文字符序列。如果对于某个 iA_i != B_i,那么 A_1, A_2, ...和 B_1, B_2, ...这两个字符序列是不同的。

示例 1:

输入:
S = 'bccb'
输出:6
解释:
6 个不同的非空回文子字符序列分别为:'b', 'c', 'bb', 'cc', 'bcb', 'bccb'。
注意:'bcb' 虽然出现两次但仅计数一次。

示例 2:

输入:
S = 'abcdabcdabcdabcdabcdabcdabcdabcddcbadcbadcbadcbadcbadcbadcbadcba'
输出:104860361
解释:
共有 3104860382 个不同的非空回文子字符序列,对 10^9 + 7 取模为 104860361。

提示:

  • 字符串 S 的长度将在[1, 1000]范围内。
  • 每个字符 S[i] 将会是集合 {‘a’, ‘b’, ‘c’, ‘d’} 中的某一个。

思路

状态定义:dp[i][j]表示s[i, j]中回文子序列的个数。

转移方程:

  • 如果s[i] != s[j],则有dp[i][j] = dp[i][j - 1] + dp[i + 1][j] - dp[i + 1][j - 1]dp[i + 1][j - 1]dp[i][j - 1]dp[i+1][j]中都有,所以要减去一个。
  • 如果s[i] = s[j] = c,则有dp[i][j] = 2 * dp[i + 1][j - 1],乘以2意思是在s[i + 1, j - 1]中所有的回文子序列两边加上c,就构成了新的回文子序列,原来有dp[i + 1][j - 1]这么多个,两边加上c之后增加了dp[i + 1][j - 1]个回文子序列,所以乘以2得到当前的dp[i][j]

但是上述s[i] = s[j]情况下,结果存在重复计算和疏漏,需要进行去重和补充。

  • 如果在s[i + 1, j - 1]没有字符c,需要进行补充,即dp[i][j] += 2,补上的是[c], [c, c];
  • 如果在s[i + 1, j - 1]只有一个字符c,需要进行补充,即dp[i][j] += 1,补上的是[c, c];
  • 如果在s[i + 1, j - 1]有两个或多个字符c,需要进行去重,即dp[i][j] -= dp[l + 1][r - 1]li的左边第一个字符为c的位置, rj的右边第一个字符为c的位置。

初始条件:

  • dp[n - 1][n - 1] = 1表示单个字符就是一个回文子序列
  • dp[i][i - 1] = 0表示字符串为空

最终答案就是dp[0][n-1]

参考题解:
阿飞的题解
想想大司马会怎么做的题解

代码如下(用时69ms):

class Solution {public int countPalindromicSubsequences(String S) {if (S == null || S.length() == 0) {return 0;}int n = S.length();int mod = 1000000000+7;int[][] dp = new int[n][n];for (int i = 0; i < n; i++) {dp[i][i] = 1;}for (int i = n-2; i >= 0; i--) {for (int j = i+1; j < n; j++) {if (S.charAt(i) == S.charAt(j)) {dp[i][j] = 2*dp[i+1][j-1];int left = i+1;int right = j-1;while (left <= right && S.charAt(i) != S.charAt(left)) left++;while (left <= right && S.charAt(j) != S.charAt(right)) right--;if (left > right) {dp[i][j] += 2;} else if (left == right) {dp[i][j] += 1;} else {dp[i][j] -= dp[left+1][right-1];}}else {dp[i][j] = dp[i+1][j]+dp[i][j-1]-dp[i+1][j-1];}dp[i][j] = (dp[i][j] < 0) ? dp[i][j]+mod : dp[i][j]%mod;}}return dp[0][n-1];}
}

这篇关于统计不同回文子字符串--LeetCode730:分析清晰的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1024385

相关文章

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

java脚本使用不同版本jdk的说明介绍

《java脚本使用不同版本jdk的说明介绍》本文介绍了在Java中执行JavaScript脚本的几种方式,包括使用ScriptEngine、Nashorn和GraalVM,ScriptEngine适用... 目录Java脚本使用不同版本jdk的说明1.使用ScriptEngine执行javascript2.

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis主从复制的原理分析

《Redis主从复制的原理分析》Redis主从复制通过将数据镜像到多个从节点,实现高可用性和扩展性,主从复制包括初次全量同步和增量同步两个阶段,为优化复制性能,可以采用AOF持久化、调整复制超时时间、... 目录Redis主从复制的原理主从复制概述配置主从复制数据同步过程复制一致性与延迟故障转移机制监控与维

python修改字符串值的三种方法

《python修改字符串值的三种方法》本文主要介绍了python修改字符串值的三种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录第一种方法:第二种方法:第三种方法:在python中,字符串对象是不可变类型,所以我们没办法直接

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

锐捷和腾达哪个好? 两个品牌路由器对比分析

《锐捷和腾达哪个好?两个品牌路由器对比分析》在选择路由器时,Tenda和锐捷都是备受关注的品牌,各自有独特的产品特点和市场定位,选择哪个品牌的路由器更合适,实际上取决于你的具体需求和使用场景,我们从... 在选购路由器时,锐捷和腾达都是市场上备受关注的品牌,但它们的定位和特点却有所不同。锐捷更偏向企业级和专

C#中字符串分割的多种方式

《C#中字符串分割的多种方式》在C#编程语言中,字符串处理是日常开发中不可或缺的一部分,字符串分割是处理文本数据时常用的操作,它允许我们将一个长字符串分解成多个子字符串,本文给大家介绍了C#中字符串分... 目录1. 使用 string.Split2. 使用正则表达式 (Regex.Split)3. 使用