数据结构-c语言

1. 线性表

线性表的定义:

零个多个数据元素组成的有限序列

注意

  • 首先它是一个序列,也就是说元素之间是有先来后到之分。
  • 若元素存在多个,则第一个元素无前驱,而最后一个元素无后继,其他元素都有且只有一个前驱和后继。
  • 线性表强调是有限的,事实上无论计算机发展到多强大,他所能处理的元素都是有限的。
线性表的形式化定义:
640.png
模拟考题:
  1. 请问公司的组织架构是否属于线性关系?
    分析:一般公司的总经理管理几个总监,每个总监管理几个经理,每个经理都有各自的下属和员工。
    那这样的组织架构当然不是线性关系了,事实上后面你就会知道,这是一个树状结构。注意线性关系的条件是如果存在多个元素,则第一个元素无前驱,而最后一个元素无后继,其他元素都有且只有一个前驱和后继。
  2. 那么班里同学之间的友谊是否属于线性关系?
    当然也不是了,因为每个人都会跟许多同学建立纯纯的友谊关系。
  3. 情侣之间的爱情关系呢?
    当然是扯淡,这要是线性关系,也就不会有所谓的第三者插足了。。。你们的爱情关系定会是线性关系!
  4. 最后一题,一个班级的点名册,是不是线性表?看下表:
学号 姓名 性别 职位
1 张三 班长
2 花花 副班长
3 丽丽 音乐课代表
4 萱萱 美术课代表
5 明明 小组长
线性表的分类:
640 (1).png

2. 数组:

数组就是 存储在连续内存空间 的数据项的集合,目的就是将多个 具有相同数据类型 的数据存储在一段连续的内存空间。这样就可以很容易的通过数组的 首地址偏移量(offset) 计算数组中每一个元素的地址,数组的第一个元素的存储位置是整个数组的首地址(通常由数组的名称表示),默认起始地址的索引为 0,两个索引之间的差值为偏移量。

图 1-1

想象一下,你(0 号)站在地面,你的若干个朋友(编号 1,2,3,4,5)分别站在一段楼梯的不同台阶上,那么你只需要知道你的朋友上了几个台阶,就可以确定他(她)的位置。

那么你就是首地址(索引为 0),而你与你的每一位朋友之间的台阶数就是该朋友相对于你的偏移量,比如 4 号朋友的偏移量就是 4 (两个索引的差值),你要找到你的 4 号朋友就需要上 4 个台阶。

为什么是相同数据类型呢?因为只有相同数据类型才可以通过首地址和偏移量来确定其他元素的地址。

图 1-2

图 1-2 可以看作是 图 1-1 楼梯的俯视图,您在楼梯的底部。数组中的每个元素都可以通过索引唯一标识(就像您在上面的示例中通过上台阶的步数确定朋友的方式类似)。

数组大小

在 C 语言中,数组具有固定的大小。一旦指定大小,便无法更改,既无法缩小,也无法扩展。因为,如果我们对数组进行扩展,我们无法确保获得的下一个内存位置是空闲的(可分配的)。对数组进行缩小也不起作用,因为数组在声明时会获得一块静态的内存空间,只有编译器才可以销毁它。

数组中的索引类型:

0 (基于 0 的索引):数组的第一个元素用下标 0 索引。1 (基于 1 的索引):数组的第一个元素被下标 1 索引。n (基于 n 的索引):数组的基索引可以自由选择。通常编程语言允许基于 n 的索引,也允许负索引值和其他标量数据类型,如枚举,或者字符可以用作数组索引。

注意:我们的文章中的数组的第一个元素的下标均为 0 。

数组中的元素存储在连续的内存空间。

C 语言测试数组中元素存储在连续内存空间的 demo:

/*
 * C 程序说明数组存储在连续的内存空间
*/
#include <stdio.h>
int main()
{
    /* 包含 10 个整数的整型数组
     * 如果 arr[0] 存储在地址 x,则 arr[1] 存储在 x+sizeof(int),
     * arr[2] 存储在 arr[1] 的地址 + sizeof(int)
    */
    int arr[10]; // 大家也可以换成char double float long等类型
    int i; 

    printf("编译器中整型的大小:%lu\n",sizeof(int));

    for (i = 0; i < 10; i++)
        // '&' 表示取变量的地址.
        printf("arr[%d] 的地址为: %p\n", i, &arr[i]);

    return 0;
}

输出:

编译器中整型的大小:4
arr[0] 的地址为: 0028FF14
arr[1] 的地址为: 0028FF18
arr[2] 的地址为: 0028FF1C
arr[3] 的地址为: 0028FF20
arr[4] 的地址为: 0028FF24
arr[5] 的地址为: 0028FF28
arr[6] 的地址为: 0028FF2C
arr[7] 的地址为: 0028FF30
arr[8] 的地址为: 0028FF34
arr[9] 的地址为: 0028FF38

3. 队列

队列(queue)是只允许在一端进行插入操作,而在另一端进行删除操作的线性表。

与栈相反,队列是一种先进先出(First In First Out,FIFO)的线性表。


640.gif

队列的链式存储结构队列既可以用链表来实现,也可以用顺序表来实现,跟栈相反的是,栈一般我们用顺序表来实现,而队列我们常用链表来实现,简称链队列

typedef struct QNode {
    ElemType data;
    struct QNode *next;
}QNode, *QueuePrt;

typedef struct {
    QueuePrt front,rear; //对头与队尾指针
} LinkQueue;

我们将队头指针指向链队列的头结点,而队尾指针指向终端结点。(注:头结点不是必须的,但是为了方便操作,我们加上了)


640.png

空队列的时候,front和rear都指向头结点


640 (1).png

创建一个队列

创建一个队列要完成两个任务:一是在内存中创建一个头结点,二是将队列的头指针和尾指针都指向这个生成的头结点,此时生成一个空队列。

initQueue(LinkQueue *q)
{
    q->front=q->rear=(QueuePtr)malloc(sizeof(QNode));
    if( !q->front)
        exit(0)
    q->front->next = NULL;
}

入队列操作

入队列操作过程如下:


640 (1).gif

出队列操作

出队列操作就是将队列中的第一个元素移动,队头指针不发生变化,改变头结点的next指针即可。

出队列的操作过程如下:


640 (2).gif

如果原队列中只有一个元素,那么我们就应该处理一下队尾指针。

640 (3).gif
DeleteQueue(LinkQueue *q, ElemType *e)
{
    QueuePtr p;
    if( q->front == q->rear )
    {
        return;
    }
    p = q->front->next;
    *e = p->data;
    q->front->next = p->next;
    if( q->rear == p)
        q->rear = q->front;
    free(p);
}
销毁一个队列

由于链队列建立在内存的动态区,因此当一个队列不再有用时应当把它销毁掉,以免过多地占用内存空间。

DestroyQueue(LinkQueue *q)
{
    while(q->front) {
        q->rear = q->front->next;
        free(q->front);
        q->front=q->rear;
    }
}

4. 树

4.1 树的概念

树(Tree)是n(n>=0)个结点的有限集。当n=0时成为空树,在任意一棵非空树中:

有且仅有一个特定的称为根(Root)的结点;

当n>1时,其余结点可分为m(m>0)个互不相交的有限集T1、T2、...、Tm,其中每一个集合本身又是一棵树,并且称为根的子树(SubTree)。

1702348784904.jpg

n>0时,根结点是唯一的,坚决不可能存在多个根结点。
m>0时,子树的个数是没有限制的,但它们互相是一定不会相交的。

比如,下图中的就不符合树的定义:


image.png

树中结点的分类:


image.png

如图中所示,每一个圈圈我们就称为树的一个结点。结点拥有的子树数称为结点的度-(Degree),树的度取树内各结点的度的最大值。
  • 度为0的结点称为叶结点(Leaf)或终端结点;
  • 度不为0的结点称为分支结点或非终端结点,除根结点外,分支结点也称为内部结点。

树中结点之间的关系:

image.png
  • 结点的子树的根称为结点的孩子(Child),相应的,该结点称为孩子的双亲(Parent),同一双亲的孩子之间互称为兄弟(Sibling)。

  • 结点的祖先是从根到该结点所经分支上的所有结点

结点的层次:

image.png

  • 结点的层次(Level)从根开始,根为第一层,根的孩子为第二层。

  • 其双亲在同一层的结点互为堂兄弟。

  • 树中结点的最大层次称为树的深度(Depth)或高度。

4.2二叉树

世上树有万千种,唯有二叉课上讲。这里的二叉是二叉树,因为二叉树使用的范围最广,最具有代表意义,因此不论教材还是考试都以二叉树为主。

二叉树(Binary Tree)是n(n>=0)个结点的有限集合,该集合或者为空集(空二叉树),或者由一个根结点和两棵互不相交的、分别称为根结点的左子树和右子树的二叉树组成。

这个定义显然是递归形式的,所以咱看上去有点晕,因为自古有“神使用递归,人使用迭代!“ 递归在树的相关操作中简直方便的不要不要!

image.png

二叉树的特点

  • 每个结点最多有两棵子树,所以二叉树中不存在度大于2的结点。(注意:不是都需要两棵子树,而是最多可以有两棵,没有子树或者有一棵子树也都是可以的。)

  • 左子树和右子树是有顺序的,次序不能颠倒。

  • 即使树中某结点只有一棵子树,也要区分它是左子树还是右子树,下面是完全不同的二叉树:

    image.png

    二叉树的五种基本形态
    image.png

若只从形态上来考虑,拥有三个结点的普通树只有两种情况:两层或者三层。而对于二叉树来说,由于要区分左右,所以就演变成五种形态(这也是为什么考试面试笔试总选择二叉树进行考察):


image.png

满二叉树
如果二叉树中除了叶子结点,每个结点的度都为 2,则此二叉树称为满二叉树。

image.png

满二叉树除了满足普通二叉树的性质,还具有以下性质(请注意图中的标注):

  1. 满二叉树中第 n 层的节点数为 2^{n-1} 个。

  2. 深度为 k 的满二叉树必有 2^{k}-1 个节点 ,叶子数为 2^{k-1}

  3. 满二叉树中不存在度为 1 的节点,每一个分支点中都两棵深度相同的子树,且叶子节点都在最底层。

完全二叉树
如果二叉树中除去最后一层节点为满二叉树,且最后一层的结点依次从左到右分布,则此二叉树被称为完全二叉树。

image.png

完全二叉树除了具有普通二叉树的性质,它自身也具有一些独特的性质,比如说,n 个结点的完全二叉树的深度为 ⌊log_2n⌋+1(后面有证明)。

二叉树的性质

image.png

  • 二叉树的性质一:在二叉树的第i层上至多有2^{i-1}个结点(i>=1)

  • 二叉树的性质二:深度为k的二叉树至多有2^k-1个结点(k>=1)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容