【数据结构】链表的大概认识及单链表的实现

目录

一、链表的概念及结构

二、链表的分类

三、单链表的实现

建立链表的节点:

尾插——尾删: 

头插——头删:

查找:

指定位置之后删除——插入:

指定位置之前插入——删除指定位置:

销毁链表:

打印:

四、链表面试题

五、总体代码


一、链表的概念及结构

概念:链表是一种 物理存储结构上非连续 、非顺序的存储结构,数据元素的 逻辑顺序 是通过链表
中的 指针链接 次序实现的 。

二、链表的分类

实际中链表的结构非常多样,以下情况组合起来就有 8 种链表结构:
1. 单向或者双向
2. 带头或者不带头

3. 循环或者非循环

虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:

1. 无头单向非循环链表: 结构简单 ,一般不会单独用来存数据。实际中更多是作为 其他数据结 构的子结构 ,如哈希桶、图的邻接表等等。另外这种结构在 笔试面试 中可能出现比较多。
2. 带头双向循环链表: 结构最复杂 ,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。

三、单链表的实现

建立链表的节点:

链表中的节点结构体大概有这些内容:节点数据,下一个节点的地址

typedef int SLTDateType;
typedef struct SListNode
{
 SLTDateType data;
 struct SListNode* next;
}SListNode;

尾插——尾删: 

不管是尾插还头插还是任意位置插入,我们都需要先创建一个新节点

尾插和尾删都需要注意一些点:

用二级指针接收:

因为我们这个是无头的链表,在链表没有一个节点的时候,我们需要创建一个节点。然后将链表的头指向这个新节点,这个时候就涉及到需要修改一级结构体的一级指针。一级指针类型的变量需要修改的话,就要用到二级指针。

尾插:是否链表一个数据都没有,这个时候我们要特殊处理一下,检查链表有效性

尾删:是否链表只有一个数据 ,检查是否有数据可以删除,检查链表有效性

