【滑动窗口】LeetCode2953:统计完全子字符串

作者推荐

[二分查找]LeetCode2040:两个有序数组的第 K 小乘积

本题其它解法

【离散差分】LeetCode2953:统计完全子字符串

题目

给你一个字符串 word 和一个整数 k 。
如果 word 的一个子字符串 s 满足以下条件,我们称它是 完全字符串:
s 中每个字符 恰好 出现 k 次。
相邻字符在字母表中的顺序 至多 相差 2 。也就是说,s 中两个相邻字符 c1 和 c2 ,它们在字母表中的位置相差 至多 为 2 。
请你返回 word 中 完全 子字符串的数目。
子字符串 指的是一个字符串中一段连续 非空 的字符序列。
示例 1:
输入:word = “igigee”, k = 2
输出:3
解释:完全子字符串需要满足每个字符恰好出现 2 次,且相邻字符相差至多为 2 :igigee, igigee, igigee 。
示例 2:
输入:word = “aaabbbccc”, k = 3
输出:6
解释:完全子字符串需要满足每个字符恰好出现 3 次,且相邻字符相差至多为 2 :aaabbbccc, aaabbbccc, aaabbbccc, aaabbbccc, aaabbbccc, aaabbbccc 。
参数范围
1 <= word.length <= 105
word 只包含小写英文字母。
1 <= k <= word.length

滑动窗口

周赛的时候,忽略了:完全子字符串的长度是k的1到26倍。

时间复杂度

O(nmm)。n是字符串的长度,m的字符种类,也就是26。枚举字符结尾:O(n);枚举窗口长度:O(m);统计合法字符数量O(m)。

变量解析

m_aCharNumsiWindowWidth个字符,各字母的数量

代码

核心代码

class Solution {
public:
int countCompleteSubstrings(string word, int k) {
m_iK = k;
int pre = 0;
int iRet = 0;
for (int i = 0; i < word.length(); i++)
{
if (i && (abs(word[i] - word[i - 1]) > 2))
{
iRet += Do(word.data()+pre, i - pre);
pre = i;
}
}
iRet += Do(word.data() + pre, word.length() - pre);
return iRet;
}
int Do(const char* p ,int len)
{
int iRet = 0;
auto AddNum = &
{
for (int i = 0; i < 26; i++)
{
if ((0 != m_aCharNums[i]) && (m_iK != m_aCharNums[i]))
{
return;
}
}
iRet++;
};
int iWindowWidth = 0;
for (int i = 1; (i <= 26)&&((iWindowWidth=m_iK*i)<= len); i++)
{
memset(m_aCharNums, 0, sizeof(m_aCharNums));
int j = 0;
for (; j < iWindowWidth; j++)
{
m_aCharNums[p[j] - ‘a’]++;
}
AddNum();
for (;j < len; j++)
{
m_aCharNums[p[j] - ‘a’]++;
m_aCharNums[p[j - iWindowWidth] - ‘a’]–;
AddNum();
}
}
return iRet;
}
int m_aCharNums[26];
int m_iK;
};

测试用例

template
void Assert(const vector& v1, const vector& v2)
{
if (v1.size() != v2.size())
{
assert(false);
return;
}
for (int i = 0; i < v1.size(); i++)
{
assert(v1[i] == v2[i]);
}
}

template
void Assert(const T& t1, const T& t2)
{
assert(t1 == t2);
}

int main()
{
string s;
int k, res;
{
Solution slu;
s = “ba”;
k = 1;
auto res = slu.countCompleteSubstrings(s, k);
Assert(3, res);
}
{
Solution slu;
s = “gvgvvgv”;
k = 2;
auto res = slu.countCompleteSubstrings(s, k);
Assert(1, res);
}
{
Solution slu;
s = “igigee”;
k = 2;
auto res = slu.countCompleteSubstrings(s, k);
Assert(3, res);
}
{
Solution slu;
s = “aaabbbccc”;
k = 3;
auto res = slu.countCompleteSubstrings(s, k);
Assert(6, res);
}

//CConsole::Out(res);

}

优化

不用每次都判断无法字符的数量,只会修改两个字符的数量,只判断这两个字符就可以了。

时间复杂度

O(nm)。

代码

