数据结构:程序加图示分析单链表的插入和删除操作

下图展示了单链表的基本结构:

head指针是链表的头指针,指向第一个节点,每个节点的next指针域指向下一个节点,最后一个节点的next指针域为NULL,在图中用0表示。

下面先来看程序(栈的链式存储实现,另外一个实现点这里)和对应的输出(注意输出前进行了链表反转(见《单链表反转》,否则程序后面的while循环输出的顺序是250,200,100),接着来分析程序:

/* linkedlist.h */

#ifndef LINKEDLIST_H
#define LINKEDLIST_H

typedef struct node *link;
struct node
{
    unsigned char item;
    link next;
};

link make_node(unsigned char item);
void free_node(link p);
link search(unsigned char key);
void insert(link p);
void deletep(link p);
void traverse(void (*visit)(link));
void reverse(void);
void destroy(void);
void push(link p);
link pop(void);

#endif
/* linkedlist.c */
#include <stdlib.h>
#include <stdio.h>
#include "linkedlist.h"

static link head = NULL;

link make_node(unsigned char item)
{
    link p = malloc(sizeof(*p));
    p->item = item;
    p->next = NULL;
    printf("make node from Item %d\n", item);
    return p;
}

void free_node(link p)
{
    printf("free node ...\n");
    free(p);
}

link search(unsigned char key)
{
    link p;
    printf("search by key %d\n", key);
    for (p = head; p; p = p->next)
        if (p->item == key)
            return p;
    return NULL;
}

void insert(link p)
{
    printf("insert node from head ...\n");
    p->next = head;
    head = p;
}

/*
void delete(link p)
{
    link pre;
    printf("delete node from ptr ...\n");
    if (p == head) {
        head = p->next;
        return;
    }
    for (pre = head; pre; pre = pre->next) 
        if (pre->next == p) {
            pre->next = p->next;
            return;
     }
}
*/

void deletep(link p)
{
    link *pnext;
    printf("delete node from ptr ...\n");
    for (pnext = &head; *pnext; pnext = &(*pnext)->next)
        if (*pnext == p)
        {
            *pnext = p->next;
            return;
        }
}

void traverse(void (*visit) (link))
{
    link p;
    printf("linkedlist traverse ...\n");
    for (p = head; p; p = p->next)
        visit(p);
}

void reverse(void)
{
    link pnode = head;
    link pprev = NULL;
    printf("reverse linkedlist ...\n");
    while (pnode != NULL)
    {
        // get the next node, and save it at pNext
        link pnext = pnode->next;
        // reverse the linkage between nodes
        pnode->next = pprev;
        // move forward on the the list
        pprev = pnode;
        pnode = pnext;

    }
    head = pprev;
}

void destroy(void)
{
    link q, p = head;
    printf("destory linkedlist ...\n");
    head = NULL;
    while (p)
    {
        q = p;
        p = p->next;
        free_node(q);
    }
}

void push(link p)
{
    printf("push item from head ...\n");
    insert(p);
}

link pop(void)
{
    if (head == NULL)
        return NULL;
    else
    {
        link p = head;
        printf("pop item from head ...\n");
        head = head->next;
        return p;
    }
}
/*************************************************************************
    > File Name: main.c
    > Author: Simba
    > Mail: dameng34@163.com
    > Created Time: Fri 28 Dec 2012 01:22:24 PM CST
 ************************************************************************/

#include<stdio.h>
#include "linkedlist.h"

void print_item(link p)
{
    printf("print item %d \n", p->item);
}

int main(void)
{
    link p = make_node(10);
    insert(p);
    p = make_node(5);
    insert(p);
    p = make_node(90);
    insert(p);
    p = search(5);
    deletep(p);
    free_node(p);
    traverse(print_item);
    destroy();
    printf("..................\n");
    p = make_node(100);
    push(p);
    p = make_node(200);
    push(p);
    p = make_node(250);
    push(p);

    reverse();//链表反转

    while ((p = pop()))
    {

        print_item(p);
        free_node(p);
    }

    return 0;
}

输出为:

分析:

在初始化时把头指针head初始化为NULL,表示空链表(不带头结点)。然后main函数调用make_node创建几个节点,分别调用insert插入到链表中。

链表的插入操作如下图:

正如上图所示,insert函数虽然简单,其中也隐含了一种特殊情况(Special Case)的处理,当head为NULL时,执行insert操作插入第一个节点之后,head指向第一个节点,而第一个节点的next指针域成为NULL,这很合理,因为它也是最后一个节点。所以空链表虽然是一种特殊情况,却不需要特殊的代码来处理,和一般情况用同样的代码处理即可,这样写出来的代码更简洁,但是在读代码时要想到可能存在的特殊情况。当然,insert函数传进来的参数p也可能有特殊情况,传进来的p可能是NULL,甚至是野指针,本章的函数代码都假定调用者的传进来的参数是合法的,不对参数做特别检查。事实上,对指针参数做检查是不现实的,如果传进来的是NULL还可以检查一下,如果传进来的是野指针,根本无法检查它指向的内存单元是不是合法的,C标准库的函数通常也不做这种检查,例如strcpy(p, NULL)就会引起段错误。

接下来main函数调用search在链表中查找某个节点,如果找到就返回指向该节点的指针,找不到就返回NULL。

search函数其实也隐含了对于空链表这种特殊情况的处理,如果是空链表则循环体一次都不执行,直接返回NULL。

然后main函数调用delete从链表中摘除用search找到的节点,最后调用free_node释放它的存储空间。

链表的删除操作如下图:

从上图可以看出,要摘除一个节点需要首先找到它的前趋然后才能做摘除操作,而在单链表中通过某个节点只能找到它的后继而不能找到它的前趋,所以删除操作要麻烦一些,需要从第一个节点开始依次查找要摘除的节点的前趋。delete操作也要处理一种特殊情况,如果要摘除的节点是链表的第一个节点,它是没有前趋的,这种情况要用特殊的代码处理,而不能和一般情况用同样的代码处理。这样很不爽,能不能把这种特殊情况转化为一般情况呢?可以把delete函数改成上述程序那样:

消除特殊情况的链表删除操作如下图:

定义一个指向指针的指针pnext,在for循环中pnext遍历的是指向链表中各节点的指针域,这样就把head指针和各节点的next指针统一起来了,可以在一个循环中处理。(其实增加一个头节点也可以消除delete的特殊情况《线性表的链式存储结构》) 然后main函数调用traverse函数遍历整个链表,调用destroy函数销毁整个链表。

参考:《linux c 编程一站式学习》

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏与神兽党一起成长

LinkedHashSet内部是如何工作的(翻译)

LinkedHashSet是HashSet的一个“扩展版本”,HashSet并不管什么顺序,不同的是LinkedHashSet会维护“插入顺序”。HashSet...

1014
来自专栏python成长之路

Fibonacci数列使用迭代器实现

1163
来自专栏小灰灰

JDK容器学习之LinkedHashMap (一):底层存储结构分析

LinkedHashMap 底层存储结构分析 HashMap 是无序的kv键值对容器,TreeMap 则是根据key进行排序的kv键值对容器,而LinkedH...

1795
来自专栏Python

Python 操作redis有序集合(sorted set)

Zadd 命令用于将一个或多个成员元素及其分数值加入到有序集当中。 如果某个成员已经是有序集的成员,那么更新这个成员的分数值,并通过重新插入这个成员元素,来保证...

1101
来自专栏深度学习自然语言处理

Python学习——collections系列

一 ,计数器(counter) Counter是对字典类型的补充,用于追踪值得出现次数 ps:具备字典的所有功能 + 自己的功能 例: >>> from ...

35614
来自专栏奔跑的蛙牛技术博客

Java中实现的简单算法 && 计算二分查找次数

如果采用其他方式对列表进行排序可以使用List接口的sort方法传入一个Comarable的一个对象

702
来自专栏和蔼的张星的图像处理专栏

138. 子数组之和 map存储加规律

给定一个整数数组,找到和为零的子数组。你的代码应该返回满足要求的子数组的起始位置和结束位置。 假定一定存在这样的字数组。 样例 给出 [-3, 1, 2,...

861
来自专栏青枫的专栏

C语言中%d,%p,%u,%lu等都有什么用处

%d   有符号10进制整数(%ld 长整型,%hd短整型 ) %hu   无符号短整形(%u无符号整形,%lu无符号长整形) %i    有符号10进制整数 ...

351
来自专栏测试开发架构之路

总结了一些指针易出错的常见问题(五)

指针与链表及其操作 //结构体定义 typedef struct _person{ char* firstname; char* lastna...

2635
来自专栏趣学算法

数据结构 第3讲 顺序表

顺序表是最简单的一种线性结构,逻辑上相邻的数据在计算机内的存储位置也是相邻的,可以快速定位第几个元素,中间不允许有空,所以插入、删除时需要移动大量元素。

943

扫码关注云+社区