// 动态申请一个节点
SListNode* BuySListNode(SLTDateType x)
{
	SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
	if (newnode == NULL)
	{
		perror("malloc fail");
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}
// 单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x)
{
	assert(pplist);
	SListNode* newnode = BuySListNode(x);
	if (*pplist == NULL)
	{
		*pplist = newnode;
		return;
	}
	//找尾巴
	SListNode* tail = *pplist;
	while (tail->next != NULL)
		tail = tail->next;
	tail->next = newnode;
}
// 单链表的尾删
void SListPopBack(SListNode** pplist)
{
	assert(pplist);
	assert(*pplist);//有数据才能删,没有数据不删,至少有一个及其以上节点
	if ((*pplist)->next == NULL)
	{
		free(*pplist);
		*pplist = NULL;
	}
	else
	{
		//找尾巴
		SListNode* prev = NULL;
		SListNode* tail = *pplist;
		while (tail->next != NULL)
		{
			prev = tail;
			tail = tail->next;
		}
		free(tail);
		prev->next = NULL;
	}
}

头插——头删:

 头插和头删在链表中还是比较简单,一个是把头节点保存起来,另一个是头节点的下一个节点保存起来,然后进行删除就行了。

头插:注意需要检查链表有效性

头删:注意这里需要检查数据个数,有数据才能删除;

// 单链表的头插
void SListPushFront(SListNode** pplist, SLTDateType x)
{
	assert(pplist);
	SListNode* newnode = BuySListNode(x);
	newnode->next = *pplist;
	*pplist = newnode;
}
// 单链表头删
void SListPopFront(SListNode** pplist)
{
	assert(pplist);
	assert(*pplist);
	SListNode* next = (*pplist)->next;
	free(*pplist);
	*pplist = next;
}

查找:

通过简单遍历就行了,注意返回的是节点的指针;

我们在外面定义的是一个链表节点的指针,打印的时候按照传值的方式传递变量,他会把变量的内容(链表中首节点的地址)拷贝过来,最终返回的也是链表节点的地址

// 单链表查找
SListNode* SListFind(SListNode* plist, SLTDateType x)
{
	assert(plist);
	SListNode* cur = plist;
	while (cur)
	{
		if (cur->data == x)
			return cur;
		cur = cur->next;
	}
	return NULL;
}

指定位置之后删除——插入:

注意:写这两个接口,我们首先就要想到的是检查 指定位置的有效性 还有检查 链表有效性

代码还是很简单的,这里的删除要检查是不是最后一个节点

// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x)
{
	assert(pos);
	SListNode* newnode = BuySListNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
}
// 单链表删除pos位置之后的值void SListEraseAfter(SListNode* pos)
{
	assert(pos);
    assert(pos->next);//检查是不是最后一个
	SListNode* next = pos->next;
	pos->next = next->next;
	free(next);
}

指定位置之前插入——删除指定位置:

这两个接口:我们需要像尾插那个样子 遍历找到指定位置前的位置:然后进行插入,删除,同时也要注意检查链表有效性,

注意: 如果链表只有一个位置或者指定位置为第一个节点,那删除就变成了头删,可以复用原来的头删接口。尾删则没有必要,我们已经遍历一遍找到尾了,不需要调用尾删接口再遍历一遍了,直接像尾删一样删除就行了。

// 在pos的前面插入
void SLTInsert(SListNode** pphead, SListNode* pos, SLTDateType x)
{
	assert(pphead);
    assert((!pos&&!(*pphead))||(pos&&(*pphead)))
	//这里我们让他都为空(头插)或者都不为空
    //我们想暴露出一个问题,不允许乱位置插入  限定pos一定是有效节点
	if (pos == *pphead)
	{
		SListPushFront(pphead, x);
		return;
	}
	SListNode* prev = *pphead;
	while (prev->next!=pos)
	{
		prev = prev->next;
	}
	SListNode* newnode = BuySListNode(x);
	newnode->next = pos;
	prev->next = newnode;
}
// 删除pos位置
void SLTErase(SListNode** pphead, SListNode* pos)
{
	assert(pphead);
	assert(*pphead);
	assert(pos);//这里进行检查pos,必须有这个节点才能删除
	if (pos == *pphead)
	{
		SListPopFront(pphead);
		return;
	}
	SListNode* prev = *pphead;
	while (prev->next != pos)
	{
		prev = prev->next;
	}
	prev->next = pos->next;
	free(pos);
	pos = NULL;
}

销毁链表:

这个按照遍历的同时free就行了。

注意:遍历完了,全部空间释放了把链表置空

void SLTDestroy(SListNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	SListNode* cur = *pphead;
	while (cur)
	{
		SListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	*pphead = NULL;
}

打印:

为了好测试写的接口是否正确,我们还是写一个打印接口,方便我们观察:同样的,循环遍历打印就行了;这里不需要检查链表是否为空,空链表也可以打印

// 单链表打印
void SListPrint(SListNode* plist)
{
	SListNode* cur = plist;
	while (cur)
	{
		printf("%d-> ", cur->data);
		cur = cur->next;
	}
	printf("\n");
}

四、链表面试题

1. 删除链表中等于给定值 val 的所有结点。  203. 移除链表元素 - 力扣(LeetCode)
这个题我们找到与 给定值相等的节点和他的前一个节点就可以进行删除了,但是我们要处理两种情况:
1:链表为空
2:需要删除头(需要循环删除,因为有可能连续好几个都需要删)

参考代码: 

struct ListNode* removeElements(struct ListNode* head, int val) {
    //处理链表为空
    if(!head) return NULL;
    //处理链表第一个就是该删的元素
    if(head->val==val)
    {
        while(head&&head->val==val)
        {
            struct ListNode* next=head->next;
            free(head);
            head=next;
        }
    }
    struct ListNode* cur=head;
    struct ListNode* prev=NULL;
    while(cur)
    {
        if(cur->val==val)
        {
            prev->next=cur->next;
            free(cur);
            cur=prev->next;
        }
        else
        {
            prev=cur;
            cur=cur->next;
        }
    }
    return head;
}
2. 反转一个单链表。  206. 反转链表 - 力扣(LeetCode)

 这个题,就是一个简单头插就行了

当然我们也可以用三个指针将链表的指向反转(注意检查链表是否为空),这里我们用两种方法实现,两种方法任选其一都能通过,

struct ListNode* reverseList(struct ListNode* head) {
    //头插处理
    // struct ListNode* newhead=NULL;
    // while(head)
    // {
    //     struct ListNode* next=head->next;
    //     head->next=newhead;
    //     newhead=head;
    //     head=next;
    // }
    // return newhead;

    //直接反转
    if(!head)
    return head;
    struct ListNode* n1=NULL,*n2=head,*n3=head->next;
    while(n2)
    {
        n2->next=n1;
        n1=n2;
        n2=n3;
        if(n3)
        n3=n3->next;        
    }
    return n1;
}
3. 给定一个带有头结点 head 的非空单链表,返回链表的中间结点。如果有两个中间结点,则返回第二个中间结点。 876. 链表的中间结点 - 力扣(LeetCode)
这个题是一个标准的 快慢指针

值得注意的是:我们判断条件是快指针为NULL或者快指针的下一个节点为NULL,他们在循环中判断的先后为先判断自己再判断下一个,因为先判断下一个的话,有可能出现野指针问题。 

struct ListNode* middleNode(struct ListNode* head) {
    //快慢指针法
    struct ListNode* fast=head;
    struct ListNode* slow=head;
    while(fast&&fast->next)
    {
        fast=fast->next->next;
        slow=slow->next;
    }
    return slow;
}

 4. 输入一个链表,输出该链表中倒数第k个结点。 面试题 02.02. 返回倒数第 k 个节点 - 力扣(LeetCode)

这个题也是一个经典的双指针,可以让快指针先走k次,然后快慢指针同时走,每次走一下:
int kthToLast(struct ListNode* head, int k){
    struct ListNode* fast=head;
    struct ListNode* slow=head;
    while(k--)
    fast=fast->next;
    while(fast)
    {
        fast=fast->next;
        slow=slow->next;
    }
    return slow->val;
}
5. 将两个有序链表合并为一个新的有序链表并返回。新链表是通过拼接给定的两个链表的所有结点组成的。 21. 合并两个有序链表 - 力扣(LeetCode)

 这个题就要用到并归和尾插了,选择小的尾插到新链表里面,这个尾插是移动他的链表节点到我们新的链表里面。

需要注意的是:1.需要处理空链表情况,一个为空或者两个为空

                         2.处理头节点

 

struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {
    //处理为空的情况  
    if(!list1)
    return list2;
    if(!list2)
    return list1;
    struct ListNode* head=NULL;
    struct ListNode* tail=NULL;
    while(list1&&list2)
    {
        if(list1->val<list2->val)
        {
            struct ListNode* next=list1->next;
            if(head==NULL)//处理头节点
            head=tail=list1;
            else
            {
                tail->next=list1;
                tail=tail->next;
                tail->next=NULL;//这个可以不处理,后面剩余节点的尾巴也一定是NULL
            }
            list1=next;
        }
        else
        {
            struct ListNode*next=list2->next;
            if(head==NULL)//处理头节点
            head=tail=list2;
            else
            {
                tail->next=list2;
                tail=tail->next;
                tail->next=NULL;
            }
            list2=next;
        }
    }
    if(list1==NULL)//处理剩余未插入的节点
    tail->next=list2;
    else
    tail->next=list1;
    return head;
}
6. 编写代码,以给定值 x 为基准将链表分割成两部分,所有小于 x 的结点排在大于或等于 x 的结
点之前 。 链表分割_牛客题霸_牛客网 (nowcoder.com)
这个题呢,我们用把链表分成两条,小于给定值的在一条,大于给定值的在一条,然后在连接起来就行了。
注意:这里我们要用到 带头的单链表,无头的单链表会有很多问题要处理。我们还需要将链表移动后 置空,不然后面连接起来可能会出现很多问题
class Partition {
  public:
    ListNode* partition(ListNode* pHead, int x) {
        ListNode* head1, *tail1, *head2, *tail2;
        head1 = tail1 = (ListNode*)malloc(sizeof(ListNode));//第一条链表
        head2 = tail2 = (ListNode*)malloc(sizeof(ListNode));//第二条链表
        tail1->next = tail2->next = NULL;
        while (pHead) 
        {
            if (pHead->val < x) 
            {
                ListNode* next = pHead->next;
                tail1->next = pHead;
                tail1 = tail1->next;
                tail1->next = NULL;
                pHead = next;
            } 
            else 
            {
                ListNode* next = pHead->next;
                tail2->next = pHead;
                tail2 = tail2->next;
                tail2->next = NULL;
                pHead = next;
            }
        }
        tail1->next=head2->next;
        ListNode* re=head1->next;
        free(head2);
        free(head1);
        return re;
    }
};

7. 链表的回文结构。OJ链接

这个题,我们需要找到中间节点(先遍历找出节点个数即可找到),然后翻转一半节点到创建的另一个链表当中,然后进行比对即可得到是否为回文结构;
class PalindromeList {
public:
    bool chkPalindrome(ListNode* A) {
        if(!A) return true;
        int count=0;
        ListNode* cur=A;
        while(cur)
        {
            cur=cur->next;
            count++;
        }
        count=(count+1)/2;
        int i=count;
        ListNode*head=NULL;
        while(i--)
        {
            ListNode*next=A->next;
            A->next=head;
            head=A;
            A=next;
        }
        while(head&&A)
        {
            if(head->val!=A->val)
            return false;
            head=head->next;
            A=A->next;
        }
        return true;
    }
};
8. 输入两个链表,找出它们的第一个公共结点。 160. 相交链表 - 力扣(LeetCode)

这个题我们可以首先想到暴力解法:先在定一条链表中的某个节点,然后再另一个链表中找该节点,如果找到了有返回该节点,不过这样的解法时间复杂度太高了

第二种方法是我们先遍历各自链表,找出其中个数,先让长的那一条链表先走他们的节点个数差,然后两条链表同时遍历,找出相交节点;

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {
    int cnt1=0,cnt2=0;
    struct ListNode* cur1=headA;
    while(cur1)
    {
        cur1=cur1->next;
        cnt1++;
    }
    struct ListNode* cur2=headB;
    while(cur2)
    {
        cur2=cur2->next;
        cnt2++;
    }
    if(cur2!=cur1)//如果链表到最后都不相等,说明不相交,没有交点
    return NULL;
    //假定A比B长
    struct ListNode* ha=headA;
    struct ListNode* hb=headB;
    if(cnt1<cnt2)
    {
        ha=headB;
        hb=headA;
    }
    int cnt=abs(cnt1-cnt2);
    while(cnt--)//让长的先走个数差
    ha=ha->next;
    while(ha)//两条链表同时遍历
    {
        if(ha==hb)//注意两个节点相等,不是节点值相等  !!!!!!!
        return ha;
        ha=ha->next;
        hb=hb->next;
    }
    return NULL;
}

9.给定一个链表,判断链表中是否有环。141. 环形链表 - 力扣(LeetCode)

这是一个经典快慢指针:

bool hasCycle(struct ListNode *head) {
    struct ListNode* fast=head;
    struct ListNode* slow=head;
    while(fast&&fast->next)
    {
        fast=fast->next->next;
        slow=slow->next;
        if(fast==slow)
        return true;
    }
    return false;
}
10.给定一个链表,返回链表开始入环的第一个结点。 如果链表无环,则返回  NULL
142. 环形链表 II - 力扣(LeetCode)

这个题需要我么用到简单的数学知识:

 

struct ListNode *detectCycle(struct ListNode *head) {
    struct ListNode* fast=head;
    struct ListNode* slow=head;
    while(fast&&fast->next)
    {
        fast=fast->next->next;
        slow=slow->next;
        if(fast==slow)
        {
        struct ListNode* cur=head;
        while(cur!=fast)
        {
            cur=cur->next;
            fast=fast->next;
        }
        return cur;
        }
    }
    return NULL;
}
11. 给定一个链表,每个结点包含一个额外增加的随机指针,该指针可以指向链表中的任何结点或空结点。 要求返回这个链表的深度拷贝。 138. 随机链表的复制 - 力扣(LeetCode)

这个题我们可以用这个方法: 在原链表中,每个节点都复制一个节点插入在自己的后面

然后进行random处理,如果原节点的random存在,则新节点的random指向原节点的random的下一个。最后,将两条链表分离即可。

struct Node* copyRandomList(struct Node* head) {
    if(!head) return NULL;
	struct Node* cur=head;
    while(cur)
    {
        struct Node* newnode=(struct Node*)malloc(sizeof(struct Node));
        newnode->next=cur->next;
        newnode->val=cur->val;
        newnode->random=NULL;
        cur->next=newnode;
        cur=newnode->next;
    }
    cur=head;
    while(cur)
    {
        struct Node* new=cur->next;
        if(cur->random)
        new->random=cur->random->next;
        cur=new->next;
    }
    cur=head;
    struct Node* newhead=cur->next;
    struct Node* tail=cur->next;
    while(cur)
    {
        cur->next=tail->next;
        cur=cur->next;
        if(cur)
        tail->next=cur->next;
        tail=tail->next;
    }
    return newhead;
}

五、总体代码

SList.h

#pragma once
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
// slist.h
typedef int SLTDateType;
typedef struct SListNode
{
	SLTDateType data;
	struct SListNode* next;
}SListNode;

// 动态申请一个节点
SListNode* BuySListNode(SLTDateType x);
// 单链表打印
void SListPrint(SListNode* plist);
// 单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x);
// 单链表的头插
void SListPushFront(SListNode** pplist, SLTDateType x);
// 单链表的尾删
void SListPopBack(SListNode** pplist);
// 单链表头删
void SListPopFront(SListNode** pplist);
// 单链表查找
SListNode* SListFind(SListNode* plist, SLTDateType x);
// 单链表在pos位置之后插入x
// 分析思考为什么不在pos位置之前插入?
void SListInsertAfter(SListNode* pos, SLTDateType x);
// 单链表删除pos位置之后的值
// 分析思考为什么不删除pos位置?
void SListEraseAfter(SListNode* pos);

// 在pos的前面插入
void SLTInsert(SListNode** pphead, SListNode* pos, SLTDateType x);
// 删除pos位置
void SLTErase(SListNode** pphead, SListNode* pos);
void SLTDestroy(SListNode** pphead);

SList.h

#define _CRT_SECURE_NO_WARNINGS 1
#include"SList.h"

// 动态申请一个节点
SListNode* BuySListNode(SLTDateType x)
{
	SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
	if (newnode == NULL)
	{
		perror("malloc fail");
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}
// 单链表打印
void SListPrint(SListNode* plist)
{
	assert(plist);
	SListNode* cur = plist;
	while (cur)
	{
		printf("%d-> ", cur->data);
		cur = cur->next;
	}
	printf("\n");
}
// 单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x)
{
	assert(pplist);
	SListNode* newnode = BuySListNode(x);
	if (*pplist == NULL)
	{
		*pplist = newnode;
		return;
	}
	//找尾巴
	SListNode* tail = *pplist;
	while (tail->next != NULL)
		tail = tail->next;
	tail->next = newnode;
}
// 单链表的头插
void SListPushFront(SListNode** pplist, SLTDateType x)
{
	assert(pplist);
	SListNode* newnode = BuySListNode(x);
	newnode->next = *pplist;
	*pplist = newnode;
}
// 单链表的尾删
void SListPopBack(SListNode** pplist)
{
	assert(pplist);
	assert(*pplist);//有数据才能删,没有数据不删
	if ((*pplist)->next == NULL)
	{
		free(*pplist);
		*pplist = NULL;
	}
	else
	{
		//找尾巴
		SListNode* prev = NULL;
		SListNode* tail = *pplist;
		while (tail->next != NULL)
		{
			prev = tail;
			tail = tail->next;
		}
		free(tail);
		prev->next = NULL;
	}
}
// 单链表头删
void SListPopFront(SListNode** pplist)
{
	assert(pplist);
	assert(*pplist);
	SListNode* next = (*pplist)->next;
	free(*pplist);
	*pplist = next;
}
// 单链表查找
//plist是一个变量,保存的是地址,传的时候也是地址,传过去就是将变量里面的地址拷贝一份传过去了
SListNode* SListFind(SListNode* plist, SLTDateType x)
{
	assert(plist);
	SListNode* cur = plist;
	while (cur)
	{
		if (cur->data == x)
			return cur;
		cur = cur->next;
	}
	return NULL;
}
// 单链表在pos位置之后插入x
// 分析思考为什么不在pos位置之前插入?
void SListInsertAfter(SListNode* pos, SLTDateType x)
{
	assert(pos);
	SListNode* newnode = BuySListNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
}
// 单链表删除pos位置之后的值
// 分析思考为什么不删除pos位置?
void SListEraseAfter(SListNode* pos)
{
	assert(pos);
	assert(pos->next);//检查是不是最后一个
	SListNode* next = pos->next;
	pos->next = next->next;
	free(next);
}

// 在pos的前面插入
void SLTInsert(SListNode** pphead, SListNode* pos, SLTDateType x)
{
	assert(pphead);
	assert(*pphead);
	//这里没有检查pos是否为NULL,我认为在最后一个节点之后插入也是可以的
	if (pos == *pphead)
	{
		SListPushFront(pphead, x);
		return;
	}
	SListNode* prev = *pphead;
	while (prev->next!=pos)
	{
		prev = prev->next;
	}
	SListNode* newnode = BuySListNode(x);
	newnode->next = pos;
	prev->next = newnode;
}
// 删除pos位置
void SLTErase(SListNode** pphead, SListNode* pos)
{
	assert(pphead);
	assert(*pphead);
	assert(pos);//这里进行检查pos,必须有这个节点才能删除
	if (pos == *pphead)
	{
		SListPopFront(pphead);
		return;
	}
	SListNode* prev = *pphead;
	while (prev->next != pos)
	{
		prev = prev->next;
	}
	prev->next = pos->next;
	free(pos);
	pos = NULL;
}
void SLTDestroy(SListNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	SListNode* cur = *pphead;
	while (cur)
	{
		SListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	*pphead = NULL;
}

Test.c

#define _CRT_SECURE_NO_WARNINGS 1
#include"SList.h"
void test1()
{
	SListNode* SL = NULL;
	SListPushBack(&SL, 1);
	SListPushBack(&SL, 2);
	SListPushBack(&SL, 3);
	SListPushBack(&SL, 4);
	SListPushBack(&SL, 5);

	SListPrint(SL);

	SListPushFront(&SL, 6);
	SListPushFront(&SL, 7);
	SListPushFront(&SL, 8);
	SListPushFront(&SL, 9);
	SListPushFront(&SL, 10);
	SListPushFront(&SL, 99);
	SListPushFront(&SL, 88);
	SListPushFront(&SL, 77);

	SListPrint(SL);

	SListPopBack(&SL);
	SListPopBack(&SL);
	SListPopBack(&SL);
	SListPopBack(&SL);

	SListPrint(SL);

	SListPopFront(&SL);
	SListPopFront(&SL);
	SListPopFront(&SL);

	SListPrint(SL);

	SListInsertAfter(SListFind(SL, 1), 99);
	SListInsertAfter(SListFind(SL, 10), 99);

	SListPrint(SL);

	SListEraseAfter(SListFind(SL, 1));
	SListEraseAfter(SListFind(SL, 10));
	//printf("%d\n", SListFind(SL, 1)->data);

	SListPrint(SL);

	SLTInsert(&SL, SListFind(SL, 10), 99);
	SLTInsert(&SL, NULL, 99);

	SListPrint(SL);

	SLTErase(&SL, SListFind(SL, 99));
	SLTErase(&SL, SListFind(SL, 99));

	SListPrint(SL);

	SLTDestroy(&SL);

}
int main()
{
	test1();
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/735952.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

浏览器插件利器-allWebPluginV2.0.0.14-bata版发布

allWebPlugin简介 allWebPlugin中间件是一款为用户提供安全、可靠、便捷的浏览器插件服务的中间件产品&#xff0c;致力于将浏览器插件重新应用到所有浏览器。它将现有ActiveX插件直接嵌入浏览器&#xff0c;实现插件加载、界面显示、接口调用、事件回调等。支持谷歌、火狐等浏…

C语言中的字符输入/输出和验证输入

在C语言中&#xff0c;字符输入/输出功能允许程序与用户进行交互&#xff0c;读取用户的输入信息并展示输出结果。同时&#xff0c;验证输入的作用在于确保用户输入的数据符合预期&#xff0c;以提高程序的稳定性和可靠性&#xff0c;防止无效输入引发的错误或异常行为&#xf…

外排序(C语言实现)

前言 本篇博客讲解一下外排序&#xff0c;看这篇排序你的先去看一下&#xff1a;八大经典排序算法-CSDN博客 &#x1f493; 个人主页&#xff1a;普通young man-CSDN博客 ⏩ 文章专栏&#xff1a;排序_普通young man的博客-CSDN博客 若有问题 评论区见&#x1f4dd; &#x1f3…

二叉树的基础讲解

二叉树在遍历&#xff0c;查找&#xff0c;增删的效率上面都很高&#xff0c;是数据结构中很重要的&#xff0c;下面我们来基础的认识一下。(高级的本人还没学&#xff0c;下面的代码用伪代码或C语言写的)我会从树&#xff0c;树的一些专有名词&#xff0c;树的遍历&#xff0c…

【博客719】时序数据库基石:LSM Tree的增删查改

时序数据库基石&#xff1a;LSM Tree的增删查改 LSM结构 LSM树将任何的对数据操作都转化为对内存中的Memtable的一次插入。Memtable可以使用任意内存数据结构&#xff0c;如HashTable&#xff0c;BTree&#xff0c;SkipList等。对于有事务控制需要的存储系统&#xff0c;需要在…

web安全渗透测试十大常规项(一):web渗透测试之JAVA反序列化

渗透测试之PHP反序列化 1. Java反序列化1.1 Java安全-反序列化-原生序列化类函数1.1.1 原生序列化类函数:1.2 Java安全-SpringBoot框架-泄漏&CVE1. Java反序列化 1、序列化与反序列化 序列化:将内存中的对象压缩成字节流 反序列化:将字节流转化成内存中的对象2、为什么…

huggingface官网下载并处理ImageNet2012数据集

文章目录 一、下载imagenet2012数据集二、转换imagenet数据集格式 ImageNet数据集可以直接从ImageNet官方网站获取数据&#xff0c;但通常需要注册并遵守使用协议。另外&#xff0c;由于数据集较大&#xff0c;往往下载需要花费大量的时间空间&#xff0c;而通过huggingface下载…

达梦8 通过SF_INJECT_HINT解决新排序机制下失控语句影响其他SQL执行的问题

达梦数据库有两种排序机制。当SORT_FLAG设置0时&#xff0c;采用旧排序机制&#xff1b;当SORT_FLAG1时&#xff0c;采用新排序机制。详见《达梦新老排序机制的对比》 两种排序机制各有优缺点。 新排序机制引入了全局排序区概念&#xff0c;虽然避免了内存溢出导致系统OOM&am…

[数据概念|方案实操]清华数据大讲堂5-数据要素化治理的理论方法与工程实践

“ 数据要素化是资产化的重要前提和实现路径” 鼹鼠哥公众号链接在 [数据概念|方案实操]清华数据大讲堂5-数据要素化治理的理论方法与工程实践 (qq.com) 2024年6月5日&#xff0c;清华数据大讲堂第五讲开讲。 中国电子信息产业集团副总 陆志鹏 以《数据要素化治理的理论方法与…

扎克伯格2017年哈佛大学毕业典礼演讲:Mark Zuckerberg Harvard Commencement 2017

Facebook Founder Mark Zuckerberg Commencement Address | Harvard Commencement 2017 Link: https://www.youtube.com/watch?vBmYv8XGl-YU 文章目录 Facebook Founder Mark Zuckerberg Commencement Address | Harvard Commencement 2017SummarySummary of Mark Zuckerberg…

[图解]建模相关的基础知识-16

1 00:00:00,350 --> 00:00:04,130 刚才那个&#xff0c;就相当于&#xff0c;12这个我们可以认为是什么 2 00:00:05,020 --> 00:00:11,360 我们用类图来表达就是&#xff0c;员工、电话 3 00:00:13,320 --> 00:00:15,080 多个 4 00:00:15,090 --> 00:00:16,440 …

MySQL 超出月份最大日期(工作总结)

前几天帮同事修改了一个bug&#xff0c;这个bug是怎么造成的呢。先来看需求&#xff0c;系统需要统计某个月份的数据。很简单的一个需求。 同事的写的MySQL语句 SELECTREPLACE(FORMAT(sum(count_value),2), ,, ) as value,<if test"type day">count_date as…

068、PyCharm 关于Live Template模板

在 PyCharm 编辑器中&#xff0c;Live Templates 是一种功能强大的工具&#xff0c;可以帮助我们快速插入常用的代码片段或模板。 以下是在 PyCharm 中添加 Live Templates 的步骤&#xff1a; 添加 Live Templates 步骤&#xff1a; 打开 PyCharm 编辑器。 转到菜单栏中的 …

分布式,容错:10台电脑坏了2台

由10台电脑组成的分布式系统&#xff0c;随机、任意坏了2台&#xff0c;剩下的8台电脑仍然储存着全部信息&#xff0c;可以继续服务。这是怎么做到的&#xff1f; 设N台电脑&#xff0c;坏了H台&#xff0c;要保证上述性质&#xff0c;需要有冗余&#xff0c;总的存储量降低为…

三、MyBatis实践:提高持久层数据处理效率

三、MyBatis实践&#xff1a;提高持久层数据处理效率 目录 一、Mybatis简介 1.1 简介1.2 持久层框架对比1.3 快速入门&#xff08;基于Mybatis3方式&#xff09; 二、MyBatis基本使用 2.1 向SQL语句传参 2.1.1 mybatis日志输出配置2.1.2 #{}形式2.1.3 ${}形式 2.2 数据输入 2…

redis主从复制、哨兵、集群

在实际的生活环境中&#xff0c;如果只使用一个redis进行读写操作&#xff0c;那么面对庞大的访问人群是崩溃的&#xff0c;所以可以有几个redis&#xff0c;一个用来做主机&#xff0c;提供修改数据操作&#xff0c;而这个主机用来控制其他redis&#xff0c;即将更新的发送&am…

【七】【QT开发应用】跨UI发送信号,跨线程发送信号

跨UI发送信号 基本框架 新建窗口 自定义信号 跨线程发送信号 新建线程 查看线程号 完整代码 跨UI发送信号 setdialog.h #ifndef SETDIALOG_H #define SETDIALOG_H#include <QDialog>namespace Ui { class setdialog; }class setdialog : public QDialog {Q_OBJECTpub…

【Python】已解决:ModuleNotFoundError: No module named ‘paddle’

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决&#xff1a;ModuleNotFoundError: No module named ‘paddle’ 一、分析问题背景 在Python编程中&#xff0c;ModuleNotFoundError是一个常见的错误&#xff0c;它通常发生…

C语言的数据结构:树与二叉树(树篇)

前言 之前所学到的数据结构都是线性结构特征&#xff0c;所谓线性就是在结构上&#xff0c;将节点连接起来时&#xff0c;像一条线一样。如链表则是上一个节点包含下一个节点地址的指针&#xff0c;这样依次下去。而串、队列、栈则实现方式都依赖于链表或顺序表而实现&#xf…

Inception_V2_V3

Inception_V2_V3 CNN卷积网络的发展史 1. LetNet5(1998) 2. AlexNet(2012) 3. ZFNet(2013) 4. VGGNet(2014) 5. GoogLeNet(2014) 6. ResNet(2015) 7. DenseNet(2017) 8. EfficientNet(2019) 9. Vision Transformers(2020) 10. 自适应卷积网络(2021) 上面列出了发展到现在CNN的…