串(字符串)是编程中最常用的结构,但 C语言 中没有“字符串”这种变量,只能通过字符数组的形式表示字符串。

  C语言 为我们提供了一个 string.h 的头文件,通过这个头文件,我们可以实现对字符串的各种操作,如拷贝、比较等,具体用法请参考【C语言库函数】

  当然,我们对字符串的操作可能不仅仅局限于 string.h 这个头文件给我们提供的这些方法,因此,我们可以自己实现一个字符串的数据结构,通过在这里面编写一些实用的方法,实现我们的需求。

  以下是实用 C语言 编写的一个字符串的头文件 String.h,代码如下:

/**
* 串(顺序存储方式)
* 注意:字符串都是以“\0”符号结尾的
*/
#include <Constant.h> // 定义字符串的数据结构体
typedef struct String {
char* data; // 字符串中的数据
int currLength; // 字符串的当前长度
int maxLength; // 字符串的总长度
} String; // 0.获取某个字符串的长度
int getCharArrayLength(char* string) {
int length = ;
while(string[length] != '\0') {
length++;
}
return length;
} // 1.初始化字符串(创建一个新的字符串,其中包含string中的所有字符)
Status initString(String* S, char* string) {
int i = ;
// 获取字符串的长度
int length = getCharArrayLength(string);
// 字符串赋值
S->data = (char*)malloc(length * sizeof(char));
if(S->data == NULL) {
printf("initString => 空间分配失败,初始化字符串失败!\n");
return FAILURE;
}
S->currLength = length;
S->maxLength = length;
for(i = ; i < length; i++) {
S->data[i] = string[i];
}
return SUCCESS;
} // 2.复制字符串(将字符串string中的所有字符复制到字符串S中)
Status copyString(String* S, char* string) {
int i;
int length = getCharArrayLength(string);
if(S->data == NULL) {
printf("copyString => 字符串不存在,复制失败!\n");
return FAILURE;
}
if(length > S->maxLength) {
S->data = (char*)realloc(S->data, length * sizeof(char));
if(S->data == NULL) {
printf("copyString => 重分配空间失败,复制字符串失败!\n");
return FAILURE;
}
S->maxLength = length;
}
S->currLength = length;
for(i = ; i < length; i++) {
S->data[i] = string[i];
}
return SUCCESS;
} // 3.判断字符串是否为空
Status isStringEmpty(String* S) {
if(S->data == NULL) {
printf("isStringEmpty => 字符串不存在!\n");
exit();
}
if(S->currLength == ) {
return TRUE;
}
return FALSE;
} // 4.比较两个字符串的大小(返回的是S1与S2比较的结果)
// 当两个字符串的长度相等,且对应字符都相同时,称这两个字符串相等;否则,看第一个不相等的字符比较结果,字符较大的字符串较大
Status compareString(String* S1, String* S2) {
int i = ;
// 判空
if(S1->data == NULL || S2->data == NULL) {
printf("compareString => 其中一个字符串不存在!\n");
exit();
}
// 某一个字符串为空
if(S1->currLength == ) {
if(S2->currLength == ) {
return EQUAL;
} else {
return SMALLER;
}
}
if(S2->currLength == ) {
if(S1->currLength == ) {
return EQUAL;
} else {
return BIGGER;
}
}
// 两个字符串都不为空时,逐个字符比较
for(i = ; ;i++) {
if(i == S1->currLength && i == S2->currLength) {
return EQUAL;
}
if(i >= S1->currLength) {
return SMALLER;
}
if(i >= S2->currLength) {
return BIGGER;
}
if(S1->data[i] > S2->data[i]) {
return BIGGER;
} else if(S1->data[i] < S2->data[i]) {
return SMALLER;
}
}
} // 5.获取字符串的长度
int getStringLength(String* S) {
if(S->data == NULL) {
printf("getStringLength => 字符串不存在!\n");
exit();
}
return S->currLength;
} // 6.清空字符串
Status clearString(String* S) {
if(S->data == NULL) {
printf("clearString => 字符串不存在!\n");
return FAILURE;
}
S->currLength = ;
return SUCCESS;
} // 7.将字符串S2连接到字符串S1后面并返回
Status concatString(String* S1, String* S2) {
if(S1->data == NULL || S2->data == NULL) {
printf("concatString => 其中一个字符串不存在!\n");
return FAILURE;
}
int i;
int len1 = getStringLength(S1);
int len2 = getStringLength(S2);
if(S1->maxLength < len1 + len2) {
S1->data = (char*)realloc(S1->data, (len1 + len2) * sizeof(char));
if(S1->data == NULL) {
printf("concatString => 重分配空间失败,字符串拼接失败!\n");
return FAILURE;
}
S1->maxLength = len1 + len2;
}
for(i = ; i < len2; i++) {
S1->data[len1 + i] = S2->data[i];
}
S1->currLength = len1 + len2;
return SUCCESS;
} // 8.返回字符串S中从pos位置开始,长度为len的子串
char* getSubString(String* S, int pos, int len) {
char* result;
int i;
if(S->data == NULL) {
printf("getSubString => 字符串不存在!\n");
exit();
}
if(pos < || pos >= S->currLength) {
printf("getSubString => pos参数超出范围!\n");
exit();
}
if(len > S->currLength - pos) {
printf("getSubString => 子串长度超出范围!\n");
exit();
}
for(i = ; i < len; i++) {
*(result + i) = S->data[pos + i];
}
*(result + i) = '\0';
return result;
} // 9.返回字符串S中从pos位置开始的与子串string相等的第一个子串的位置
int locateSubString(String* S, char* string, int pos) {
int i, j;
int length = getCharArrayLength(string);
if(S->data == NULL) {
printf("locateSubString => 字符串不存在!\n");
exit();
}
if(pos < || pos >= S->currLength) {
printf("locateSubString => pos参数超出范围!");
exit();
}
if(length + pos > S->currLength) {
printf("locateSubString => 子串长度超出范围!\n");
exit();
}
for(i = pos; i <= S->currLength - length; i++) {
for(j = ; j < length; j++) {
if(S->data[i + j] != string[j]) {
break;
}
}
if(j == length) {
return i;
}
}
return -;
} // 10.在字符串S的pos位置插入字符串string
Status stringInsert(String* S, int pos, char* string) {
int i;
int length = getCharArrayLength(string);
if(S->data == NULL) {
printf("stringInsert => 字符串不存在,插入字符失败!\n");
return FAILURE;
}
if(pos < || pos > S->currLength) {
printf("stringInsert => pos参数超出范围,插入字符失败!\n");
return FAILURE;
}
if(S->currLength + length > S->maxLength) {
S->data = (char*)realloc(S->data, (S->currLength + length) * sizeof(char));
if(S->data == NULL) {
printf("stringInsert => 重分配空间失败,插入字符失败!\n");
return FAILURE;
}
S->maxLength = S->currLength + length;
}
for(i = S->currLength - ; i >= pos; i--) {
S->data[i + length] = S->data[i];
}
for(i = ; i < length; i++) {
S->data[pos + i] = string[i];
}
S->currLength += length;
return SUCCESS;
} // 11.删除字符串S中从pos位置开始的len个字符
Status stringDelete(String* S, int pos, int len) {
int i;
if(S->data == NULL) {
printf("stringDelete => 字符串不存在,删除字符失败!\n");
return FAILURE;
}
if(pos < || pos >= S->currLength) {
printf("stringDelete => pos参数超出范围,删除字符失败!\n");
return FAILURE;
}
if(pos + len > S->currLength) {
printf("stringDelete => 子串长度超出范围,删除字符失败!\n");
return FAILURE;
}
for(i = pos + len; i < S->currLength; i++) {
S->data[i - len] = S->data[i];
}
S->currLength -= len;
return SUCCESS;
} // 12.用字符串newStr替换字符串S中出现的所有与子串oldStr相同的不重叠的子串
Status replaceString(String* S, char* oldStr, char* newStr) {
int index;
int oldLen = getCharArrayLength(oldStr);
int newLen = getCharArrayLength(newStr);
if(S->data == NULL) {
printf("replaceString => 字符串不存在,替换失败!\n");
return FAILURE;
}
index = locateSubString(S, oldStr, );
while(index >= && index + oldLen <= S->currLength) {
stringDelete(S, index, oldLen);
stringInsert(S, index, newStr);
if(oldLen + index + newLen >= S->currLength) {
break;
}
index = locateSubString(S, oldStr, index + newLen);
}
return SUCCESS;
} // 13.遍历字符串
void traverseString(String* S) {
int i;
if(S->data == NULL) {
printf("traverseString => 字符串不存在,遍历失败!\n");
exit();
}
printf("遍历字符串:");
for(i = ; i < S->currLength; i++) {
printf("%c", S->data[i]);
}
printf("\n");
} // 14.销毁字符串
Status destroyString(String* S) {
if(S->data == NULL) {
printf("destroyString => 字符串不存在,不需要销毁!\n");
return FAILURE;
}
free(S->data);
S->data = NULL;
S->currLength = ;
S->maxLength = ;
return SUCCESS;
} // 测试函数
int testString() {
// 声明变量
String str1, str2;
// 初始化字符串
if(initString(&str1, "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ()<>[]") == SUCCESS) {
printf("初始化字符串S1成功!\n");
traverseString(&str1); // 遍历
}
if(initString(&str2, "abc") == SUCCESS) {
printf("初始化字符串S2成功!\n");
traverseString(&str2);
}
// 清空字符串
if(clearString(&str2) == SUCCESS) {
printf("清空字符串S2成功!\n");
}
// 判断字符串是否为空
printf("字符串S1是否为空?%s\n", isStringEmpty(&str1) ? "是" : "否");
printf("字符串S2是否为空?%s\n", isStringEmpty(&str2) ? "是" : "否");
// 字符串复制
if(copyString(&str2, "abcdefg") == SUCCESS) {
printf("字符串复制成功!\n");
traverseString(&str2);
}
// 字符串连接
if(concatString(&str1, &str2) == SUCCESS) {
printf("将字符串S2连接到S1后面成功!\n");
traverseString(&str1);
}
// 比较两个字符串的大小
printf("S1比S2的关系?%d\n", compareString(&str1, &str2));
// 字符串的长度
printf("字符串S1的长度:%d\n", getStringLength(&str1));
printf("字符串S2的长度:%d\n", getStringLength(&str2));
// 取字符串的子串
printf("S1从58位置开始7个长度的子串是:%s\n", getSubString(&str1, , ));
// 返回子串第一次出现的位置
printf("字符串S1中从20位置起,ABCDE子串第一次出现的位置是%d\n", locateSubString(&str1, "ABCD", ));
// 插入字符串
if(stringInsert(&str1, , "||||||") == SUCCESS) {
printf("在S1的26位置插入字符串||||||成功!\n");
traverseString(&str1);
}
// 删除字符串
if(stringDelete(&str1, , ) == SUCCESS) {
printf("从S1的26位置删除6个字符成功!\n");
traverseString(&str1);
}
// 替换字符串
if(replaceString(&str1, "abcdefg", "") == SUCCESS) {
printf("成功将S1中的所有abcdefg替换为0123456789!\n");
traverseString(&str1);
}
// 销毁字符串
if(destroyString(&str1) == SUCCESS) {
printf("销毁字符串S1成功!\n");
}
if(destroyString(&str2) == SUCCESS) {
printf("销毁字符串S2成功!\n");
}
return ;
}

  常量头文件 Constant.h 中的代码如下:

