数据结构---串(赋值,求子串,比较,定位)

目录

一.初始化

顺序表中串的存储 

串的链式存储

二.赋值操作:将str赋值给S

链式表

顺序表

三.复制操作:将chars复制到str中

链式表

顺序表

四.判空操作

链式表

顺序表

五.清空操作

六.串联结

链式表

顺序表

七.求子串

链式表

顺序表

八. 比较串的大小

链式表

顺序表

九.定位操作

链式表

顺序表


一.初始化

顺序表中串的存储 

#define MaxLen 255
typedef struct{
    char ch[MaxLen];//静态数组,系统自动回收
    int length;

}SString;

typedef struct{
    char *ch;
    int length;

}HString;

HString S;
S.ch=(char*)malloc(MaxLen*sizeof(char));//在堆区分配存储空间,所以需要free手动回收
S.length=0;

串的链式存储

typedef struct StringNode{
    char ch;//一个字符占一个字节
    struct StringNode *next;//占4个字节/8个字节

}StringNode,*String;
//以上定义的结构体类型存储密度低,可以采用以下方案定义结构体

typedef struct StringNode{
    char *ch;//或一个结点存放更多字符
    struct StringNode *next;
}StringNode,*String;

二.赋值操作:将str赋值给S

链式表

void StrAssign(const StringNode* str, char** S) {
    int len = 0;
    const StringNode* currentNode = str;
    
    // 计算字符串长度
    while (currentNode != NULL && currentNode->ch != '\0') {
        len++;
        currentNode = currentNode->next;
    }
    
    *S = (char*)malloc((len + 1) * sizeof(char)); // 分配空间,要考虑'\0'所以长度加1
    
    int i = 0;
//将 currentNode 指向 str,我们可以从链表的头部开始逐个访问节点,并依次处理每个节点的字符
    currentNode = str;
    // 复制字符到新的字符串里
    while (currentNode != NULL && currentNode->ch != '\0') {
        (*S)[i] = currentNode->ch;
        i++;
        currentNode = currentNode->next;
    }
    
    (*S)[i] = '\0'; // 在新的字符串末尾添加'\0'表示结束
}

顺序表

void StrAssign(SString S, SString& sub) {
    int length = 0;
    while (S.ch[length] != '\0') {
        sub.ch[length] = S.ch[length];
        length++;
    }
    sub.ch[length] = '\0';
}

三.复制操作:将chars复制到str中

链式表

void StrCopy(StringNode* str, const char* chars) {
    int len = strlen(chars);
    int i;

    for (i = 0; i < len; i++) {
        str->ch = chars[i];
        if (i < len - 1) {
            str->next = (StringNode*)malloc(sizeof(StringNode));
            // 若字符串还没有复制完毕则手动扩展空间
            str->next->ch = '\0'; // 将下一个节点的 ch 字段设置为空字符
            str = str->next;
        }
    }

    str->next = NULL;
}

顺序表

和赋值操作相同

void StrCopy(SString S, SString& sub) {
    int length = 0;
    while (S.ch[length] != '\0') {
        sub.ch[length] = S.ch[length];
        length++;
    }
    sub.ch[length] = '\0';
}

四.判空操作

链式表

bool StrEmpty(String str)
{
    if(str->ch[0]=='\0')
        return true;
    else
        return false;

}

顺序表

bool StrEmpty(SString S) {
    if (S.ch[0] == '\0')
        return true;
    else
        return false;
}

五.清空操作

void StrClear(StringNode* str)
{
    while(str!=NULL){
        free(str);
        str=str->next;
    }
}

六.串联结

链式表

void Concat(StringNode* str, char* s1, char* s2) {
    int len1 = strlen(s1);
    int len2 = strlen(s2);

    // 创建一个新的结点用于存储连接后的字符串
    StringNode* newNode = (StringNode*)malloc(sizeof(StringNode));
    newNode->ch = (char*)malloc((len1 + len2 + 1) * sizeof(char)); // 加1是为了存放字符串结束符 '\0'
    newNode->next = NULL;

    // 连接字符串并保存到新结点的ch字段
    strcpy(newNode->ch, s1);
    strcat(newNode->ch, s2);

    // 找到链表末尾并将新结点连接到其后
    while (str->next != NULL) {
        str = str->next;
    }
    str->next = newNode;
}

