730. 统计不同回文子字符串

给定一个字符串 S,找出 S 中不同的非空回文子序列个数,并返回该数字与 10^9 + 7 的模。

通过从 S 中删除 0 个或多个字符来获得子字符序列。

如果一个字符序列与它反转后的字符序列一致,那么它是回文字符序列。

如果对于某个 i,A_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’} 中的某一个。

PS:

因为只有四种字符,dp的第一位是哪几种字符,第二个是字符串的一个索引,第三个是字符串的第二个索引

class Solution {
public int countPalindromicSubsequences(String S) {
int n = S.length();
int mod = 1000000007;
int[][][] dp = new int[4][n][n]; for (int i = n-1; i >= 0; --i) {
for (int j = i; j < n; ++j) {
for (int k = 0; k < 4; ++k) {
char c = (char) ('a' + k);
if (j == i) {
if (S.charAt(i) == c) dp[k][i][j] = 1;
else dp[k][i][j] = 0;
} else { // j > i i是倒着循环的所有是i+1(上一个)
if (S.charAt(i) != c) dp[k][i][j] = dp[k][i+1][j];
// j是正着循环的,所以是j-1
else if (S.charAt(j) != c) dp[k][i][j] = dp[k][i][j-1];
else { // S[i] == S[j] == c
//如果是两个的话,就是两种i+1和j
if (j == i+1) dp[k][i][j] = 2; // "aa" : {"a", "aa"}
else { // length is > 2
dp[k][i][j] = 2;
for (int m = 0; m < 4; ++m) { // 既然相等即可每一次都算一种
dp[k][i][j] += dp[m][i+1][j-1];
dp[k][i][j] %= mod;
}
}
}
}
}
}
} int ans = 0;
for (int k = 0; k < 4; ++k) {
ans += dp[k][0][n-1];
ans %= mod;
} return ans;
}
}

PS:

大佬的二维数组的代码

class Solution {
int[][] memo, prv, nxt;
byte[] A;
int MOD = 1_000_000_007; public int countPalindromicSubsequences(String S) {
int N = S.length();
prv = new int[N][4];
nxt = new int[N][4];
memo = new int[N][N];
for (int[] row: prv) Arrays.fill(row, -1);
for (int[] row: nxt) Arrays.fill(row, -1); A = new byte[N];
int ix = 0;
for (char c: S.toCharArray()) {
A[ix++] = (byte) (c - 'a');
} int[] last = new int[4];
Arrays.fill(last, -1);
//前i位离i最近的位置(含有相同的字符)
for (int i = 0; i < N; ++i) {
last[A[i]] = i;
for (int k = 0; k < 4; ++k)
prv[i][k] = last[k];
}
//同理
Arrays.fill(last, -1);
for (int i = N-1; i >= 0; --i) {
last[A[i]] = i;
for (int k = 0; k < 4; ++k)
nxt[i][k] = last[k];
} return dp(0, N-1) - 1;
}
//记忆化搜索
public int dp(int i, int j) {
if (memo[i][j] > 0) return memo[i][j];
int ans = 1;
if (i <= j) {
for (int k = 0; k < 4; ++k) {
int i0 = nxt[i][k];
int j0 = prv[j][k];
//后i位的有相同字符的最近索引大于等于当前索引,自己可以算一个
if (i <= i0 && i0 <= j) ans++;
//k字符出现过,并且后面也出现过
if (-1 < i0 && i0 < j0) ans += dp(i0 + 1, j0 - 1);
if (ans >= MOD) ans -= MOD;
}
}
memo[i][j] = ans;
return ans;
} }

