单链表:生成,插入,删除
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;
}
}
}
输出: