【C】单链表

单链表

声明一个指向自身的结构体

struct Test
{
        int x;
        int y;
        struct Test test;
};

输出结果:

error!

程序报错,因为这样会造成无限的循环。当编译器解析到struct Test testtest是结构体Test的成员,定义test成员需要Test,而结构体Test自身又是不完整的,那么程序就无法定义一个结构体变量,所以会陷入无限的递归。修改方法如下:

struct Test
{
        int x;
        int y;
        struct Test *test;
};

test加上星号(*)将其变为一个指向Test结构体自身的指针变量

单链表

单链表是最简单的一种链表实现方式,它包含两个域,一个信息域和一个指针域:

信息域用来存放链表节点的内容(相当于数组里的内容)

指针域用来指向下一个和它一模一样的节点

当最后一个节点的指针域指向NULL时,表示单链表就此结束

head:真正的单链表它还需要一个头指针,用于存放指向链表第一个节点的地址

链表中各个元素在内存中不是紧密存放,而是通过指针连接在一起的

对于 Book 结构体来说,要把它变成链表的其中一个元素,我们只需要为其添加一个指向自身的成员即可:

struct Book
{
        char title[128]; //信息域
        char author[40];
        struct Book *next;
};

单链表中插入元素(头插法)

在单链表中插入元素,事实上只需要修改指针的指向即可:

将书籍添加到单链表的代码我们这么可以写:

#include<stdio.h>
#include<stdlib.h>
struct Book
{
        char title[128]; //信息域
        char author[40];

        struct Book *next;
};
void getInput(struct Book *boos);
void addBook(struct Book **library);
void printLibrary(struct Book *library)
void releaseLibrary(struct Book **library);

void getInput(struct Book *book)
{
    printf("请输入书名 : ");
    scanf("%s",book->title);
    printf("请输入作者 : ");
    scanf("%s",book->author);
}
void addBook(struct Book **library)
//要修改library(即head指针)的值,所以要传入library指针的地址
//即用二级指针**library(指向指针的指针)
{
        struct Book *book, *temp;

        book = (struct Book *)malloc(sizeof(struct Book));
        if (book == NULL)
        {
                printf("内存分配失败!\n");
                exit(1);
        }

        getInput(book);

        if (*library != NULL)
        {
                temp = *library;
                *library = book;
                book->next = temp;
        }
        else
        {
                *library = book;
                book->next = NULL;
        }
}
void printLibrary(struct Book *library)
{
    struct Book *book;
    int count = 1;
    book = library;
    while (book != NULL)
    {
        printf("Book%d: ", count);
        printf("书名 : %s", book->title);
        printf("作者 : %s", book->author);
        book = book->next;
        count++;
    }
}

void releaseLibrary(struct Book **library)
{
    struct Book *temp;
    while (library != NULL)
    {
        temp = *library;
        *library = (*library)->next;
        free(temp);//释放内存
    }
}

int main(void)
{
    struct Book *library = NULL;  //定义头指针,指向一个空的单链表
    addBook(&library);
    return 0;
}

尾插法

头插法,就是将数据插入到单链表的头部位置。相对应的还有另一个种方法:尾插法 —— 将数据插入到单链表的尾部位置

void addBook(struct Book **library)
{
        struct Book *book, *temp;

        book = (struct Book *)malloc(sizeof(struct Book));
        if (book == NULL)
        {
                printf("内存分配失败!\n");
                exit(1);
        }

        getInput(book);

        if (*library != NULL)
        {
            temp = *library;
            //定位单链表尾部位置
            while (temp->next != NULL)
            {
                temp = temp->next;
            }
            temp->next = book;
            //当temp->next == NULL时,插入book节点
            book->next = NULL;
        }
        else
        {
                *library = book;
                book->next = NULL;
        }
}

上面的程序虽然能成功运行,但每插入一次数据都要遍历一次链表,效率不高。

void addBook(struct Book **library)
{
    struct Book *book;
    static struct Book *tail;
    //指针tail用来记录单链表尾部位置
    //因为每调用一次addbook函数指针tail都会初始化
    //为了让指针tail永远记录上一次插入数据后的尾部位置
    //用static将tail转为静态指针变量

    book = (struct Book *)malloc(sizeof(struct Book));
    if (book == NULL)
    {
        printf("内存分配失败!\n");
        exit(1);
    }

    getInput(book);

    if (*library != NULL)
    {
        tail->next = book;
        book->next = NULL;
    }
    else
    {
        *library = book;
        book->next = NULL;
    }

    tail = book;
}

搜索单链表

有时候我们可能会对单链表进行搜索操作,比如输入这个书名或者作者的名字,可以找到相关的节点数据。

struct Book *searchBook(struct Book *library, char *target)
{
        struct Book *book;

        book = library;
        while (book != NULL)
        {
                if (!strcmp(book->title, target) || !strcmp(book->author, target))
                {
                        break;
                }
                book = book->next;
        }

        return book;
}

void printBook(struct Book *book)
{
        printf("书名:%s\n", book->title);
        printf("作者:%s\n", book->author);
}
...
int main(void)
{
        ...
        printf("\n请输入书名或作者:");
        scanf("%s", input);

        book = searchBook(library, input);
        if (book == NULL)
        {
                printf("很抱歉,没能找到!\n");
        }
        else
        {
                do
                {
                        printf("已找到符合条件的书籍...\n");
                        printBook(book);
                } while ((book = searchBook(book->next, input)) != NULL);
        }

        releaseLibrary(&library);

        return 0;
}
posted @ 2025-02-14 21:30  芝麻凛  阅读(19)  评论(0)    收藏  举报