字符串的详解-程序员宅基地

技术标签: 算法  数据结构  

一、字符串的定义

1.1字符串的定义

串是由零个或多个字符组成的有限序列,在计算机科学中是一种重要的数据类型。字符可以是字母、数字、符号等,对应于计算机中的编码。串可以用来表示文本、图像、音频等数据

1.2字符串的特点

串是由一系列字符组成的数据结构,具有以下特点:

  • 顺序性:串中的每个字符都有一个确定的位置,按照一定的次序排列。
  • 可变性:串的长度可以根据需要进行改变,可以动态地添加或删除字符。
  • 不可变性:一旦串被创建,其中的字符就不能被修改。
  • 常用操作:串的常用操作包括查找、插入、删除、替换、比较、拼接等。

二、字符串的定义实现与使用

2.1字符串的接口定义

typedef struct
{
	char* data;
	int max;
	int len;
}sstring;

/*初始化串*/
int init(sstring *S, int max);

/*字符拷贝*/
int sstrcpy(sstring *S, char* from);

/*模式匹配*/
int sstrmatch(sstring *S, sstring *sub);

//串比较
int sstrcmp(const sstring* s1, const sstring* s2);

//串长
int sstrlen(sstring* str);

//串的联接 
void sstrcat(sstring* s1, sstring* s2);

2.2字符串的实现

2.2.1字符串初始化
int init(sstring *S, int max)
{
	S->data = (char*)malloc(sizeof(char)*max);

	if(!S->data)
	{
		printf("申请内存失败!1000");
		return 1000;
	}
	S->max = max;
	S->len = 0;
}
2.2.2字符串拷贝
int sstrcpy(sstring *S, char* from)
{
	int i;
	int len = strlen(from);

	if(len>S->max)
	{
		printf("超出了字符串S的长度!1001\n");
		return 1001;
	}
	for(i=0;i<len;i++)
	{
		S->data[i] = from[i];
	}
	S->data[i] = '\0';
	S->len = len;
	return 0;
}
2.2.3字符串模式匹配
int sstrmatch(sstring *S, sstring *sub)
{
	int i, j, k;
	int m = S->len;
	int n = sub->len;

	if(S->len==0||sub->len==0){
		printf("字符串为空");
		return;
	} 

	for(i=0;i<=m-n;i++)
	{
		j=0;
		k=i;
		while(j<n && S->data[k] == sub->data[j])
		{
			j++;
			k++;
		}
		if(j==n)
		{
			return i+1;
		}
	}
	return -1;
}
2.2.4字符串比较
int sstrcmp(const sstring* s1, const sstring* s2)
{
    int i;
    int result;

    // 比较两个字符串的长度
    if (s1->len < s2->len) {
        return -1;
    } else if (s1->len > s2->len) {
        return 1;
    }

    // 长度相等,比较每个字符
    for (i = 0; i < s1->len; i++) {
        if (s1->data[i] < s2->data[i]) {
            return -1;
        } else if (s1->data[i] > s2->data[i]) {
            return 1;
        }
    }

    // 字符串相等
    return 0;
}
2.2.5字符串长度
int sstrlen(sstring* str) {
    return str->len;
}
2.2.5字符串拼接
void sstrcat(sstring* s1, sstring* s2) {
    if (s1->len + s2->len > s1->max) {
        s1->data = (char*)realloc(s1->data, (s1->len + s2->len + 1) * sizeof(char));
        s1->max = s1->len + s2->len + 1;
    }
    strcat(s1->data + s1->len, s2->data);
    s1->len += s2->len;
}

2.3字符串的使用

#include <stdio.h>
#include "sstring.h"
#include <stdlib.h>
#include "welcome.h"

int main(int argc, char* argv[])
{
	
	sstring S;
	sstring sub;
	int index;
	char str1[100];
	char str2[100];
	int cmd,i,m,n;

	int result;
	for(i=0;i<strlen(welcome);i++)
		{
			printf("%c",welcome[i]);
			for(m=0;m<1000;m++)
				for(n=0;n<1000;n++)
				{
					;
				}
		}

	printf("\n\n\n");

	do
	{	
		printf("-----------字符串演示-----------\n");
		printf(" 1. 初始化\n");
		printf(" 2. 输入串\n");
		printf(" 3. 从字符常量拷贝\n");
		printf(" 4. 模式匹配?\n");
		printf(" 5. 串比较\n");
		printf(" 6. 串长\n");
		printf(" 7. 输出串\n");
		printf(" 8. 串的拼接\n");
		printf(" 9. 帮助\n");
		printf(" 0. 退出\n");
		printf(" 请选择(0~9):");
		scanf("%d",&cmd);
		switch(cmd){
		case 1:
			init(&S, 1000);
			init(&sub, 100);
			printf("初始化成功!\n");
			break;
		case 2:
			printf("请输入主串:");
			scanf("%s", str1);
			printf("请输入子串:");
			scanf("%s", str2);
			break;
		case 3:
			sstrcpy(&S, &str1);
			sstrcpy(&sub, &str2);
			printf("从字符复制完成!\n");
			break;
		case 4:
			index = sstrmatch(&S,&sub);
			if(index>=0)
			{
				printf("匹配成功,子串在主串的%d位置\n", index); 
			}
			else
			{
				printf("子串不在主串种!\n");
			}
			break;
		case 5:
			if(sstrcmp(&S,&sub) == 1){
				printf("主串的长度大于子串的长度\n");
			}else if(sstrcmp(&S,&sub) == -1){
				printf("主串的长度小于子串的长度\n");
			}
			else if(sstrcmp(&S,&sub) == 0){
				printf("两个字符串相等\n");
			}
			break;
		case 6:
			printf("主串:%s的长度为%d\n",str1,sstrlen(&S));
			printf("子串:%s的长度为%d\n",str2,sstrlen(&sub));
			break;
		case 7:
		    printf("主串打印:%s",S.data);
		    printf("子串打印:%s\n",sub.data);
		    break;
		case 8:
		    sstrcat(&S, &sub);
		    printf("主串和子串的拼接:%s\n",S.data);	
		    break;
		case 9:
		    printf("本程序为字符串的演示程序,由黄彬森设计开发!\n");
		    break;
		}
		
	}while(cmd!=0);
}

三、完整代码

3.1 main.c文件代码

#include <stdio.h>
#include "sstring.h"
#include <stdlib.h>
#include "welcome.h"

int main(int argc, char* argv[])
{
	
	sstring S;
	sstring sub;
	int index;
	char str1[100];
	char str2[100];
	int cmd,i,m,n;

	int result;
	for(i=0;i<strlen(welcome);i++)
		{
			printf("%c",welcome[i]);
			for(m=0;m<1000;m++)
				for(n=0;n<1000;n++)
				{
					;
				}
		}

	printf("\n\n\n");

	do
	{	
		printf("-----------字符串演示-----------\n");
		printf(" 1. 初始化\n");
		printf(" 2. 输入串\n");
		printf(" 3. 从字符常量拷贝\n");
		printf(" 4. 模式匹配?\n");
		printf(" 5. 串比较\n");
		printf(" 6. 串长\n");
		printf(" 7. 输出串\n");
		printf(" 8. 串的拼接\n");
		printf(" 9. 帮助\n");
		printf(" 0. 退出\n");
		printf(" 请选择(0~9):");
		scanf("%d",&cmd);
		switch(cmd){
		case 1:
			init(&S, 1000);
			init(&sub, 100);
			printf("初始化成功!\n");
			break;
		case 2:
			printf("请输入主串:");
			scanf("%s", str1);
			printf("请输入子串:");
			scanf("%s", str2);
			break;
		case 3:
			sstrcpy(&S, &str1);
			sstrcpy(&sub, &str2);
			printf("从字符复制完成!\n");
			break;
		case 4:
			index = sstrmatch(&S,&sub);
			if(index>=0)
			{
				printf("匹配成功,子串在主串的%d位置\n", index); 
			}
			else
			{
				printf("子串不在主串种!\n");
			}
			break;
		case 5:
			if(sstrcmp(&S,&sub) == 1){
				printf("主串的长度大于子串的长度\n");
			}else if(sstrcmp(&S,&sub) == -1){
				printf("主串的长度小于子串的长度\n");
			}
			else if(sstrcmp(&S,&sub) == 0){
				printf("两个字符串相等\n");
			}
			break;
		case 6:
			printf("主串:%s的长度为%d\n",str1,sstrlen(&S));
			printf("子串:%s的长度为%d\n",str2,sstrlen(&sub));
			break;
		case 7:
		    printf("主串打印:%s",S.data);
		    printf("子串打印:%s\n",sub.data);
		    break;
		case 8:
		    sstrcat(&S, &sub);
		    printf("主串和子串的拼接:%s\n",S.data);	
		    break;
		case 9:
		    printf("本程序为字符串的演示程序,由黄彬森设计开发!\n");
		    break;
		}
		
	}while(cmd!=0);
}

