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

news2025/1/23 4:46:24

目录

一.初始化

顺序表中串的存储 

串的链式存储

二.赋值操作:将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;
}

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Vue 2.x 项目升级到 Vue 3详细指南【修改清单】

文章目录 前言0.迁移过程1. 安装 Vue 32. 逐一处理迁移中的警告3. 迁移全局和内部 API4. 迁移 Vue Router 和 Vuex5. 处理其他的不兼容变更 1. Vue3特性1. Composition API2. 更好的性能3. 更好的 TypeScript 支持4. 多个根元素5. Suspense 组件6. Teleport 组件7. 全局 API 的…

python 打印人口分布金字塔图

背景 今天介绍一个不使用 matplot&#xff0c;通过DebugInfo模块打印人口金字塔图的方法。 引入模块 pip install DebugInfo打印人口金字塔图 下面的代码构建了两个人口数据&#xff08;仅做功能演示&#xff0c;不承诺任何参考价值&#xff09;&#xff0c;男性人口和女性…

基础论文学习(4)——CLIP

《Learning Transferable Visual Models From Natural Language Supervision》 CLIP的英文全称是Contrastive Language-Image Pre-training&#xff0c;即一种基于对比文本-图像对的预训练模型。CLIP是一种基于对比学习的多模态模型&#xff0c;与CV中的一些对比学习方法如moc…

软考高级架构师下篇-12层次式架构设计理论与实践

目录 1. 考情分析2. 层次式体系结构概述3. 表现层框架设计4. 中间层框架设计5. 数据访问层设计6. 数据架构规划与设计7. 物联网层次架构设计7. 前文回顾1. 考情分析 根据考试大纲,层次式架构设计理论与实践知识点会涉及单选题型(约占2~5分)和案例题(25分),本小时内容偏重于方…

lesson9: C++多线程

1.线程库 1.1 thread类的简单介绍 C11 中引入了对 线程的支持 了&#xff0c;使得 C 在 并行编程时 不需要依赖第三方库 而且在原子操作中还引入了 原子类 的概念。要使用标准库中的线程&#xff0c;必须包含 < thread > 头文件 函数名 功能 thread() 构造一个线程对象…

LCD液晶屏接口静电浪涌保护用TVS/ESD二极管,如何选型?

LCD 液晶屏是Liquid Crystal Display 的简称&#xff0c;指将玻璃和LCD驱动器集成到一起的LCD显示产品&#xff0c;为用户提供了一个标准的LCD显示驱动接口&#xff0c;用户可以按照接口&#xff08;有4位、8位、VGA等不同类型&#xff09;要求进行操作来控制LCD正确显示。众所…

Fsm onehot