#include <stdio.h>
#include <stdlib.h> #define TRUE 1
#define FALSE 0 #define SUCCESS 1
#define FAILURE 0 #define SMALLER -1
#define EQUAL 0
#define BIGGER 1 typedef int Status;

  主函数所在的文件 main.c 中的代码如下:

#include <String.h>

int main() {
testString();
return ;
}

  运行结果如下:

初始化字符串S1成功!
遍历字符串:abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ()<>[]
初始化字符串S2成功!
遍历字符串:abc
清空字符串S2成功!
字符串S1是否为空?否
字符串S2是否为空?是
字符串复制成功!
遍历字符串:abcdefg
将字符串S2连接到S1后面成功!
遍历字符串:abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ()<>[]abcdefg
S1比S2的关系?1
字符串S1的长度:65
字符串S2的长度:7
S1从58位置开始7个长度的子串是:abcdefg
字符串S1中从20位置起,ABCDE子串第一次出现的位置是26
在S1的26位置插入字符串||||||成功!
遍历字符串:abcdefghijklmnopqrstuvwxyz||||||ABCDEFGHIJKLMNOPQRSTUVWXYZ()<>[]abcdefg
从S1的26位置删除6个字符成功!
遍历字符串:abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ()<>[]abcdefg
成功将S1中的所有abcdefg替换为0123456789!
遍历字符串:0123456789hijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ()<>[]0123456789
销毁字符串S1成功!
销毁字符串S2成功! Process returned 0 (0x0) execution time : 1.783 s
Press any key to continue.