class Solution {
public:
	int countCompleteSubstrings(string word, int k) {
		m_iK = k;
		int pre = 0;
		int iRet = 0;
		for (int i = 0; i < word.length(); i++)
		{
			if (i && (abs(word[i] - word[i - 1]) > 2))
			{
				iRet += Do(word.data()+pre, i - pre);
				pre = i;
			}
		}
		iRet += Do(word.data() + pre, word.length() - pre);
		return iRet;
	}
	int Do(const char* p ,int len)
	{
		int iRet = 0;		
		int iWindowWidth = 0;
		for (int i = 1; (i <= 26)&&((iWindowWidth=m_iK*i)<= len); i++)
		{
			memset(m_aCharNums,  0, sizeof(m_aCharNums));
			int j = 0;
			for (; j < iWindowWidth; j++)
			{
				m_aCharNums[p[j] - 'a']++;
			}
			int iVilidCount = std::count(m_aCharNums, m_aCharNums + 26, 0) + std::count(m_aCharNums, m_aCharNums + 26, m_iK);
			if (26 == iVilidCount)
			{
				iRet++;
			}
			auto Change = [&](int index, int iAdd)
			{
				bool bOld = (0 == m_aCharNums[index]) || (m_iK == m_aCharNums[index]);
				m_aCharNums[index] += iAdd;
				bool bNew = (0 == m_aCharNums[index]) || (m_iK == m_aCharNums[index]);
				iVilidCount += (int)bNew - (int)bOld;
			};
			for (;j < len; j++)
			{
				Change(p[j] - 'a',1);
				Change(p[j - iWindowWidth] - 'a', -1);
				iRet += (26 == iVilidCount);
			}
		}
		return iRet;
	}
	int m_aCharNums[26];
	int m_iK;
};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业

。也就是我们常说的专业的人做专业的事。 |
|如果程序是一条龙,那算法就是他的是睛|

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境:

VS2022 C++17


http://www.niftyadmin.cn/n/5239190.html

相关文章

人员备岗与能力备份

去成为那种具备能力备份的人&#xff0c;不简单是人员备岗。 11月面临多名人员请假&#xff0c;人员请假过程中不停的问自己一个问题&#xff1f;是否做好了备份。 1. 人员备份 当一个人A离开工作岗位&#xff0c;需要找B继续接上去推广系统。 实际情况是抽调B过来&#xf…

出现数据库出现没有时间格式的错误,实体类Date类型不对导致时间报错

目录 报错现场解决办法java与mysql中的日期类型及二者的对应关系和使用场景 报错现场 数据库最早时间为2023年1月1日&#xff0c;前端查询后却出现2022年12月31日的数据 数据库时间类型为date swagger接口测试 解决办法 讲until的Date改成sql类的Date&#xff0c;就可以了…

express中配置swagger并配置token信息

express中配置swagger并配置token信息 1. 安装swagger-jsdoc cnpm install swagger-jsdoc1.3.0 --save2. 在项目根目录下的config目录下新建swagger.js文件并添加配置项 // 引入swagger const swaggerJSDoc require(swagger-jsdoc) // swagger定义 const swaggerDefinitio…

从无人驾驶汽车到虚拟助手:人工智能如何改变我们的世界

人工智能对我们的生活影响有多大 近年来&#xff0c;人工智能迅速发展&#xff0c;成为影响社会各个领域的重要技术。本文将深入探讨人工智能在无人驾驶汽车、虚拟助手等领域的应用和影响&#xff0c;剖析人工智能对我们的生活、工作和社交等方面所带来的深刻变革。 目录 人…

Esxi8.0ip不够用,虚拟网卡实现虚拟机局域网互通

ps:虽然是蛮简单的一个问题&#xff0c;但是自己都有需求了&#xff0c;就分享出来让有需要的人少走点弯路吧。 需求 甲方机房搬迁后只给两个ip&#xff0c;一个运维机的ip&#xff0c;一个网关ip。 运维机是ESXI8.0系统&#xff0c;之前有点强迫症&#xff0c;非要进行业务隔…

CSS新手入门笔记整理:CSS超链接样式

超链接伪类 超链接在鼠标单击的不同时期的样式是不一样的。 默认情况下&#xff1a;字体为蓝色&#xff0c;带有下划线。鼠标单击时&#xff1a;字体为红色&#xff0c;带有下划线。鼠标单击后&#xff1a;字体为紫色&#xff0c;带有下划线。 超链接伪类简介 在CSS中&…

Meta推出了一套开源AI语言翻译模型,这些模型不仅能保留说话的表达方式,还能提升流式翻译的效果

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

JDK、JRE、JVM、SE、EE、ME的区别

一、JDK Java Development Kit&#xff0c;Java 软件开发工具包。JDK是提供给Java开发人员使用的&#xff0c;包含了JRE和一些Java开发工具&#xff0c;如编译工具&#xff08;javac.exe&#xff09;、打包工具&#xff08;jar.exe&#xff09;等。所以安装了JDK就不需…