Java实现 LeetCode 730 统计不同回文子字符串(动态规划)的更多相关文章

  1. [LeetCode] 647. Palindromic Substrings 回文子字符串

    Given a string, your task is to count how many palindromic substrings in this string. The substrings ...

  2. [Swift]LeetCode730. 统计不同回文子字符串 | Count Different Palindromic Subsequences

    Given a string S, find the number of different non-empty palindromic subsequences in S, and return t ...

  3. leetcode 730. 统计不同回文子序列(区间dp,字符串)

    题目链接 https://leetcode-cn.com/problems/count-different-palindromic-subsequences/ 题意 给定一个字符串,判断这个字符串中所 ...

  4. [LeetCode] Palindromic Substrings 回文子字符串

    Given a string, your task is to count how many palindromic substrings in this string. The substrings ...

  5. LeetCode Valid Palindrome 有效回文(字符串)

    class Solution { public: bool isPalindrome(string s) { if(s=="") return true; ) return tru ...

  6. Java实现 LeetCode 5 最长回文子串

    5. 最长回文子串 给定一个字符串 s,找到 s 中最长的回文子串.你可以假设 s 的最大长度为 1000. 示例 1: 输入: "babad" 输出: "bab&quo ...

  7. Java实现 LeetCode 516 最长回文子序列

    516. 最长回文子序列 给定一个字符串s,找到其中最长的回文子序列.可以假设s的最大长度为1000. 示例 1: 输入: "bbbab" 输出: 4 一个可能的最长回文子序列为 ...

  8. Java实现 LeetCode 409 最长回文串

    409. 最长回文串 给定一个包含大写字母和小写字母的字符串,找到通过这些字母构造成的最长的回文串. 在构造过程中,请注意区分大小写.比如 "Aa" 不能当做一个回文字符串. 注意 ...

  9. 【LeetCode】最长回文子串【动态规划或中心扩展】

    给定一个字符串 s,找到 s 中最长的回文子串.你可以假设 s 的最大长度为 1000. 示例 1: 输入: "babad"输出: "bab"注意: " ...

随机推荐

  1. python学习第七天--文件系统常用模块os,os.path,pickle

    模块是一个可用代码段的打包,后缀名为py,可被别的程序引入#使用import OS模块:operting system操作系统#import os os.chdir(path) 改变当前工作目录 os ...

  2. python3语法学习第四天--序列

    序列是Python中最基本的数据结构. 序列中的每个元素都分配一个索引从0开始依此类推. Python有6个序列的内置类型,但最常见的是列表和元组. 序列可以的操作:索引,切片,加,乘,检查成员. 此 ...

  3. Mysql 常用函数(8)- concat 函数

    Mysql常用函数的汇总,可看下面系列文章 https://www.cnblogs.com/poloyy/category/1765164.html concat 的作用 连接多个字符串 concat ...

  4. python-修改文件

    1.修改文件1 # fw = open('username','w')# fw.write('hhhh')# fw.flush()  #强制把缓冲区里面的数据写到磁盘上1.简单粗暴直接#  1.打开一 ...

  5. uwsgi模块以参数形式运行项目

    1.虚拟环境中下载uwsgi模块-------pip install uwsgi 2.脚本运行案例 新建一个test.py脚本文件,写入如下内容: def application(env, start ...

  6. 苏浪浪 201771010120 第三周 Java基本程序设计总结

    理论知识: Java有五种语句: (1)方法调用语句(2)表达式语句(3)复合语句(4)控制语句(5)package.import语句 3.8控制流程 3.9大数值 *如果基本的整型和浮点型数据无法达 ...

  7. excel2007灵活计算2个日期之间的工作日

    C1单元格公式:=NETWORKDAYS(A1,B1,$F$2:$F$10)+COUNTIFS($I$2:$I$3,">="&A1,$I$2:$I$3,"& ...

  8. Kubernetes实践踩坑系列(一).应用管理的难题

    应用管理的两大难题  今天我们主要讨论这两个方面的挑战: 对应用研发而言,K8s API 针对简单应用过于复杂,针对复杂应用难以上手: 对应用运维而言,K8s 的扩展能力难以管理:K8s 原生的 AP ...

  9. CentOS下搭建Git服务器

    1.首先需要安装Git,可以使用yum源在线安装: [root@localhost Desktop]# yum install -y git 2.创建一个git用户,用来运行git服务 # addus ...

  10. UPD链接实现稳健传输案例

    使用的类    DatagramSocket  用于发送数据和接收数据    此类的构造方法:        DatagramSocket();        DatagramSocket(端口号); ...