数据结构 第四章 串、数组和广义表

2024-06-22 18:38

本文主要是介绍数据结构 第四章 串、数组和广义表,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述:

串:字符串的简称,串是一种特殊的线性表,特殊在其数据元素都是一个字符。
数组和广义表可以看做是线性表的扩充,即线性表的数据元素自身又是一个数据结构。

串 String

本结主要讲述:串的存储结构和基本操作
定义:主要是有0个或多个字符组成的序列。
存储结构:顺序存储和链式存储,但是串一般使用顺序存储结构。

顺序存储

typedef struct {char *ch;   //若为非空串,则按串长分配存储区,否则ch为nullint length;  //串长度
}HString;

链式存储:

#define CHUNKSIZE 80;  //可有用户定义的块大小
typedef struct Chunk{char ch[CHUNKSIZE ];   struck Chunk *next;
}Chunk;typedef struct {Chunk *head,*tail;//串的头尾指针int  curlen;   //串的当前长度
}LString;

串的存储密度=串值所占的存储位/实际分配的存储位
密度小,运算处理才方便;

串的模式匹配算法
子串的定位运算通常称为串的模式匹配或串匹配。
应用场景:搜索引擎、拼音检查、语言翻译、数据压缩等。

eg:有两个字符串S-主串、T-子串(也称模式);
著名的模式匹配算法有两种:BF和KMP算法:
1. BF算法:最简单直观的模式匹配算法,Brute-Force算法

int Indext(SString S,SString T,int pos){int i=pos,j=1;  //i指向主串,j指向子串while(S[0]>=i && j<=T[0]){if(S[i] == T[j]){++i;++j;}else{i=i-j+2;j=1;}}if(j>T[0]){return i-T[0];}else{return 0;}}

主串长:N ,子串长:M
算法的时间复杂度:在最好的情况下(N+M)1/2即O(N+M),最坏的情况下:M(N-M+2)*1/2即O(N+M);
BF思路直观简明,但是时间复杂度高为:O(N+M),

2.KMP算法:由Kunth Morris Pratt共同设计所以称为KMP算法;
特点:无需回溯主串的指针,当模式串与主串中存在许多“部分匹配”的情况下才显得比BF快,所以BF至今任然被采用。
时间复杂度:O(m+n)

int Indext_KMP(SString S,SString T,int pos){//利用模式串Tnext函数求T在主串S中第pos个字符之后的位置//其中,T非空,1<=pos<=Strlength(S)int i=pos,j=1;  //i指向主串,j指向子串while(S[0]>=i && j<=T[0]){if(j==0 || S[i] == T[j]){++i;++j;}else{j=next[j]; //模式串向右移动 }}if(j>T[0]){return i-T[0];}else{return 0;}}T的Next函数,算法时间复杂度为O(m)
void get_next(SString T,int next[]){int i = 1;next[1] = 0 ;int j = 0 ;while(i<T[0]){if(j == 0 || T[i] == T[j]){++i;++j;next[i] = j;}else{j=next[j];}}
}
J12345678
模式串abaabcac
next[j]01122312

上面的next算法有缺陷,下面有修正版的:void get_nextval(SString T,int nextval[]){int i = 1;nextval[1] = 0 ;int j = 0 ;while(i<T[0]){if(j == 0 || T[i] == T[j]){++i;++j;if(T[i] != T[j]){nextval[i] = j;}else{nextval[i] = nextval[j];}}else{j=nextval[j];}}
}

next函数修正值:

J12345
模式串aaaab
next[j]01234
nextval[j]00004

数组

本结主要讲述:数组的内部实现和特殊的二维数组如何实现压缩存储。
定义:由类型相同的数据元素构成的有序集合。

一维数组可以看成线性表
二维数组是数据元素为线性表的线性表;
数组一般不做插入和删除操作,所以一般采用顺序存储结构。

二维数组有两种存储方式:列序为主序的存储方式,行序为主序的存储方式。
java、C、Basic、Pasical都是行序为主序的编程语言;
Fortran是以列序为主序的编程语言;

每个元素占L个存储单元,二维数组A[0~m-1,0~n-1](A[m,n])中任一元素aij的存储位置:
LOC(i,j)=LOC(0,0)+(n x i+j)L;

由于计算各个元素存储位置的时间相等,所以存取数组中任一元素的时间也相等,即数组是一种随机存取结构。


矩阵的压缩存储

矩阵用二维数组来表示是最自然的。
压缩存储:指为多个值相同的元只分配一个存储空间,对0元不分配空间;
特殊矩阵:对值相同的元素或0元素在矩阵中的分布有一定规律;
主要有三种特殊矩阵:对称矩阵、三角矩阵、对角矩阵
若n阶矩阵A中的元满足Aij=Aji,称为n阶对称矩阵。
–可将n2个元素压缩到n(n+1)/2个元的空间中。
设一维数组Sa[n(n+1)/2]作为矩阵A的存储结构,则sa[k]和矩阵元aij的关系:
k=i(i-1)/2+j-1 条件(i < j)
k=j(j-1)/2+i-1 条件(i > j)


广义表

本结主要讲述:广义表的概念和存储结构。
广义表是线性表的推广,也称为列表。
广泛的用于人工智能领域的表处理语言LISP语言。
记为:LS=(a1,a2,a3,a4…..an)

//广义表的头尾链表存储表示
typedef enum{ATOM,LIST
}ElemTag;typedef struct GLNode{ElemTag tag;   union{AtomType atom;struct{struct GLNode *hp;struct GLNode *tp;}ptr;};
}*GList;

这篇关于数据结构 第四章 串、数组和广义表的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1085093

相关文章

C#数据结构之字符串(string)详解

《C#数据结构之字符串(string)详解》:本文主要介绍C#数据结构之字符串(string),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录转义字符序列字符串的创建字符串的声明null字符串与空字符串重复单字符字符串的构造字符串的属性和常用方法属性常用方法总结摘

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

Java中数组转换为列表的两种实现方式(超简单)

《Java中数组转换为列表的两种实现方式(超简单)》本文介绍了在Java中将数组转换为列表的两种常见方法使用Arrays.asList和Java8的StreamAPI,Arrays.asList方法简... 目录1. 使用Java Collections框架(Arrays.asList)1.1 示例代码1.

C++一个数组赋值给另一个数组方式

《C++一个数组赋值给另一个数组方式》文章介绍了三种在C++中将一个数组赋值给另一个数组的方法:使用循环逐个元素赋值、使用标准库函数std::copy或std::memcpy以及使用标准库容器,每种方... 目录C++一个数组赋值给另一个数组循环遍历赋值使用标准库中的函数 std::copy 或 std::

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Go语言中三种容器类型的数据结构详解

《Go语言中三种容器类型的数据结构详解》在Go语言中,有三种主要的容器类型用于存储和操作集合数据:本文主要介绍三者的使用与区别,感兴趣的小伙伴可以跟随小编一起学习一下... 目录基本概念1. 数组(Array)2. 切片(Slice)3. 映射(Map)对比总结注意事项基本概念在 Go 语言中,有三种主要

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE