数据结构—总结


数据结构—总结

没写完

1.线性表

顺序表

typedef int ElemenType;
typedef struct {
	ElemenType *data; // 数据域
	int length;
}Node;
// 顺序表初始化
void InitNode(Node *t){
    t->data = (Node *)malloc(sizeof(Node) * 100);
    if(t->data == NULL)
        exit(0);
   	t->length = 0;
}

单链表

/**
出现过一个概念,头结点,而且单链表是以结点的形式存在的,所有我们操作单链表时,其实也实在操作我们的结点
所有每一次才都需要一个结点
*/
typedef int ElemenType;
typedef struct {
	ElemenType data;
	struct LNode * next;
}LNode,*LinkList;
// 单链表需要初始化头结点   
void InitLNode(LinkList *t){
    t = (LNode *)malloc(LNode);
    (*t)->next = NULL;
}
// 创造单链表
void CreateLNode(LinkList *t){
    int n = 0;
    LinkList p;
    // 单链表的长度 理清楚结点还有各种指针之间的变化
    scanf("%d",&n);
    for(int i = 0; i < n; i++){
		p = (LNode *)malloc(LNode);
        scanf("%d",&p->data);
        p->next = (*t)->next;
        (*t)->next = p;
    }
}
// 插入与删除

双向链表

typedef int ElemenType;
typedef struct {
    ElemenType data;  // 数据域
    struct DuLNode * prior; // 前指针
    struct DuLNode * next; // 后指针
}DuLNode,*DuLinkList;

循环链表

2.栈

顺序栈

// 定义一个栈
#define MAXSIZE 100
typedef char ELemenType;
typedef struct{
    ElemenType * top; // 头指针 
    ElemenType * base; // 尾指针
    int stacksize; // 栈可用最大容量
}SqStack;
// 重要条件 判断栈空与栈满
// 初始化
void InitStack(SqStack * s){
    s.base = (SqStack *)malloc(sizeof(SqStack)*MAXSIZE);
    if(!s.base){
        exit(0);
    }
    s.top = s.base; // 空栈
    s.stacksize = MAXSIZE;
}
// 入栈  当栈未满时就可以入栈
void Push(SqStack *s,ElemenType e){
	if(s.top - s.base == s.stacksize) exit(0);
    s.top++=e;
}
// 出栈  当栈未空时出栈
void Pop(SqStack *s,ElemenType e){
    if(s.top == s.base) exit(0);
    e=*--s.top;
}

链栈

// 链栈
typedef char ElemenType;
typedef struct {
    ElemenType data;  // 数据域
    struct StackNode * next;  // 指针域
}StackNode,*LinkStack;
// 初始化
void InitSqstack(LinkStack *s){
    s = NULL;  // 创建一个空栈s,栈顶指针为空,
}
// 入栈
void Push(LinkStack *s,ElemenType e){
    StackNode p = (StackNode *)malloc(sizeof(StackNode));
    p->data = e;
    p->next = s;
    s = p;
}
// 出栈
void Pop(LinkStack *s,ElemenType *e){
    if(s == NULL) exit(0);
    e = s->data;
    p = s;
    s = s->next;
    delete p;
}

3.队列

循环队列

#define MAXSIZE 100
typedef char ElemenType;
typedef struct {
    ElemenType * base; // 存储空间的基地址
    int front;  // 队头指针
    int rear;   // 队尾指针
}SqQueue;
/**
使用一种方式来实现循环队列  
队空 : Q.front == Q.rear;
队满 : (Q.rear + 1)%MAXSIZE == Q.front;
*/
// 初始化
void InitQueue(SqQueue *q){
    q.base = (SqQueue *)malloc(sizeof(SqQueue)*MAXSIZE);
    if(!q.base) exit(0);
    q.front = q.rear = 0; //  队头队尾置为0,队列为空
}
// 入队  入队要判断是不是满队
void EnQueue(SqQueue *q,ElemenType e){
    if((q.rear + 1)%MAXSIZE == q.front)
        exit(0);
    q.base[q.rear] = e;
    q.rear = (q.rear + 1)% MAXSIZE;
}
// 出队 出队要看是不是空队
void DeQueue(SqQueue *q,ElemenType *e){
    if(q.front == q.rear) exit(0);
    e = q.base[q.front];
    q.front = (q.front + 1)%MAXSIZE;
    
}

链队

typedef char ElemenType;
typedef struct{
    ElemenType data;
    struct QNode * next;
}QNode,*QueuePtr;
typedef struct {
    QueuePtr front;  // 队头指针
    QueuePtr read;  // 队尾指针
}LinkQueue;

4.串,数组,广义表

串(string)(或字符串)是由零个或多个字符组成的有限序列,一般记为 s = “a1a2a3…….an”(n>=0) 其中,s是串的名,用双引号括起来的字符序列是串的值;ai(1<=i<=n)可以是字母,数字或其他符号;串中的字符的数目n称为串的长度,零个字符的串称为空串(null string) ,其长度为零。

串中任意个连续的字符组成的子序列称为该串的子串,包含字串的串相应的称为主串。通常称字符在序列中的序号为该字符在串中的位置。字串在主串中的位置则以字串的第一个字符在主串中的位置来表示。

// ------ 串的定长顺序存储结构 ----
#define MAXLEN 255         // 串的最大长度
typedef struct {
    char ch[MAXLEN + 1];   // 存储串的一维数组
    int length;            // 串的当前长度
}SString;
// ------ 串的堆式顺序存储结构 ----
typedef struct{
    char *ch;              // 若是非空串,则按串长分配存储区,否则ch为NULL
    int length;            // 串的当前长度
}HString;
// ------ 串的链式存储结构 -------
#define CHUNKSIZE 80  	   // 可由用户定义的块的大小
typedef struct Chunk{
    char ch [CHUNKSIZE];
    struct Chunk *next;
}Chunk;
typedef struct{
    Chunk *head,*tail;     // 串的头和尾指针
    int length;			   // 串的当前长度
}LString;

串中两种重要算法 BF KMP 算法

// ----- BF (Brute-Force)算法 ---------
/*
【概述】
模式匹配不一定是从主串的第一个位置开始,可以指定主串中查找的起始位置pos.如果采用字符串顺序存储结构,
可以写出不依赖于其他串操作的匹配算法
【算法步骤】
1.分别利用计数指针i和j指示主串S和模式T中当前正待比较的字符位置,i初始化pos,j初值为1.
2.如果两个串均未比较到串尾,即i和j分别指示串中下个位置,继续比较后续字符;
 2.1 S.ch[i] 和 T.ch[j] 比较,若相等,则i和j分别知识串中下个位置,继续比较后续字符;
 2.1 若不等,指针后退重新开始匹配,从主串的下一个字符(i=i-j+2)起再重新和模式的第一个字符(j=1)比较
3.如果j>T.length,说明模式T中的每个字符串依次和主串S中的一个连续字符序列相等,则匹配成功返回和模式T中第一个字符相等的字符串在主串S中的序号(i-T.length);否则称匹配不成功,返回0。
*/
//【算法描述】
int Index_BF(SString S,SString T,int pos){
    // 返回模式T在主串S中第pos个字符开始第一次出现的位置
    // 其中,T非空,1 <= pos <= S.length;
    int i,j;
    i = pos; j = 1;                            // 初始化
    while(i <= S.length && j <= T.length){     // 两个串均为比较到串尾
        if(S[i].ch == T[j].ch){                // 继续比较后继字符
            ++i;
            ++j;
        }
        else {                                 // 指针后退重新开始匹配
            i = i - j + 2;
            j = 1
        }
    }
    if(j>T.length)                             // 匹配成功
        return i - T.length; 
    else    
        return 0;                             // 匹配失败
}
// ----- KMP 算法 ---------
// 【算法描述】
int Index_KMP(SString S,SString T,int pos){
    // 利用模式串T的next函数求T在主串S中第pos个字符之后的位置
    // 其中,T非空,1 <= pos <= S.length
    i = pos, j = 1;
    while(i <= S.length && j <= S.length){
        if(i == 0 || S[i] == T[j]){
            ++i;
            ++j;
        }else {
            j = next[j];
        }
    }
    if(j > T[0])
        return i - T[0];
    else return 0;
}
// 计算next函数值
void get_next(SString T,int next[]){
    //求模式串T的next函数值并存入数组next
    i = 1;
    next[1] = 0;
    j = 0;
    while(i < T.length){
        if(j == 0 || T.ch[i]){
            ++i;
            ++j;
            next[i] = j;
        }else {
            j = next[j];
        }
    }
}
// next函数的修订值
void get_nextval(SString T, int nextval[]){
    i = 1;
    nextval[1] = 0;
    j = 0;
    while(i < T.length){
        if(j == 0 || T.ch[i] == T.ch[j]){
            ++i;
            ++j;
            if(T.ch[i] != T.ch[j]) 
                nextval[i] = j;
            else 
                nextval[i] = nextval[j];
        }else {
            j = nextval[j];
        }
    }
}

数组

数据是由类型相同的数据元素构成的有序集合,每个元素称为数组元素,每个元素受n(n >= 1)个线性关系的约束,每个元素在n个线性关系中的序号i1,i2,…….in称为该元素的下标,可以通过下标访问该数组元素。因为数组中每个元素处于n(n >= 1) 个关系中,故称该数组为n维数组。


广义表

广义表是线性表的推广,也称为列表。广泛用于人工智能等领域的表处理语言LISP语言,吧广义表作为基本的数据结构,就连程序也表示为一系列的广义表

//—————————广义表的头尾链表存储表示————————————
typedef enum{ATOM.LIST} ElemTag;   // ATOM == 0; 原子 ;  LIST == 1; 子表
typedef struct GLNode{
	ElemTag tag;  				// 公共部分
    union{        				// 原子结点和表结点的联合部分
        AtomType atom;  		// atom是原子结点的值域,AtomType由用户定义
        struct{
            struct GLNode * hp,*tp;//ptr是表结点的指针域,ptr.hp和ptr.tp分别指向表头和表尾
 		}ptr;
    };
}*GList;     				  // 广义表类型

5.树和二叉树

树(Tree)是n个结点的有限集,它或为空树;或为非空树,对于非空树T;

1) 有且有一个称之为根的结点

2)除根结点以外的其余节点可分为m个互不相交的有限集T1,T2,……Tm,其中每一个集合本身又是一颗树,并且呈为根的字数(Sub Tree)

//--------二叉树的顺序存储结构--------
#define MAXTSiZE 100
typedef TElemType SqBiTree[MAXTSIZE];
SqBiTree bt;
/*
顺序存储就跟数组存储差不多
*/
//--------二叉树的链式存储结构--------
typedef struct BiTNode{
    TElemType data;
    struct BiTNode *lchild,*rchild;
}BiTNode,*BiTree;
//--------二叉树的二叉线索存储结构--------
typedef struct BiThrNode{
    TElemType data;
    struct BiThrNode *lchild,*rchild;
    int LTag,RTag;
}BiThrNode,*BiThrTree;
//--------树的二叉链表(孩子-兄弟)存储表示-
typedef struct CSNode{
    ElemType data;
    struct CSNode *firstchild,*nextsibling;
}CSNode,*CSTree;
//--------哈夫曼存储结构--------
typedef struct {
	int weight;
    int parent,lchild,rchild;
}HTNode,*HuffmanTree;

6.图

图G由两个集合V和E组成,记为G=(V,E),其中v是顶点的有穷非空集合,E是V中定点偶对的有穷集合,这些顶点偶对边称为边,V为点集,E为边集

//--------图的邻接矩阵存储结构--------
#define MaxInt 32767
#define MVNum 100
typedef char VerTexType;
typedef int ArcType;
typedef struct{
    VerTexType vexs[MVNum];
    ArcType	   arcs[MVNum][MVNum];
    int vexnum,arcnum;
}AMGraph;
//--------图的邻接表存储结构--------
#define MVNum 100
typedef struct ArcNode;
{
	int adjvex;
    struct ArcNode * nextarc;
    Otherinfo info;
}ArcNode;
typedef struct VNode{
    VerTexType data;
    ArcNode *firstarc;
}VNode,AdjList[MVNum];
typedef struct{
    AdjList vertices;
    int vexnum,arcnum;
}ALGraph;
//-------有向图的十字链表存储表示存储结构--------
#define MAX_VERTEX_NUM 20
typedef struct ArcBox{
	int tailvex,headvex;
    struct ArcBox *hlink,*tlink;
    infoType *info;
}ArcBox;
typedef struct VexNode{
	VextexType data;
    ArcBox *firstin,*firstout;
}VexNode;
typedef struct {
    VexNode xlist[MAX_VERTEX_NUM];
    int vexnum,arcnum;
}OLGraph;
//--------无向图的邻接多重表存储结构--------
#define MAX_VERTEX_NUM 20
typedef enum{unisited,visited} Visitlf;
typedef struct EBox{
    Visitlf mark;
    int ivex,jvex;
    struct EBox *ilink,*jlink;
    infoType *info;
}EBox;
typedef struct VexBox {
    VertexType data;
    EBox *firstedge;
}VexBox;
typedef struct {
    VexBox adjmulist[MAX_VERTEX_NUM];
    int vexnum,edgenum;
}AMLGraph;

7.查找

查找方法:

  1. 线性表查找
    • 顺序查找
    • 折半查找
    • 分块查找
  2. 树表查找
    • 二叉排序树
    • 平衡二叉树
    • B-树
    • B+树
  3. 散列表查找

8.排序

查找方法:

  1. 内部排序法
    • 插入类:直接插入排序,折半插入排序,希尔排序
    • 交换类:冒泡排序,快速排序
    • 选择类:简单选择排序,树形选择排序,堆排序
    • 归并类:2-路归并排序
    • 分配类:基数排序
  2. 外部排序法

原创文章,作者:ItWorker,如若转载,请注明出处:https://blog.ytso.com/tech/pnotes/278853.html

(0)
上一篇 2022年8月4日 14:00
下一篇 2022年8月4日 14:09

相关推荐

发表回复

登录后才能评论