3.2  sstring.c文件代码

#include "sstring.h"


/*初始化串*/
int init(sstring *S, int max)
{
	S->data = (char*)malloc(sizeof(char)*max);

	if(!S->data)
	{
		printf("申请内存失败!1000");
		return 1000;
	}
	S->max = max;
	S->len = 0;
}


/*字符拷贝*/
int sstrcpy(sstring *S, char* from)
{
	int i;
	int len = strlen(from);

	if(len>S->max)
	{
		printf("超出了字符串S的长度!1001\n");
		return 1001;
	}
	for(i=0;i<len;i++)
	{
		S->data[i] = from[i];
	}
	S->data[i] = '\0';
	S->len = len;
	return 0;
}


/*模式匹配*/
int sstrmatch(sstring *S, sstring *sub)
{
	int i, j, k;
	int m = S->len;
	int n = sub->len;

	if(S->len==0||sub->len==0){
		printf("字符串为空");
		return;
	} 

	for(i=0;i<=m-n;i++)
	{
		j=0;
		k=i;
		while(j<n && S->data[k] == sub->data[j])
		{
			j++;
			k++;
		}
		if(j==n)
		{
			return i+1;
		}
	}
	return -1;
}


//串比较
int strcmp(const sstring* s1, const sstring* s2)
{
    int i;
    int result;

    // 比较两个字符串的长度
    if (s1->len < s2->len) {
        return -1;
    } else if (s1->len > s2->len) {
        return 1;
    }

    // 长度相等,比较每个字符
    for (i = 0; i < s1->len; i++) {
        if (s1->data[i] < s2->data[i]) {
            return -1;
        } else if (s1->data[i] > s2->data[i]) {
            return 1;
        }
    }

    // 字符串相等
    return 0;
}

//串长
int sstrlen(sstring* str) {
    return str->len;
}

//串的拼接
void sstrcat(sstring* s1, sstring* s2) {
    if (s1->len + s2->len > s1->max) {
        s1->data = (char*)realloc(s1->data, (s1->len + s2->len + 1) * sizeof(char));
        s1->max = s1->len + s2->len + 1;
    }
    strcat(s1->data + s1->len, s2->data);
    s1->len += s2->len;
}

3.2  sstring.h文件代码

typedef struct
{
	char* data;
	int max;
	int len;
}sstring;

/*初始化串*/
int init(sstring *S, int max);

/*字符拷贝*/
int sstrcpy(sstring *S, char* from);

/*模式匹配*/
int sstrmatch(sstring *S, sstring *sub);

//串比较
int sstrcmp(const sstring* s1, const sstring* s2);

//串长
int sstrlen(sstring* str);

//串的联接 
void sstrcat(sstring* s1, sstring* s2);

3.4  welcome.h文件代码

char welcome[] = "\n\
|     .-.\n\ 
|    /   、         .-.\n\
|   /     、       /   、       .-.     .-.\n\
+--/-------、-----/-----、-----/---、---/---、---/-、-/-、/、/---\n\
| /         、   /       、   /     '-'     '-'              \n\
|/           '-'         '-'                               \n\
";

四、代码运行结果


|     .-.
|    /   、         .-.
|   /     、       /   、       .-.     .-.
+--/-------、-----/-----、-----/---、---/---、---/-、-/-、/、/---
| /         、   /       、   /     '-'     '-'
|/           '-'         '-'



-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):1
初始化成功!
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):2
请输入主串:aaaa
请输入子串:bbb
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):3
从字符复制完成!
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):5
主串的长度大于子串的长度
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):6
主串:aaaa的长度为4
子串:bbb的长度为3
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):8
主串和子串的拼接:aaaabbb
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):7
主串打印:aaaabbb子串打印:bbb
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):4
匹配成功,子串在主串的5位置
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):9
本程序为字符串的演示程序,由黄彬森设计开发!
-----------字符串演示-----------
 1. 初始化
 2. 输入串
 3. 从字符常量拷贝
 4. 模式匹配?
 5. 串比较
 6. 串长
 7. 输出串
 8. 串的拼接
 9. 帮助
 0. 退出
 请选择(0~9):0

--------------------------------
Process exited after 59.68 seconds with return value 0
请按任意键继续. . .

五、小结

字符串是由0个或者多个字符组成的有限序列,一般用来表示文本,图像等的数据。

字符串基本操作如:

        串的长度:串中字符的个数,常用len(s)表示。

        串的比较:可以按字典序比较两个串的大小,常用==!=<><=>=等运算符进行比较。

        串的子串:串中任意连续的一段字符,可以通过s[i:j]表示,其中i是子串的起始位置,j是子串的终止位置+1。

        串的连接:将两个串连接在一起形成一个新的串,常用s1+s2表示。

        串的复制:将一个串复制n次形成一个新的串,常用s*n表示。

        串的查找:在一个串中查找某个子串的位置。

六、参考文献

【1】CSDN

【2】数据结构(C语言)

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_54938554/article/details/134153105

智能推荐

分布式光纤传感器的全球与中国市场2022-2028年:技术、参与者、趋势、市场规模及占有率研究报告_预计2026年中国分布式传感器市场规模有多大-程序员宅基地

文章浏览阅读3.2k次。本文研究全球与中国市场分布式光纤传感器的发展现状及未来发展趋势,分别从生产和消费的角度分析分布式光纤传感器的主要生产地区、主要消费地区以及主要的生产商。重点分析全球与中国市场的主要厂商产品特点、产品规格、不同规格产品的价格、产量、产值及全球和中国市场主要生产商的市场份额。主要生产商包括:FISO TechnologiesBrugg KabelSensor HighwayOmnisensAFL GlobalQinetiQ GroupLockheed MartinOSENSA Innovati_预计2026年中国分布式传感器市场规模有多大

07_08 常用组合逻辑电路结构——为IC设计的延时估计铺垫_基4布斯算法代码-程序员宅基地

文章浏览阅读1.1k次,点赞2次,收藏12次。常用组合逻辑电路结构——为IC设计的延时估计铺垫学习目的:估计模块间的delay,确保写的代码的timing 综合能给到多少HZ,以满足需求!_基4布斯算法代码

OpenAI Manager助手(基于SpringBoot和Vue)_chatgpt网页版-程序员宅基地

文章浏览阅读3.3k次,点赞3次,收藏5次。OpenAI Manager助手(基于SpringBoot和Vue)_chatgpt网页版

关于美国计算机奥赛USACO,你想知道的都在这_usaco可以多次提交吗-程序员宅基地

文章浏览阅读2.2k次。USACO自1992年举办,到目前为止已经举办了27届,目的是为了帮助美国信息学国家队选拔IOI的队员,目前逐渐发展为全球热门的线上赛事,成为美国大学申请条件下,含金量相当高的官方竞赛。USACO的比赛成绩可以助力计算机专业留学,越来越多的学生进入了康奈尔,麻省理工,普林斯顿,哈佛和耶鲁等大学,这些同学的共同点是他们都参加了美国计算机科学竞赛(USACO),并且取得过非常好的成绩。适合参赛人群USACO适合国内在读学生有意向申请美国大学的或者想锻炼自己编程能力的同学,高三学生也可以参加12月的第_usaco可以多次提交吗

MySQL存储过程和自定义函数_mysql自定义函数和存储过程-程序员宅基地

文章浏览阅读394次。1.1 存储程序1.2 创建存储过程1.3 创建自定义函数1.3.1 示例1.4 自定义函数和存储过程的区别1.5 变量的使用1.6 定义条件和处理程序1.6.1 定义条件1.6.1.1 示例1.6.2 定义处理程序1.6.2.1 示例1.7 光标的使用1.7.1 声明光标1.7.2 打开光标1.7.3 使用光标1.7.4 关闭光标1.8 流程控制的使用1.8.1 IF语句1.8.2 CASE语句1.8.3 LOOP语句1.8.4 LEAVE语句1.8.5 ITERATE语句1.8.6 REPEAT语句。_mysql自定义函数和存储过程