顺序表

void Concat(SString& sub, SString S, SString T) {
    int i = 0;

    // 复制字符串 S 到 sub
    while (S.ch[i] != '\0') {
        sub.ch[i] = S.ch[i];
        i++;
    }

    // 复制字符串 T 到 sub
    int j = 0;
    while (T.ch[j] != '\0') {
        sub.ch[i] = T.ch[j];
        i++;
        j++;
    }

    sub.ch[i] = '\0'; // 添加字符串结束符
}

七.求子串

链式表

void Insert(StringNode** str, char ch) {
    StringNode* newNode = (StringNode*)malloc(sizeof(StringNode));
    newNode->ch = ch;
    newNode->next = NULL;

    if (*str == NULL) {
        *str = newNode;
        return;
    }

    StringNode* currentNode = *str;
    while (currentNode != NULL) {
        currentNode = currentNode->next;
    }

    currentNode->next = newNode;
}

StringNode* SubString(StringNode* str, int pos, int len) {
    StringNode* subStr = NULL;
    StringNode* currentNode = str;
    int currentPos = 0;

    while (currentNode != NULL && currentPos < pos + len) {
        if (currentPos >= pos) {
            Insert(&subStr, currentNode->ch);
        }
        currentNode = currentNode->next;
        currentPos++;
    }

    return subStr;
}

顺序表

bool SubString(SString &Sub,SString S,int pos,int len)
{
    if((pos+len-1)>S.length)
        return false;
    for(int i=pos;i<pos+len;i++)
    {
        Sub.ch[i-pos+1]=S.ch[i];
    }
    Sub.length=len;
    return true;
}

八. 比较串的大小

链式表

int getListLength(StringNode* head) {
    int length = 0;
    ListNode* current = head;
    while (current != nullptr) {
        length++;
        current = current->next;
    }
    return length;
}

int compareLinkedListLength(StringNode* list1, StringNode* list2) {
    int length1 = getListLength(list1);
    int length2 = getListLength(list2);

    if (length1 > length2) {
        return 1;
    } else if (length1 < length2) {
        return -1;
    } else {
        return 0;
    }
}

顺序表

若S>T,则返回值>0;若S=T,则返回值=0;若S<T,则返回值<0

int StrCompare(SString S,SString T)
{
    for(int i=1;i<S.length&&i<T.length;i++)
    {
        if(S.ch[i]!=T.ch[i]);
            return S.ch[i]-T.ch[i];
    }
//扫描过的所有字符都相同,则长度长的串更大
    return S.length-T.length;
}

九.定位操作

链式表

ListNode* locateNode(StringNode* head, int position) {
    if (position <= 0 || head == nullptr) {
        return nullptr;
    }

    ListNode* current = head;
    int count = 1;
    while (current != nullptr && count < position) {
        current = current->next;
        count++;
    }

    return current;
}

顺序表

若主串S中存在与串T值相同的子串,则返回它在主串S中第一次出现的位置,否则函数值为0

#define MaxLen 255
typedef struct {
    char ch[MaxLen]; // 静态数组,系统自动回收
    int length;
} SString;

void SubString(SString& sub, SString S, int start, int length) {
    int i;
    for (i = 0; i < length; ++i) {
        sub.ch[i] = S.ch[start + i];
    }
    sub.ch[length] = '\0'; // 添加字符串结束符
    sub.length = length;
}

int StrCompare(SString S, SString T) {
    int i = 0;
    while (S.ch[i] != '\0' && T.ch[i] != '\0') {
        if (S.ch[i] != T.ch[i]) {
            return S.ch[i] - T.ch[i];
        }
        i++;
    }

    return S.length - T.length;
}

int Index(SString S, SString T) {
    int i = 1, n = S.length, m = T.length; // n, m 分别为字符串 S 和 T 的长度
    SString sub;
    while (i <= n - m + 1) {//确保了字符串 T 在字符串 S 中进行匹配时不会越界
//假设 n = 10,m = 3,那么 n - m + 1 = 8
//在循环中,当 i = 9 或 i = 10 时,剩余的字符串 S 的长度已经小于 T 的长度,无法再进行匹配
        SubString(sub, S, i, m);
        if (StrCompare(sub, T) != 0)
            ++i;
        else
            return i;
    }
    return 0;
}

以上代码如有错误,请大佬们赐教!!💖💖感谢到大佬们指出错误

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/84302.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

系统架构合理性的思考 | 京东云技术团队

最近牵头在梳理部门的系统架构合理性&#xff0c;开始工作之前&#xff0c;我首先想到的是如何定义架构合理性&#xff1f; 从研发的角度来看如果系统上下文清晰、应用架构设计简单、应用拆分合理应该称之为架构合理。 基于以上的定义可以从以下三个方面来梳理评估&#xff1…

java面试基础 -- 深克隆 浅克隆

引例 说到java的克隆你还记得多少? 一说到克隆你可能就会想起来那个接口, 没错, 他就是Cloneable Cloneable是java里面内置的很常用的接口, 我们说 Object类中也有一个clone方法: 但是要想合法调用 clone 方法, 必须要先实现 Clonable 接口, 否则就会抛出 CloneNotSupportedEx…

【哈希表】HashSet HashMap LeetCode习题

目录 136.只出现一次的数字 137.只出现一次的数字 || 217.存在重复元素 219.存在重复元素 || 771.宝石与石头 旧键盘(牛客) 首先需要导包 import java.utli.*; 表中常用的是前两个&#xff0c;时间复杂度低。O(1) Set<E> set new HashSet<>(); set.conta…

6.Web后端开发【SpringBoot入门】

文章目录 1 SpringBoot快速入门1.1 Web分析 2. HTTP协议2.1 HTTP-概述2.1.1 介绍2.2.2 特点 2.2 HTTP-请求协议2.3 HTTP-响应协议2.3.1 格式介绍2.3.2 响应状态码 常见的相应状态码 3 WEB服务器3.1 服务器概述 1 SpringBoot快速入门 Spring的官网Spring Boot 可以帮助我们非常…

行为型(二) - 模板模式

一、概念 模板模式&#xff08;Template Pattern&#xff09;&#xff1a;模板方法模式在一个方法中定义一个算法骨架&#xff0c;并将某些步骤推迟到子类中实现。模板方法模式可以让子类在不改变算法整体结构的情况下&#xff0c;重新定义算法中的某些步骤。 二、实现 这里…

Matlab使用

Matlab使用 界面介绍 新建脚本&#xff1a;实际上就是新建一个新建后缀为.m的文件 新建编辑器&#xff1a;ctrlN 打开&#xff1a;打开最近文件&#xff0c;以找到最近写过的文件 点击路径&#xff0c;切换当前文件夹 预设&#xff1a;定制习惯用的界面 常见简单指令 ;…

一体全栈、开箱即用!麒麟信安与灵雀云携手打造“操作系统+云平台”联合解决方案

近日麒麟信安与北京凌云雀科技有限公司&#xff08;以下简称“灵雀云”&#xff09;开展生态合作&#xff0c;共同完成了灵雀云企业级全栈云原生平台ACPV3与麒麟信安操作系统V3等系列产品的兼容性认证测试。基于双方产品兼容性良好、稳定运行、性能表现卓越&#xff0c;麒麟信安…

工业类LMQ61460AASRJRR,汽车类LMQ61460AFSQRJRRQ1、LMQ61460AASQRJRRQ1 6A、降压转换器简化原理图

一、LMQ61460AASRJRR器件概述&#xff1a; LMQ61460 是一款具有集成旁路电容器的高性能直流/直流同步降压转换器。该器件具有集成式高侧和低侧MOSFET&#xff0c;能够在 3.0V 至 36V 的宽输入电压范围内提供高达 6A 的输出电流&#xff1b;可耐受 42V 电压&#xff0c;简化了输…

Spring Boot

前言 什么是Spring Boot&#xff1f;为什么要学Spring Boot&#xff1f; Spring 的诞⽣是为了简化Java 程序的开发的&#xff0c;⽽Spring Boot 的诞⽣是为了简化Spring 程序开发 的。Spring就像汽车&#xff0c;相比以前人只能其自行车走路&#xff0c;汽车可以帮助人们更快…

cad图怎么转换成pdf格式?一招教你轻松转换

