统计不同回文子字符串--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实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Java使用正则提取字符串中的内容的详细步骤

《Java使用正则提取字符串中的内容的详细步骤》:本文主要介绍Java中使用正则表达式提取字符串内容的方法,通过Pattern和Matcher类实现,涵盖编译正则、查找匹配、分组捕获、数字与邮箱提... 目录1. 基础流程2. 关键方法说明3. 常见场景示例场景1:提取所有数字场景2:提取邮箱地址4. 高级

IDEA与MyEclipse代码量统计方式

《IDEA与MyEclipse代码量统计方式》文章介绍在项目中不安装第三方工具统计代码行数的方法,分别说明MyEclipse通过正则搜索(排除空行和注释)及IDEA使用Statistic插件或调整搜索... 目录项目场景MyEclipse代码量统计IDEA代码量统计总结项目场景在项目中,有时候我们需要统计

Python Flask实现定时任务的不同方法详解

《PythonFlask实现定时任务的不同方法详解》在Flask中实现定时任务,最常用的方法是使用APScheduler库,本文将提供一个完整的解决方案,有需要的小伙伴可以跟随小编一起学习一下... 目录完js整实现方案代码解释1. 依赖安装2. 核心组件3. 任务类型4. 任务管理5. 持久化存储生产环境

Python 字符串裁切与提取全面且实用的解决方案

《Python字符串裁切与提取全面且实用的解决方案》本文梳理了Python字符串处理方法,涵盖基础切片、split/partition分割、正则匹配及结构化数据解析(如BeautifulSoup、j... 目录python 字符串裁切与提取的完整指南 基础切片方法1. 使用切片操作符[start:end]2

MyBatis的xml中字符串类型判空与非字符串类型判空处理方式(最新整理)

《MyBatis的xml中字符串类型判空与非字符串类型判空处理方式(最新整理)》本文给大家介绍MyBatis的xml中字符串类型判空与非字符串类型判空处理方式,本文给大家介绍的非常详细,对大家的学习或... 目录完整 Hutool 写法版本对比优化为什么status变成Long?为什么 price 没事?怎

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、