链表结构
链表是一种以不连续的方式存储数据的数据结构。对于最基本的单链表,其每个节点不仅包含值val
,也包含一个指向下一个节点地址的指针next
。以C语言实现为例:
#include<stdio.h>
#include<stdlib.h>
struct ListNode
{
int val;
struct ListNode *next;
};
//判断链表为空
int isEmpty(struct ListNode *head)
{
return head==NULL;
}
//查找节点,找不到返回NULL
struct ListNode* find(int x, struct ListNode* head)
{
while(head && head->val!=x)
{
head=head->next;
}
return head;
}
//查找前驱节点
struct ListNode* findPrevious(int x, struct ListNode *head)
{
if(head->val==x)
{
return NULL;
}
struct ListNode* pre = head;
while(head && head->val!=x)
{
pre=head;
head=head->next;
}
return head==NULL?NULL:pre;
}
//插入节点
void insertNode(int x, struct ListNode*position)
{
struct ListNode *node = (struct ListNode *)malloc(sizeof(struct ListNode));
node->val=x;
struct ListNode *temp=position->next;
position->next=node;
node->next=temp;
}
//删除节点
void deleteNode(int x, struct ListNode *head)
{
//头删
if(head->val==x)
{
struct ListNode *next = head->next;
free(head);
*head=*next;
}
struct ListNode *pre = findPrevious(x,head);
if(pre)
{
struct ListNode *cur = pre->next;
pre->next=cur->next;
free(cur);
}
}
void printList(struct ListNode *head)
{
int idx = 0;
while(head)
{
printf("%d --- element val is %d \n",idx,head->val);
head=head->next;
}
}
void main()
{
struct ListNode *dumpy = (struct ListNode *)malloc(sizeof(struct ListNode));
dumpy->val=1;
dumpy->next=NULL;
printList(dumpy);
printf("is empty ? %s \n",isEmpty(dumpy)?"yes":"no");
insertNode(2,find(1,dumpy));
printf("after insert :\n");
printList(dumpy);
deleteNode(2,dumpy);
printf("after delete :\n");
printList(dumpy);
}
注意:free()
函数不会删除指针,而是将指针指向的内存释放。在删除节点的函数中,如果是头删,先获得后一个节点,即第二个节点,然后将头指针指向的内存释放,再将头指针指向第二个节点,*head=*next
.
链表算法题
解决链表的问题,一般可以有两种方法:迭代和递归。
题一:反转链表
核心思路:让节点的next
值指向上一个节点,这里需要同时保存当前节点和前一个节点,并同时移动。
- 双指针迭代法
遍历每个节点,并将当前节点的next
指向上一个节点。这里需要定义指针变量pre
保存前一个节点的指针,初始化为NULL
(因为反转后的尾节点就是反转前的首节点。),并定义指针变量cur
保存遍历到的节点。 当遍历完成时,pre
保存的是原链表的尾节点,也就是新链表的首节点。
//反转链表
struct ListNode * reverseList(struct ListNode *head)
{
//如果链表为空或者只有一个节点,直接返回原链表
if(!head || !head->next)
{
return head;
}
struct ListNode *cur = head;
struct ListNode *pre = NULL;
while(head)
{
head=head->next;
cur->next=pre;
pre=cur;
cur=head;
}
return pre;
}
- 递归
递归函数的作用是让节点的next
指向上一节点。
//递归反转
struct ListNode *reverseList2(struct ListNode *head)
{
if(!head || !head->next)
{
return head;
}
struct ListNode *temp = reverseList2(head->next);
head->next->next=head;
head->next=NULL;
return temp;
}
题二:合并连个有序链表
输入两个递增排序的链表,合并这两个链表并使新链表中的节点仍然是递增排序的。
- 迭代法
定义一个新链表,一次比较两个链表的值,取小的值加到新链表上,直到一个链表遍历完毕,将另一个链表的剩余部分加到新链表上。
//合并有序链表
struct ListNode *mergeList(struct ListNode * l1,struct ListNode* l2)
{
struct ListNode* l3 = (struct ListNode *)malloc(sizeof(struct ListNode));
struct ListNode* dumpy = l3;
while(l1 && l2)
{
if(l1->val<l2->val)
{
l3->next=l1;
l1=l1->next;
}
else
{
l3->next=l2;
l2=l2->next;
}
l3=l3->next;
}
l3->next=l1==NULL?l2:l1;
return dumpy->next;
}
- 递归
递归函数的作用是创建一个节点,依次比较两个链表值,取小的值作为新节点的值,next
指向的值也是通过比较所得。
struct ListNode* mergeList2(struct ListNode* l1,struct ListNode* l2)
{
if(l1==NULL)
{
return l2;
}
if(l2==NULL)
{
return l1;
}
struct ListNode *node = (struct ListNode *)malloc(sizeof(struct ListNode));
if(l1->val<l2->val)
{
node->val=l1->val;
node->next=mergeList2(l1->next,l2);
}
else
{
node->val=l2->val;
node->next=mergeList2(l1,l2->next);
}
return node;
}
迭代法和递归法是处理链表问题的常见方法,迭代相对更好理解。
更多习题,详见我的github