【数据结构】之串(C语言描述)的更多相关文章

  1. 数据结构与算法分析——C语言描述 第三章的单链表

    数据结构与算法分析--C语言描述 第三章的单链表 很基础的东西.走一遍流程.有人说学编程最简单最笨的方法就是把书上的代码敲一遍.这个我是头文件是照抄的..c源文件自己实现. list.h typede ...

  2. 最小正子序列(序列之和最小,同时满足和值要最小)(数据结构与算法分析——C语言描述第二章习题2.12第二问)

    #include "stdio.h" #include "stdlib.h" #define random(x) (rand()%x) void creat_a ...

  3. C语言学习书籍推荐《数据结构与算法分析:C语言描述(原书第2版)》下载

    维斯 (作者), 冯舜玺 (译者) <数据结构与算法分析:C语言描述(原书第2版)>内容简介:书中详细介绍了当前流行的论题和新的变化,讨论了算法设计技巧,并在研究算法的性能.效率以及对运行 ...

  4. 数据结构与抽象 Java语言描述 第4版 pdf (内含标签)

    数据结构与抽象 Java语言描述 第4版 目录 前言引言组织数据序言设计类P.1封装P.2说明方法P.2.1注释P.2.2前置条件和后置条件P.2.3断言P.3Java接口P.3.1写一个接口P.3. ...

  5. 《数据结构与算法分析——C语言描述》ADT实现(NO.00) : 链表(Linked-List)

    开始学习数据结构,使用的教材是机械工业出版社的<数据结构与算法分析——C语言描述>,计划将书中的ADT用C语言实现一遍,记录于此.下面是第一个最简单的结构——链表. 链表(Linked-L ...

  6. 《数据结构与算法分析-Java语言描述》 分享下载

    书籍信息 书名:<数据结构与算法分析-Java语言描述> 原作名:Data Structures and Algorithm Analysis in Java 作者: 韦斯 (Mark A ...

  7. 《数据结构与算法分析:C语言描述_原书第二版》CH3表、栈和队列_reading notes

    表.栈和队列是最简单和最基本的三种数据结构.基本上,每一个有意义的程序都将明晰地至少使用一种这样的数据结构,比如栈在程序中总是要间接地用到,不管你在程序中是否做了声明. 本章学习重点: 理解抽象数据类 ...

  8. 读书笔记:《数据结构与算法分析Java语言描述》

    目录 第 3 章 表.栈和队列 3.2 表 ADT 3.2.1 表的简单数组实现 3.2.2 简单链表 3.3 Java Collections API 中的表 3.3.1 Collection 接口 ...

  9. 【数据结构与算法分析——C语言描述】第二章总结 算法分析

    算法 算法(algorithm)是为求解一个问题需要遵循的.被清楚地指定的简单指令的集合. 数学基础 四个定义: 1.大O表示法: 如果存在正常数 c 和 n0 使得当 N ≥ n0时,T(N) ≤ ...

  10. 《数据结构与算法分析——C语言描述》ADT实现(NO.05) : 散列(Hash)

    散列(Hash)是一种以常数复杂度实现查找功能的数据结构.它将一个关键词Key,通过某种映射(哈希函数)转化成索引值直接定位到相应位置. 实现散列有两个关键,一是哈希函数的选择,二是冲突的处理. 对于 ...

随机推荐

  1. Map文件从IDA到OD

    目录 什么是map文件 IDA与OD导出使用map文件 注意事项 使用OD载入导出的map文件 什么是map文件 什么是 MAP 文件? 简单地讲, MAP 文件是程序的全局符号.源文件和代码行号信息 ...

  2. access 2013下载 access 2010下载 access 2007下载 Access 2003下载 安装交流的论坛

    在网上搜索了一个access 2013下载 access 2010下载 access 2007下载 Access 2003下载 安装交流的论坛 office安装的常见问题: http://www.of ...

  3. tensorflow制作tfrecord格式数据

    tf.Example msg tensorflow提供了一种统一的格式.tfrecord来存储图像数据.用的是自家的google protobuf.就是把图像数据序列化成自定义格式的二进制数据. To ...

  4. [2018-01-12] laravel--路由(路由与控制器)

    路由只用来接收请求 目前我们大致了解了laravel,在开始一个Http程序需要先定义路由.之前的例子中,我们的业务逻辑都是在路由里实现的,这对于简单的网站或web应用没什么问题,当我们需要扩大规模, ...

  5. 【工利其器】Android Lint篇——为Android量身定做的静态代码审查工具

    前言 我们在进行代码优化的时候,往往是通过开发者的经验来判断哪些代码可能存在潜在问题,哪些资源的使用不合规范等.实际上Android SDK提供了一款功能非常强大的工具,来帮助开发者自动检测代码的质量 ...

  6. spring session源码解析

    模块划分 core部分代码 存储实现部分部分: jdbc实现 具体存储的实现类 例如:org.springframework.session.jdbc.JdbcOperationsSessionRep ...

  7. docker安装mysql,tomcat,并且在tomcat可以访问到mysql

    1.uname -an 查看当前系统版本 2.yum -y install docker 下载安装docker 3.service docker start  启动docker服务 4.docker ...

  8. 搜索框(SearchView)用法

    SearchView是Android原生的搜索框控件,它提供了一个用户界面,可以让用户在文本框内输入文字,并允许通过看监听器监控用户输入,当用户输入完成后提交搜索时,也可通过监听器执行实际的搜索. S ...

  9. GPIO硬件资源的申请,内核空间和用户空间的数据交换,ioctl(.....),设备文件的自动创建

    1.通过GPIO库函数控制LED   open("/dev/myleds",...)       close(fd)   ----------------------------- ...

  10. Flash、RAM、ROM的区别

    一. ROM(Read Only Memory)    ROM(Read Only Memory),只读存储器.用来存储和保存数据.ROM数据不能随意更新,但是在任何时候都可以读取.即使是断电,ROM ...