module top_module(input in,input [9:0] state,output [9:0] next_state,output out1,output out2);assign out1(state[8]|state[9])?1:0;assign out2(state[9]|state[7])?1:0;assign next_state[0](state[0]&(~in)) |(state[1]&(~in)) |(state[2]&(~in)) |(sta…

LVS+Keepalived 实验

Keepalived 是什么 Keepalived 是一个基于VRRP协议来实现的LVS服务高可用方案&#xff0c;可以解决静态路由出现的单点故障问题的一款检查工具 在一个LVS服务集群中通常有主服务器&#xff08;MASTER&#xff09;和备份服务器&#xff08;BACKUP&#xff09;两种角色的服务器…

ShardingSphere01-docker环境安装

使用docker安装数据库是一个非常好的选择&#xff0c;后续的读写分离、数据分片等功能的数据库都是由docker创建。 一、安装准备 1、前提条件 Docker可以运行在Windows、Mac、CentOS、Ubuntu等操作系统上 Docker支持以下的CentOS版本&#xff1a; CentOS 7 (64-bit)CentOS …

LLM预训练大型语言模型Pre-training large language models

在上一个视频中&#xff0c;您被介绍到了生成性AI项目的生命周期。 如您所见&#xff0c;在您开始启动您的生成性AI应用的有趣部分之前&#xff0c;有几个步骤需要完成。一旦您确定了您的用例范围&#xff0c;并确定了您需要LLM在您的应用程序中的工作方式&#xff0c;您的下…

第59步 深度学习图像识别:误判病例分析(TensorFlow)

基于WIN10的64位系统演示 一、写在前面 本期内容对等于机器学习二分类系列的误判病例分析&#xff08;传送门&#xff09;。既然前面的数据可以这么分析&#xff0c;那么图形识别自然也可以。 本期以mobilenet_v2模型为例&#xff0c;因为它建模速度快。 同样&#xff0c;基…

Linux下Jenkins安装 (最新)

环境概述 随着软件开发需求及复杂度的不断提高&#xff0c;团队开发成员之间如何更好地协同工作以确保软件开发的质量已经慢慢成为开发过程中不可回避的问题。Jenkins自动化部署可以解决集成、测试、部署等重复性的工作&#xff0c;工具集成的效率明显高于人工操作&#xff1b…

Android动画进阶指北

原文链接 Android Animation Advanced Tricks 前面的文章介绍了动画的基本使用方法&#xff0c;本文来聊一聊涉及到动画的高级技巧&#xff0c;以及一些非常优质的学习资源和动画三方库和框架。 页面之间的过渡动画 常规的动画都是针对某一页面上的某个元素做动画&#xff0c…

3、Spring_容器执行

容器执行点 1.整合 druid 连接池 添加依赖 <dependency><groupId>com.alibaba</groupId><artifactId>druid</artifactId><version>1.2.8</version> </dependency>1.硬编码方式整合 新建德鲁伊配置 <?xml version"1.…

微信小程序 车牌号输入组件

概述 一个小组件&#xff0c;用于方便用户输入车牌号码 详细 概述 有时候我们开发过程中会遇到需要用户输入车牌号的情况&#xff0c;让客户通过自带键盘输入&#xff0c;体验不好且容易出错&#xff0c;例如车牌号是不能输入O和I的&#xff0c;因此需要有一个自定义的键盘…

FPGA中锁存器(latch)、触发器(flip-flop)以及寄存器(register)详解

文章目录 1 定义1.1 锁存器&#xff08;latch&#xff09;1.2 触发器&#xff08;flip-flop&#xff09;1.3 寄存器&#xff08;register&#xff09; 2 比较2.1 锁存器&#xff08;Latch&#xff09;危害即产生原因2.2 寄存器和锁存器的区别2.3 锁存器和触发器的区别 3 结构3.…

dvwa环境搭建

靶场搭建 环境作用apache网站代理&#xff0c;部署网站php一种开源通用脚本语言&#xff0c;用来处理程序mysql数据库&#xff0c;用来存储数据 1.打开sever 2008&#xff0c;远程桌面连接&#xff0c;将文件拷贝到虚拟机&#xff0c;打开微软常用集合运行库&#xff0c;等待…

「Vue|网页开发|前端开发」01 快速入门:用vue-cli快速写一个Vue的HelloWorld项目

本文主要介绍如何用vue开发的标准化工具vue-cli快速搭建一个符合实际业务项目结构的hello world网页项目并理解vue的代码文件结构以及页面渲染流程。 文章目录 一、准备工作&#xff1a;安装node.js二、项目搭建创建项目目录全局安装vue-cli使用Webpack初始化项目启动项目学会…

RedisDesktopManager 连接redis

redis查看是否启动成功 ps -ef | grep redis以上未启动成功 cd /usr/local/bin/ 切换根目录 sudo -i 开启服务端 ./redis-server /usr/local/redis/redis.conf 开启客户端 ./redis-cli

Ganache 本地测试网远程连接

文章目录 前言1. 安装Ganache2. 安装cpolar3. 创建公网地址4. 公网访问连接5. 固定公网地址 前言 Ganache 是DApp的测试网络&#xff0c;提供图形化界面&#xff0c;log日志等&#xff1b;智能合约部署时需要连接测试网络。 Ganache 是一个运行在本地测试的网络,通过结合cpol…