用C语言写一段代码实现给定一个字符串s和两个字符c1,c2,均保证只含有小写字母,求出有多少个s的子串,符合以c1开头,以C2结束。n的范围为1到10的5次方使用并查集

时间: 2023-06-11 09:04:38 浏览: 58
很高兴能为您解答问题!以下是用C语言写一段代码实现给定一个字符串s和两个字符c1,c2,均保证只含有小写字母,求出有多少个s的子串,符合以c1开头,以C2结束,同时使用并查集的代码: ```c #include <stdio.h> #include <string.h> const int MAXN = 1e5 + 5; int fa[MAXN], cnt[MAXN], used[MAXN]; int find(int x) { if (fa[x] == x) return x; return fa[x] = find(fa[x]); } void merge(int x, int y) { x = find(x); y = find(y); if (x != y) { fa[x] = y; cnt[y] += cnt[x]; } } int main() { char s[MAXN], c1, c2; scanf("%s %c %c", s, &c1, &c2); int n = strlen(s); for (int i = 0; i < n; i++) { fa[i] = i; cnt[i] = 1; used[i] = 0; } for (int i = 0; i < n; i++) { if (s[i] == c1) used[i] = 1; else if (i > 0 && used[i - 1]) merge(i - 1, i); } int ans = 0; for (int i = n - 1; i >= 0; i--) { if (s[i] == c2 && used[i]) ans += cnt[find(i)]; else if (i < n - 1 && used[i + 1]) merge(i, i + 1); } printf("%d\n", ans); return 0; } ``` 具体思路: 我们可以先遍历一遍字符串s,并用并查集来维护以字符c1开头的子串,然后再从后往前遍历字符串,用并查集来维护以字符c2结尾的子串,并在遍历过程中计算符合要求的子串个数。 具体来说,我们可以遍历一遍字符串s,用一个数组used来记录字符串中以字符c1开头的位置,然后再遍历一遍字符串s,每次遇到以字符c2结尾的位置i,如果used[i]为1,说明以字符c1开头、以字符c2结尾的子串就是以i结尾、以c1开头的所有子串的总数,我们可以通过并查集来维护这一点。 具体来说,我们可以用一个数组cnt来记录每个集合(即并查集中的一棵树)中元素的个数,然后每次合并两个集合时,我们将cnt[x]的值加到cnt[y]上即可。这样,对于以i结尾、以c1开头的所有子串的总数,我们可以通过查找集合中以i结尾的元素个数得到。 具体实现时,我们可以用并查集来维护以字符c1开头、以字符c2结尾的子串,具体如下: - 初始化并查集:将每个位置i初始化为一个集合,并将其父亲设为i,初始的计数器cnt[i]都为1。 - 对于以字符c1开头的位置i,我们将used[i]设为1,表示这个位置可以作为一个以c1开头的子串,然后如果前一个位置j有used[j]=1,说明当前位置可以和前一个位置构成一个符合要求的子串,我们就把它们合并。 - 从后往前遍历,对于每个位置i,如果s[i]等于c2且used[i]等于1,说明以i结尾、以c1开头的子串是符合要求的子串,我们就累加以i结尾、以c1开头的所有子串的数量cnt[find(i)]。 - 如果s[i]不等于c2,或者used[i]等于0,我们就可以通过查看当前节点和其下一个节点所在的集合,并在需要时将两个集合合并。 最后统计cnt[find(i)]即可。 希望我的回答能解决您的问题。如有疑问,欢迎继续追问!

相关推荐

最新推荐

recommend-type

C语言实现输入一个字符串后打印出该字符串中字符的所有排列

主要介绍了C语言实现输入一个字符串后打印出该字符串中字符的所有排列的方法,是数学中非常实用的排列算法,需要的朋友可以参考下
recommend-type

c++ 连接两个字符串实现代码 实现类似strcat功能

c++ 连接两个字符串实现代码 实现类似strcat功能,需要的朋友可以参考下
recommend-type

C语言之实现字符串小写变大写的实例

主要介绍了C语言之实现字符串小写变大写的实例的相关资料,需要的朋友可以参考下
recommend-type

C语言实现将字符串转换为数字的方法

主要介绍了C语言实现将字符串转换为数字的方法,涉及系统函数atoi()函数的使用技巧,需要的朋友可以参考下
recommend-type

C语言字符串快速压缩算法代码

主要介绍了C语言字符串快速压缩算法代码,将字符串中连续出席的重复字母进行压缩,其主要的压缩字段的格式为”字符重复的次数+字符”。有需要的小伙伴参考下吧。
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

实现实时数据湖架构:Kafka与Hive集成

![实现实时数据湖架构:Kafka与Hive集成](https://img-blog.csdnimg.cn/img_convert/10eb2e6972b3b6086286fc64c0b3ee41.jpeg) # 1. 实时数据湖架构概述** 实时数据湖是一种现代数据管理架构,它允许企业以低延迟的方式收集、存储和处理大量数据。与传统数据仓库不同,实时数据湖不依赖于预先定义的模式,而是采用灵活的架构,可以处理各种数据类型和格式。这种架构为企业提供了以下优势: - **实时洞察:**实时数据湖允许企业访问最新的数据,从而做出更明智的决策。 - **数据民主化:**实时数据湖使各种利益相关者都可
recommend-type

用 Python 画一个可以动的爱心

好的,我可以帮助您使用Python来绘制一个动态的爱心。您可以使用turtle库来实现。以下是实现代码: ```python import turtle import math # 设置画布和画笔 canvas = turtle.Screen() canvas.bgcolor("black") pencil = turtle.Turtle() pencil.speed(0) pencil.color("red", "pink") pencil.pensize(3) # 定义爱心函数 def draw_love(heart_size, x_offset=0, y_offset=0):
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。