单链表,双链表,双指针的学习

单链表:生成,插入,删除

1.单链表定义
单链表的节点不仅包含值,还包含前往下一节点的引用字段,用自己的话来说就是,单链表的每个节点都包含了节点本身的值Val,还有指向下一节点的指针next;
在这里插入图片描述
大多数情况使用头指针来表示整个链表,因而查询的时候链表的时间复杂度为o(N)。

2.链表加入节点
链表中加入节点,可以理解为一个项链中加入一个小部件,那我们需要做的就是将其中的一个节点拆开,然后将前一个节点的next指针指向当前节点,并把当前节点的指针next指向下一个节点。
在这里插入图片描述
同为存储结构,这就是链表和数组的不同,链表在查询的时候的时间复杂度是o(N),但是在添加和删除节点上的时间复杂度为o(1),这就是链表和数组最大的不同。

3.链表删除节点
链表删除节点,我们同样拿项链来理解,对于一个项链,我们需要将其中一个部分删除掉,那我们需要做的就是将项链拆开,然后将上一个节点的末端指向下一个节点(相对于当前节点而言),就完成了指针的迁移。因为链表是链式的存储结构,因而我们不需要将当前节点的指针进行改变。
在这里插入图片描述
在删除头节点和末节点的时候,链表的操作会有所不同,只需要把链表的头节点传递给下一个节点即可。而对原来的头节点不需要做其余的改动。
在这里插入图片描述
对于末节点,我们同样只需要将倒数第二个节点的next指针指向NULL即可,不需要对原来的末尾节点进行操作。
单链表的C语言实现:
链表结构体初始化:
typedef struct node { int data; struct node *next; }*PNODE,NODE;
链表初始化:

PNODE init()
{
    PNODE temp = (PNODE)malloc(sizeof(NODE));//初始化一个头顶点。
    temp -> next = NULL;
    return temp;
}

初始化的目的是产生链表的一个顶点,链表的查询和增删节点都是由头节点进入,然后进行相应的操作。
链表的头部插入:

void insert(PNODE head,int newdata)//头部插入
{
    PNODE temp = (PNODE)malloc(sizeof(struct node));
    temp -> data = newdata;
    temp -> next = head -> next;
    head -> next = temp;
}

链表的尾部插入:

void insert(PNODE head,int newdata)//尾部插入
{
    PNODE p;
    while(p -> next != NULL) p = p -> next;
    p -> next = temp;    
    temp -> data = newdata;
    temp -> next = NULL;
}

输出整个链表值:

void print(PNODE head)
{
    PNODE temp = head -> next;
    while(temp->next != NULL)
    {
        printf("%d->",temp->data);
        temp = temp->next;
    }
    printf("NULL\n");//将链表每个结点的值进行打印
}

最后释放内存:

void freeAllNode(PNODE head)
{
    PNODE p = head,temp;
    while(p != NULL)
    {
        temp = p->next;
        free(p);
        p = temp;
    }
}

主函数内容:
主函数内进行了简单的初始化,并将链表的内容进行了输出。输出的原理也比较简单,就是对链表内的每个节点进行一次遍历,然后输出对应的节点值即可。

int main(){
    PNODE head = init();
    for( int i = 0 ; i < 10 ; i++ )
        insert(head,i);
    print(head);
    DeleteNode(head,0);
    print(head);
    freeAllNode(head);
    return 0;
}

删除节点:
删除节点就是将上一个节点的next指针指向下一个节点的地址,(这里可能会考虑到头尾指针的特殊性,可能有人想对头尾指针分别进行一次讨论,但是因为指针是直接对地址进行操作的,我们不需要再次进行讨论,只需要保证前面的tail指针正常就行)

void DeleteNode(PNODE head,int data)
{
    PNODE p = head,tail;//上一个节点和下一个节点怎么表示?
    PNODE temp;
    while(p != NULL){
        if(p->data == data)
        {
            temp = p;//要删除的节点
            p = p->next;//删了之后后移
            tail->next = p;//上一个节点后移
            free(temp);
        }       //由于指针是直接对地址进行操作,所以不用分类讨论头节点和尾节点的情况。
        else{
            tail = p;
            p = p->next;
        }
    }
}

输出:
在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值