半导体基础知识与PN结_本征半导体电流为0-程序员宅基地

文章浏览阅读188次。半导体二极管——集成电路最小组成单元。_本征半导体电流为0

随便推点

【Unity3d Shader】水面和岩浆效果_unity 岩浆shader-程序员宅基地

文章浏览阅读2.8k次,点赞3次,收藏18次。游戏水面特效实现方式太多。咱们这边介绍的是一最简单的UV动画(无顶点位移),整个mesh由4个顶点构成。实现了水面效果(左图),不动代码稍微修改下参数和贴图可以实现岩浆效果(右图)。有要思路是1,uv按时间去做正弦波移动2,在1的基础上加个凹凸图混合uv3,在1、2的基础上加个水流方向4,加上对雾效的支持,如没必要请自行删除雾效代码(把包含fog的几行代码删除)S..._unity 岩浆shader

广义线性模型——Logistic回归模型(1)_广义线性回归模型-程序员宅基地

文章浏览阅读5k次。广义线性模型是线性模型的扩展,它通过连接函数建立响应变量的数学期望值与线性组合的预测变量之间的关系。广义线性模型拟合的形式为:其中g(μY)是条件均值的函数(称为连接函数)。另外,你可放松Y为正态分布的假设,改为Y 服从指数分布族中的一种分布即可。设定好连接函数和概率分布后,便可以通过最大似然估计的多次迭代推导出各参数值。在大部分情况下,线性模型就可以通过一系列连续型或类别型预测变量来预测正态分布的响应变量的工作。但是,有时候我们要进行非正态因变量的分析,例如:(1)类别型.._广义线性回归模型

HTML+CSS大作业 环境网页设计与实现(垃圾分类) web前端开发技术 web课程设计 网页规划与设计_垃圾分类网页设计目标怎么写-程序员宅基地

文章浏览阅读69次。环境保护、 保护地球、 校园环保、垃圾分类、绿色家园、等网站的设计与制作。 总结了一些学生网页制作的经验:一般的网页需要融入以下知识点:div+css布局、浮动、定位、高级css、表格、表单及验证、js轮播图、音频 视频 Flash的应用、ul li、下拉导航栏、鼠标划过效果等知识点,网页的风格主题也很全面:如爱好、风景、校园、美食、动漫、游戏、咖啡、音乐、家乡、电影、名人、商城以及个人主页等主题,学生、新手可参考下方页面的布局和设计和HTML源码(有用点赞△) 一套A+的网_垃圾分类网页设计目标怎么写

C# .Net 发布后,把dll全部放在一个文件夹中,让软件目录更整洁_.net dll 全局目录-程序员宅基地

文章浏览阅读614次,点赞7次,收藏11次。之前找到一个修改 exe 中 DLL地址 的方法, 不太好使,虽然能正确启动, 但无法改变 exe 的工作目录,这就影响了.Net 中很多获取 exe 执行目录来拼接的地址 ( 相对路径 ),比如 wwwroot 和 代码中相对目录还有一些复制到目录的普通文件 等等,它们的地址都会指向原来 exe 的目录, 而不是自定义的 “lib” 目录,根本原因就是没有修改 exe 的工作目录这次来搞一个启动程序,把 .net 的所有东西都放在一个文件夹,在文件夹同级的目录制作一个 exe._.net dll 全局目录

BRIEF特征点描述算法_breif description calculation 特征点-程序员宅基地

文章浏览阅读1.5k次。本文为转载,原博客地址:http://blog.csdn.net/hujingshuang/article/details/46910259简介 BRIEF是2010年的一篇名为《BRIEF:Binary Robust Independent Elementary Features》的文章中提出,BRIEF是对已检测到的特征点进行描述,它是一种二进制编码的描述子,摈弃了利用区域灰度..._breif description calculation 特征点

房屋租赁管理系统的设计和实现,SpringBoot计算机毕业设计论文_基于spring boot的房屋租赁系统论文-程序员宅基地

文章浏览阅读4.1k次,点赞21次,收藏79次。本文是《基于SpringBoot的房屋租赁管理系统》的配套原创说明文档,可以给应届毕业生提供格式撰写参考,也可以给开发类似系统的朋友们提供功能业务设计思路。_基于spring boot的房屋租赁系统论文