数据结构:顺序表、链表篇
- 线性表
- 一、顺序表
- (一)顺序表的结构定义
- (二)顺序表的功能实现
- 1、初始化
- 2、销毁
- 3、扩容
- 4、插入
- 5、删除
- (三)顺序表例题分析
- 1、删除有序数组中的重复项
- 2、合并两个有序数组
- (四)顺序表的弊端
- 二、链表
- (一)链表的结构定义
- (二)链表的功能实现
- 1、链表的初始化
- 2、链表的插入
- 3、链表的删除
- 4、链表的销毁
- (三)链表的例题分析
- 1、移除链表元素
- 2、反转链表
- 题目分析
- 3、链表的中心结点
- 题目分析
- 4、合并两个有序链表
- 题目分析
- 5、链表的回文结构
- 题目分析
- 方法一
- 方法二
- 6、相交链表
- 题目分析
- 7、环形链表
- 题目分析
- 8、环形链表||
- 题目分析
- 9、随机链表的复制
- 题目分析
- 结束语
线性表
线性表:线性表是具有逻辑结构是连续,物理结构不一定是连续的一类数据结构的集合。链表和顺序表都是线性表
顺序表 : 物理结构连续,逻辑结构连续
链表 : 物理结构不一定连续(动态内存申请的空间可能是连续的,但是一般不会), 逻辑结构连续
一、顺序表
用物理地址连续的存储单元依次储存数据结构的线性结构,一般采用数组实现。顺序表分为动态顺序表和静态顺序表,为了防止空间过度浪费,空间不足,我们一般采用动态顺序表。
(一)顺序表的结构定义
typedef int SLdataType;typedef struct SeqList {SLdataType* data;int count, size;
}SeqList;
用到typedef, 可以使得我们的顺序表存放数据的类型更加的灵活。
(二)顺序表的功能实现
1、初始化
void initSeqList(SeqList* SL) {SL->data = NULL;SL->count = SL->size = 0;return;
}
2、销毁
void clearSeqList(SeqList* SL) {if (SL == NULL) return;if (SL->data) free(SL->data);SL->data = NULL;SL->count = SL->size = 0;return;
}
3、扩容
采用 realloc 进行扩容,考虑到原来的容量为 0, 不可单纯的进行乘二
void SLCheckCapacity(SeqList* SL) {if (SL->count == SL->size) {int n = SL->count == 0 ? 4 : 2 * SL->size;SLdataType* temp = (SLdataType*)realloc(SL->data, sizeof(SLdataType) * n);if (temp == NULL) {perror("realloc fail\n");exit(1);}SL->data = temp;SL->size *= n;}return;
}
4、插入
插入操作分为头插、尾插,和任意位置插入。
插入操作需要整体后移 : 从后面像前面遍历,反之会产生数据的覆盖。
//头插
void insertPushFront(SeqList* SL, SLdataType x) {SLCheckCapacity(SL);for (int i = SL->count - 1; i >= 0 ; i--) {SL->data[i + 1] = SL->data[i];}SL->data[0] = x;SL->count += 1;return;
}//尾插
void insertPushBack(SeqList* SL, SLdataType x) {SLCheckCapacity(SL);SL->data[SL->count++] = x;return;
}//任意位置插入
void insert(SeqList* SL, SLdataType x, int pos) {if (pos < 0 && pos > SL->count) return;SLCheckCapacity(SL);for (int i = SL->count - 1; i >= pos; i--) {SL->data[i + 1] = SL->data[i];}SL->data[pos] = x;SL->count += 1;return;
}
5、删除
删除操作分为头删、尾删,和任意位置删除。
删除操作需要整体前移 : 从前面向后面遍历,反之会产生数据的覆盖。
//头删
void erasePopFront(SeqList* SL) {for (int i = 1; i < SL->count; i++) {SL->data[i - 1] = SL->data[i];}SL->count -= 1;return;
}//尾删
void erasePopBack(SeqList* SL) {assert(SL);assert(SL->count);SL->count -= 1;return;
}//任意位置删除
void erase(SeqList* SL, int pos) {if (pos < 0 && pos >= SL->count) return;for (int i = pos; i < SL->count - 1; i++) {SL->data[i] = SL->data[i + 1];}SL->count -= 1;return;
}
(三)顺序表例题分析
1、删除有序数组中的重复项
https://leetcode.cn/problems/remove-duplicates-from-sorted-array/
class Solution {
public:int removeDuplicates(vector<int>& nums) {int src = 1, dst = 0;while(src < nums.size()){if(nums[src] == nums[dst]){src += 1;}else{nums[++dst] = nums[src++];}}return dst + 1;}
};
题目中我们用到双指针指针删除重复项,其中while 循环的条件设计十分巧妙
2、合并两个有序数组
https://leetcode.cn/problems/merge-sorted-array/
小结 : 采用两个指针分别指向两个数组的末尾,依次将数据放在数组一。
while 循环可以用 && 也可以用 || 采用两种代码实现
采用 || 的方式实现
class Solution {
public:void merge(vector<int>& nums1, int m, vector<int>& nums2, int n) {int l1 = m - 1, l2 = n - 1, l3 = m + n - 1;while (l1 >= 0 || l2 >= 0) {if (l2 < 0 || (l1 >= 0 && nums1[l1] > nums2[l2]))nums1[l3--] = nums1[l1--];elsenums1[l3--] = nums2[l2--];}}
};
或者采用 && 的方式实现
class Solution {
public:void merge(vector<int>& nums1, int m, vector<int>& nums2, int n) {int l1 = m - 1, l2 = n - 1, l3 = m + n - 1;while (l1 >= 0 && l2 >= 0) {if (nums1[l1] > nums2[l2])nums1[l3--] = nums1[l1--];elsenums1[l3--] = nums2[l2--];}while (l2 >= 0) {nums1[l3--] = nums2[l2--];}}
};
(四)顺序表的弊端
1、顺序表的插入删除操作的时间复杂度为O(n)
2、顺序表扩容后任然可能造成空间的浪费,并且顺序表扩容带来性能消耗
二、链表
(一)链表的结构定义
这里也用的typedef 可以使得我们的数据类型更加灵活。
typedef int SLTDataType;typedef struct SListNode {SLTDataType data;struct SListNode* next;
}SLTNode;
(二)链表的功能实现
1、链表的初始化
同顺序表相同,链表在初始化时也采取泛型的方式,适应多种数据类型。
SLTNode* BuyNode(SLTDataType x) {SLTNode* p = (SLTNode*)malloc(sizeof(SLTNode));p->data = x;p->next = NULL;return p;
}
2、链表的插入
链表的插入分为头插、尾插和任意位置插入。任意位置插入时采用双指针定向移动。
//尾插
void SLTPushBack(SLTNode** pphead, SLTDataType x) {SLTNode* node = BuyNode(x);if (*pphead == NULL) {*pphead = node;return;}SLTNode* p = *pphead;while (p->next) p = p->next;p->next = node;return;
}//头插
void SLTPushFront(SLTNode** pphead, SLTDataType x) {SLTNode* node = BuyNode(x);if (*pphead == NULL) {*pphead = node;return;}node->next = *pphead;*pphead = node;return;
}//任意位置之前插入,采用双指针的方式
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x) {assert(pos);assert(*pphead);SLTNode* node = BuyNode(x);if (*pphead == pos) {node->next = pos;*pphead = node;return;}SLTNode* fast = (*pphead)->next, * slow = *pphead;while (fast != pos) {fast = fast->next;slow = slow->next;}slow->next = node;node->next = fast;return;
}//任意位置之后插入方式会大大简便
void SLTInsertAfter(SLTNode* pos, SLTDataType x) {assert(pos);SLTNode* node = BuyNode(x);node->next = pos->next;pos->next = node;return;
}
3、链表的删除
链表的插入分为头删、尾删和任意位置删除。任意位置删除时采用双指针定向移动。
//尾删
void SLTPopBack(SLTNode** pphead) {assert(*pphead);if (!(*pphead)->next) {free(*pphead);*pphead = NULL;return;}SLTNode* ptail = *pphead;SLTNode* prev = NULL;while (ptail->next){prev = ptail;ptail = ptail->next;}prev->next = NULL;free(ptail);ptail = NULL;return;
}//头删
void SLTPopFront(SLTNode** pphead) {assert(*pphead);SLTNode* node = (*pphead)->next;free(*pphead);*pphead = node;return;
}//任意位置删除
void SLTErase(SLTNode** pphead, SLTNode* pos) {assert(*pphead);assert(pos);if (pos == *pphead) {SLTPopFront(pphead);}else {SLTNode* p = *pphead;while (p->next != pos) {p = p->next;}p->next = pos->next;free(pos);pos = NULL;}return;
}
4、链表的销毁
存储链表的下一个结点,然后进行 free
void SListDestroy(SLTNode** pphead)
{assert(pphead && *pphead);SLTNode* pcur = *pphead;while (pcur){SLTNode* next = pcur->next;free(pcur);pcur = next;}*pphead = NULL;
}
(三)链表的例题分析
1、移除链表元素
https://leetcode.cn/problems/remove-linked-list-elements/
class Solution {
public:ListNode* removeElements(ListNode* head, int val) {ListNode* newhead = NULL, *tail = NULL, *p = head;while(p){if(p->val != val){if(newhead == NULL) {newhead = tail = p;}else{tail->next = p;tail = tail->next;}}p = p->next;}if(tail) tail->next = NULL;return newhead;}
};
小结: 如果 tail 不是NULL, 要将 tail 置空。
2、反转链表
https://leetcode.cn/problems/reverse-linked-list/
题目分析
class Solution {
public:ListNode* reverseList(ListNode* head) {if(head == NULL) return NULL;ListNode* pre = NULL, * cur = head, * Next = head->next;while(cur){cur->next = pre;pre = cur;cur = Next;if(!cur) break;Next = Next->next;}return pre;}
};
小结:与另外新建一个链表的方式不同,这种算法可以在原来的链表上进行处理就能达到反转链表的效果。
3、链表的中心结点
https://leetcode.cn/problems/middle-of-the-linked-list/description/
题目分析
采用快慢指针进行分析,快指针每次走两步,慢指针每次走一步,当
fast = NULL 或者 fast -> next = NULL 时,slow指针指向的就是中间位置的指针。
class Solution {
public:ListNode* middleNode(ListNode* head) {ListNode* fast, *slow;fast = slow = head;while(fast && fast->next){fast = fast->next->next;slow = slow->next;}return slow;}
};
小结: 用快慢指针有很多好处,这道题的中间值就是一个
4、合并两个有序链表
https://leetcode.cn/problems/merge-two-sorted-lists/description/
题目分析
这道题的思路并不困难,主要是学一种新的头节点创建方式,通过 malloc 申请内存来获得头节点。
class Solution {
public:ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {ListNode* newhead, * tail;newhead = tail = (ListNode*)malloc(sizeof(ListNode));newhead->next = NULL;while(list1 && list2){if(list1->val < list2->val){tail->next = list1;tail = tail ->next;list1 = list1->next;}else{tail->next = list2;tail = tail->next;list2 = list2->next;}}if(list1) tail->next = list1;if(list2) tail->next = list2;ListNode* ret = newhead->next;free(newhead);newhead = NULL;return ret;}
};
小结:这道题可以有三种方式创建头结点
1、直接开辟变量 Node head, 返回head->next;
2、创建两个指针Node* head, * tail;
3、用 malloc 开辟空间,返回malloc 的下一个结点, 记得要free;
5、链表的回文结构
https://www.nowcoder.com/practice/d281619e4b3e4a60a2cc66ea32855bfa?tpId=49&&tqId=29370&qru=/ta/2016test/question-ranking
题目分析
这道题有两个思路:
方法一 : 采用数组,将链表的结点数据依次放入数组之中,然后创建两个指针向中间移动,依次比较。但是创建数组的时间复杂度为O(n),不可以通过。
方法二 : 中间结点后面的结点进行反转,切记反转链表反转的是指针的方向,数据位置没有变。然后同一。
方法一
class PalindromeList {
public:bool chkPalindrome(ListNode* A) {int arr[1000], i = 0;ListNode* p = A;while (p) {arr[i++] = p->val;p = p->next;}int left = 0, right = i - 1;while(left <= right) {if(arr[left++] != arr[right--]) return false;}return true;}
};
方法二
这种在原链表上进行修改的反转操作有妙用
6、相交链表
https://leetcode.cn/problems/intersection-of-two-linked-lists/description/
题目分析
将长的链表先截成和短的链表的长度,因为是后面部分相交,挨个比较直到两个指针的地址相等为相交结点
class Solution {
public:ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {ListNode* p = headA;int lenA = 0, lenB = 0;while(p){p = p->next;lenA += 1;}p = headB;while(p){p = p->next;lenB += 1;}int length = abs(lenA - lenB);ListNode* longlist = headA;ListNode* shortlist = headB;if(lenA < lenB){longlist = headB;shortlist = headA;}while(length--){longlist = longlist->next;}while(longlist != shortlist){longlist = longlist->next;shortlist = shortlist->next;}return shortlist;}
};
7、环形链表
https://leetcode.cn/problems/linked-list-cycle/
题目分析
用快慢指针,如果有环,那么快指针就会追上慢指针。
问题一 : 快指针为什么一定会追上慢指针
因为每次追逐两个指针的距离都会减一
问题二:快指针每次可以走2, 3, 4 ~步吗
下面我们以快指针每次走三步为例,结果是一定相遇,其他推理结论相同
class Solution {
public:bool hasCycle(ListNode *head) {ListNode* fast ,* slow;fast = slow = head;while(fast && fast->next){fast = fast->next->next;slow = slow->next;if(fast == slow) return true;}return false;}
};
小结:上面的文章里面,我们用快慢指针找中间结点,现在又多了一种新的用法,用来判断是否有环。
8、环形链表||
https://leetcode.cn/problems/linked-list-cycle-ii/description/
题目分析
在相遇之后,相遇点和头结点到环的起始点的距离相等,用两个指针从这两个位置同时出发,直到相遇。
9、随机链表的复制
https://leetcode.cn/problems/copy-list-with-random-pointer/description/
题目分析
如下图
步骤一 : 添加复制的结点
步骤二: 给random 赋值
步骤三: 断开原来的链表和拷贝链表
class Solution {
public:Node* BuyNode(int val) {Node* p = (Node*)malloc(sizeof(Node));p->val = val;p->next = p->random = NULL;return p;}void AddNode(Node* head) {Node *pcur = head, *next;while (pcur) {next = pcur->next;Node* node = BuyNode(pcur->val);pcur->next = node;node->next = next;pcur = next;}return;}Node* SetRandom(Node* head) {Node* pcur = head;while (pcur) {Node* temp = pcur->next;if (pcur->random) {temp->random = pcur->random->next;}pcur = pcur->next->next;}return head;}Node* getNewLinkList(Node* head) {Node *newHead, *tail;Node* pcur = head;newHead = tail = head->next;while (pcur) {pcur->next = tail->next;if (tail->next) {tail->next = pcur->next->next;tail = tail->next;}pcur = pcur->next;}// tail->next = NULL; // 确保新链表的尾部正确return newHead;}Node* copyRandomList(Node* head) {if (head == NULL)return NULL;AddNode(head);head = SetRandom(head);head = getNewLinkList(head);return head;}
};
小结:无需多言,值得反复学习
结束语
好了,小编也要睡觉了,下一篇小编会带来双向链表的博文。如果感兴趣的话记得要给博主一个关注哦,不然就再也找不到啦,小伙伴们周末快乐!