将CAD文件转换成PDF格式有很多优势。首先&#xff0c;PDF格式是一种非常流行的文件格式&#xff0c;几乎所有电脑上都可以打开。这意味着即使将PDF文件发送给其他人&#xff0c;他们也可以轻松地查看文件&#xff0c;此外&#xff0c;PDF格式可以保留CAD文件的图形和布局&#…

Vue开发中如何解决国际化语言切换问题

Vue开发中如何解决国际化语言切换问题 引言&#xff1a; 在如今的全球化时代&#xff0c;应用程序的国际化变得越来越重要。为了让不同地区的用户能够更好地使用应用程序&#xff0c;我们需要对内容进行本地化&#xff0c;以适应不同语言和文化环境。对于使用Vue进行开发的应用…

【自适应稀疏度量方法和RQAM】疏度测量、RQAM特征、AWSPT和基于AWSPT的稀疏度测量研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

软件测试报告:包含哪些内容?

软件测试报告的内容 软件测试报告通常包括以下内容&#xff1a; 1、项目背景&#xff1a;介绍测试报告的编写目的、测试系统名称、测试环境和用到的专业术语。 2、需求内容&#xff1a;罗列该项目的测试功能点,具体到每个模块功能&#xff0c;以新增的功能和修改的功能为主&…

【excel密码】如何禁止移动、删除excel工作表?

想要工作表不被他人移动、删除等操作&#xff0c;该如何设置&#xff1f;今天分享如何设置才能够禁止excel工作表移动、删除。 打开excel工作表&#xff0c;点击工具栏中的审阅 – 保护工作簿 点击保护工作簿之后&#xff0c;会有弹框出现&#xff0c;输入想要设置的excel密码…

Ubuntu本地快速搭建web小游戏网站,并使用内网穿透将其发布到公网上

文章目录 前言1. 本地环境服务搭建2. 局域网测试访问3. 内网穿透3.1 ubuntu本地安装cpolar内网穿透3.2 创建隧道3.3 测试公网访问 4. 配置固定二级子域名4.1 保留一个二级子域名4.2 配置二级子域名4.3 测试访问公网固定二级子域名 前言 网&#xff1a;我们通常说的是互联网&am…

光伏发电+boost+储能+双向dcdc+并网逆变器控制(低压用户型电能路由器仿真模型)【含个人笔记+建模参考】

MATALB代码链接&#xff1a;光伏发电boost十储能十双向dcdc十并网逆变器 个人笔记与建模参考请私信发送 包含Boost、Buck-boost双向DCDC、并网逆变器三大控制部分 boost电路应用mppt&#xff0c; 采用扰动观察法实现光能最大功率点跟踪 电流环的逆变器控制策略 双向dcdc储能系…

A. Two Semiknights Meet

题目描述 可知走法为中国象棋中的象的走法 解题思路 利用结构体来存储两个 K K K的位置 x , y x,y x,y&#xff0c;因为两个 K K K同时走&#xff0c;所以会出现两种情况 相向而行&#xff0c;两者距离减少 相反而行&#xff0c;两者距离不变 我们完全可以不考虑格子是好…

分布式 - 消息队列Kafka:Kafka 消费者消费位移的提交方式

文章目录 1. 自动提交消费位移2. 自动提交消费位移存在的问题&#xff1f;3. 手动提交消费位移1. 同步提交消费位移2. 异步提交消费位移3. 同步和异步组合提交消费位移4. 提交特定的消费位移5. 按分区提交消费位移 4. 消费者查找不到消费位移时怎么办&#xff1f;5. 如何从特定…

DSO 系列文章(2)——DSO点帧管理策略

文章目录 1.点所构成的残差Residual的管理1.1.前端残差的状态1.2.后端点的残差的状态1.3.点的某个残差的删除 2.点Point的管理2.1.如何删除点——点Point的删除2.2.边缘化时删除哪些点&#xff1f; 3.帧FrameHessian的管理 DSO代码注释&#xff1a;https://github.com/Cc19245/…

elemenPlus ElMessage 字符串如何换行问题

因为后端返回的数据是一长串&#xff0c;而且带有\r,\n等换行符&#xff0c;但是并没有生效。前端写法&#xff1a; // 抛出错误ElMessage.error(msg);我们知道\r&#xff0c;\n&#xff0c;\r\n 是在不同系统下的换行符的表示&#xff0c;但在JavaScript返回字符串中并没有生效…