数据结构与算法—双向链表

news2024/12/26 2:10:01

目录

一、链表的分类

二、双向链表原理

三、实现双向链表 

1、声明链表结构体 

2、初始化链表

3、创建新节点

4、打印链表

5、头插&尾插

头插 

尾插

6、头删&尾删

头删

尾删

7、 查找节点

8、指定节点前插入

9、删除指定节点

10、销毁链表

完整版

Lisy.h 声明

List.c函数

text.c测试代码


一、链表的分类

实际中链表的结构非常多样,以下情况组合起来就有8种链表结构:
1. 单向或者双向
2. 带头或者不带头(头又叫哨兵位)
3. 循环或者非循环
虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:
1. 无头单向非循环链表 结构简单,一般不会单独用来存数据。实际中更多是作为 其他数据结构的子结构 ,如 哈希桶、图的邻接表 等等。另外这种结构在 笔试面试 中出现很多。
2. 带头双向循环链表 结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多 优势 ,实现反而 简单了 ,后面我们代码实现了就知道了。

二、双向链表原理

 我们来对比单向链表的结构体,看一下与双向链表的结构体有什么区别:(左单向,右双向)

  1. 我们可以看出双向链表比单向链表多了一个结构体指针prev,它的作用是存储前一个结点的地址。
  2. 双向链表的每个节点既可以找到前一个节点,也可以找到后一个节点,
  3. 此外头节点和尾节点比较特殊,头节点的prev指向尾节点,尾节点的next指向头节点。
  4. 如果链表只有头节点(也叫哨兵位),那么它的prev和next都指向自己。 

三、实现双向链表 

1、声明链表结构体 

typedef int LTDataType;
typedef struct ListNode
{
	struct ListNode* next;
	struct ListNode* prev;
	LTDataType data;
}LTNode;
  • 首先构建链表结构体,将链表结点的data数据类型设置别名LTDataType,方便后续修改。
  • 定义结构体名称的别名为LTNode。 
  • 指针next指向该节点的下一个节点,
  • 指针prev指向该节点的前一个节点。
  • LTDataType类型的data用于存值。

2、初始化链表

LTNode* LTInit()
{
	LTNode* phead = BuyLTNode(-1);
	phead->prev = phead;
	phead->next = phead;
	return phead;
}
  • 函数的目的是创建一个空的双向链表并返回指向链表头部的指针。
  • 通过BuyLTNode( )函数为头节点开辟空间,头节点的data值为-1。(后续讲解)
  • 初始化时,prev和next均指向头节点。
  • 返回值为指针phead。

创建结构体指针,将初始化函数的返回值赋值给结构体指针,即为初始化完成,代码如下:

LTNode* plist = LTInit();

  这里也可以用二级指针接收链表地址初始化链表。

void LTInit(LTNode** phead)
{
	assert(phead);
	* phead = BuyLTNode(-1);
	(*phead)->prev = *phead;
	(*phead)->next = *phead;
	(*phead)->data = 0;
}

使用该函数初始化时,使用方法也有变化:

LTNode* plist;
LTInit(&plist);

3、创建新节点

LTNode* BuyLTNode(LTDataType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	if (newnode == NULL) {
		perror("malloc fail");
		return NULL;
	}
	newnode->data = x;
	newnode->prev = NULL;
	newnode->next = NULL;
	return newnode;
}
  •  使用malloc为新节点newnode开辟结构体LTNode大小的空间。
  • 开辟失败perror打印错误信息,函数提前结束。
  • 为新节点的data赋值为参数x的值,prev、next置空。
  • 返回新节点指针。

4、打印链表

void LTPrint(LTNode* phead)
{
	assert(phead);
	printf("guard<==>");
	LTNode* cur = phead->next;
	while (cur != phead) {
		if(cur!=phead->prev)
			printf("%d<==>", cur->data);
		else
			printf("%d\n", cur->data);
		cur = cur->next;
	}
}
  • assert检查链表是否为空,为空则报错。
  • 打印字符串 "guard<==>",用于表示链表的头节点。
  • 指针cur指向头节点下一个节点,也就是第一个节点。
  • while循环的结束条件为cur指向头节点,双向链表尾部不为空,则循环遍历结束条件为尾结点的next指向头节点时。
  • 当cur不指向最后一个节点,打印节点之间的连接符号<==>。

5、头插&尾插

《头插&尾插 和 头删&尾删》我们都可以通过后续的指定位置前插入函数和删除函数实现, 这里进行讲解为了更好得熟悉理解双向链表

头插 

void LTPushFront(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* newnode = BuyLTNode(x);
	LTNode* first = phead->next;
//下面两块代码的顺序可以颠倒,因为使用指针first储存了原来链表第一个元素的地址
	newnode->next = first;
	first->prev = newnode;

	phead->next = newnode;
	newnode->prev = phead;
}
  • assert检查链表是否为空,为空则报错。
  • 为新节点newnode开辟空间。
  • 指针first指向链表原第一个节点。
  • 将新节点的next指向原第一个节点,原第一个节点的prev指向新节点。
  • 头节点的next指向新节点,新节点的prev指向头节点。

 

 当然下面这种形式也对,只是后两个代码块不能调换顺序,具体原因大家可以画图试一下。

void LTPushFront(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* newnode = BuyLTNode(x);

	newnode->next = phead->next;
	phead->next->prev = newnode;

	newnode->prev = phead;
	phead->next = newnode;
}

尾插

void LTPushBack(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* newnode = BuyLTNode(x);
	LTNode* tail = phead->prev;
	
	newnode->next = phead;
	phead->prev = newnode;

	tail->next = newnode;
	newnode->prev = tail;
}
  • assert检查链表是否为空,为空则报错。
  • 为新节点newnode开辟空间。
  • 指针tail指向尾节点,直接通过头节点phead的prev即可找到尾节点。
  • 新节点的next指向头节点,头节点的prev指向新节点。
  • 原尾节点的next指向新节点,新节点的prev指向尾节点。

6、头删&尾删

头删

bool LTEmpty(LTNode* phead)
{
	assert(phead);
	return phead->next == phead;
}

void LTPopFront(LTNode* phead)
{
	assert(phead);
	assert(!LTEmpty(phead));

	LTNode* first = phead->next;
	LTNode* second = first->next;

	second->prev = phead;
	phead->next = second;
	free(first);
}
  • 借助LTEmpty函数判断链表是否只有头节点,只有头节点则不进行删除。
  • 第一个assert检查链表是否为空,为空则报错。
  • 第二个assert通过LTEmpty函数返回值判断是否可以进行删除。
  • 定义两个指针first和second分别指向第一个节点和第二个节点。
  • 将第二个节点的prev指向头节点。
  • 头节点的next指向第二个节点。
  • 释放第一个节点的空间。

尾删

void LTPopBack(LTNode* phead)
{
	assert(phead);
	assert(!LTEmpty(phead));

	LTNode* tail = phead->prev;
	LTNode* tailPrev = tail->prev;

	free(tail);
	tailPrev->next = phead;
	phead->prev = tailPrev;

}
  • 第一个assert检查链表是否为空,为空则报错。
  • 第二个assert通过LTEmpty函数返回值判断是否可以进行删除。
  • 定义两个指针tail和tailPrev,分别指向尾节点和尾节点的前一个节点。
  • 释放尾节点的空间。
  • tailPrev的next指向头节点。
  • 头节点的prev指向tailPrev(新的尾节点)。

7、 查找节点

LTNode* LTFind(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* cur = phead->next;
	while (cur != phead) {
		if (cur->data == x)
			return cur;

		cur = cur->next;
	}
	return NULL;
}
  • assert检查链表是否为空,为空则报错。
  • 指针cur指向第一个节点。
  • 循环遍历链表,节点的data值为x时,返回该节点的地址。
  • 找不到则返回NULL。

8、指定节点前插入

void LTInsert(LTNode* pos, LTDataType x)
{
	assert(pos);

	LTNode* prev = pos->prev;
	LTNode* newnode = BuyLTNode(x);

	prev->next = newnode;
	newnode->prev = prev;

	newnode->next = pos;
	pos->prev = newnode;
}
  • assert检查节点是否为空,为空则报错。
  • 指针prev指向pos节点的前一个节点。
  • 为新节点newnode开辟空间,节点的data值为x。
  • 将prev的next指向新节点,新节点的prev指向prev。
  • 新节点的next指向pos,pos的prev指向新节点。

9、删除指定节点

void LTErase(LTNode* pos)//可以判断phead是否等于哨兵位
{
	assert(pos);
	LTNode* posPrev = pos->prev;
	LTNode* posNext = pos->next;
	
	posPrev->next = posNext;
	posNext->prev = posPrev;
	free(pos);
}
  • assert检查节点是否为空,为空则报错。
  • 定义两个指针posPrev和posNext分别指向 pos的前一个节点和 pos后一个节点。
  • 前一个节点posPrev的next指向posNext。
  • 后一个节点posNext的prev指向posPrev
  • 释放pos节点的空间。

10、销毁链表

void LTDestroy(LTNode** phead)
{
	assert(phead);
	assert(*phead);
	LTNode* cur = (*phead)->next;
	while (cur != *phead) {
		LTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(*phead);
	*phead = NULL;
}
  • assert检查链表是否为空,为空则报错。
  • 检查链表头指针 *phead 是否为非空,为空则报错。
  • 指针cur指向第一个节点,循环内通过cur遍历每个节点。
  • next指针记录cur下一个节点,将cur的空间释放,cur更新为next,指向下一个节点。
  • 在循环结束后,链表中的所有节点都已经被销毁,最后一步是释放链表头指针 *phead 所占用的内存。

  • *phead = NULL 最后,将链表头指针设置为 NULL,以确保不再引用链表,避免悬挂指针或野指针的问题。

完整版

完整版代码我保留了注释,方便读者学习,以下代码的功能经测试没有问题,放心使用。 

Lisy.h 声明

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>

typedef int LTDataType;
typedef struct ListNode
{
	struct ListNode* next;
	struct ListNode* prev;
	LTDataType data;
}LTNode;

//初始化链表
LTNode* LTInit();
//void LTInit(LTNode** phead);

//打印链表
void LTPrint(LTNode* phead);

//头插&尾插
void LTPushFront(LTNode* phead, LTDataType x);
void LTPushBack(LTNode* phead, LTDataType x);

//头删&尾删
void LTPopFront(LTNode* phead);
void LTPopBack(LTNode* phead);

//查找值为x的节点,返回节点地址
LTNode* LTFind(LTNode* phead, LTDataType x);

//在指定节点前插入
void LTInsert(LTNode* pos, LTDataType x); 

//删除指定节点
void LTErase(LTNode* pos);

//销毁链表
void LTDestroy(LTNode** phead);

List.c函数

#define _CRT_SECURE_NO_WARNINGS 1
#include "List.h"

//创建新节点
LTNode* BuyLTNode(LTDataType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	if (newnode == NULL) {
		perror("malloc fail");
		return NULL;
	}
	newnode->data = x;
	newnode->prev = NULL;
	newnode->next = NULL;
	return newnode;
}

//初始化链表
LTNode* LTInit()
{
	LTNode* phead = BuyLTNode(-1);
	phead->prev = phead;
	phead->next = phead;
	return phead;
}
//void LTInit(LTNode** phead)
//{
//	assert(phead);
//	* phead = BuyLTNode(-1);
//	(*phead)->prev = *phead;
//	(*phead)->next = *phead;
//	(*phead)->data = 0;
//}

//打印链表
void LTPrint(LTNode* phead)
{
	assert(phead);
	printf("guard<==>");
	LTNode* cur = phead->next;
	while (cur != phead) {
		if(cur!=phead->prev)
			printf("%d<==>", cur->data);
		else
			printf("%d\n", cur->data);
		cur = cur->next;
	}
}

//头插
void LTPushFront(LTNode* phead, LTDataType x)
{
	LTInsert(phead->next, x);
}

//void LTPushFront(LTNode* phead, LTDataType x)
//{
//	assert(phead);
//	LTNode* newnode = BuyLTNode(x);
//	LTNode* first = phead->next;
//	//下面两块代码的顺序可以颠倒,因为使用指针first储存了原来链表第一个元素的地址
//	newnode->next = first;
//	first->prev = newnode;
//
//	phead->next = newnode;
//	newnode->prev = phead;
//}

//void LTPushFront(LTNode* phead, LTDataType x)
//{
//	assert(phead);
//	LTNode* newnode = BuyLTNode(x);
//
//	newnode->next = phead->next;
//	phead->next->prev = newnode;
//
//	newnode->prev = phead;
//	phead->next = newnode;
//}

//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{
	assert(phead);

	LTInsert(phead, x);
}
//void LTPushBack(LTNode* phead, LTDataType x)
//{
//	assert(phead);
//	LTNode* newnode = BuyLTNode(x);
//	LTNode* tail = phead->prev;
//	
//	newnode->next = phead;
//	phead->prev = newnode;
//
//	tail->next = newnode;
//	newnode->prev = tail;
//}

//监测链表是否为空
bool LTEmpty(LTNode* phead)
{
	assert(phead);
	return phead->next == phead;
}

//头删
void LTPopFront(LTNode* phead)
{
	assert(phead);
	assert(!LTEmpty(phead));

	LTErase(phead->next);
}

//void LTPopFront(LTNode* phead)
//{
//	assert(phead);
//	assert(!LTEmpty(phead));
//
//	LTNode* first = phead->next;
//	LTNode* second = first->next;
//
//	second->prev = phead;
//	phead->next = second;
//	free(first);
//}


//尾删
void LTPopBack(LTNode* phead)
{
	assert(phead);
	assert(!LTEmpty(phead));

	LTErase(phead->prev);
}

//void LTPopBack(LTNode* phead)
//{
//	assert(phead);
//	assert(!LTEmpty(phead));
//	//LTErase(phead->prev);
//
//	LTNode* tail = phead->prev;
//	LTNode* tailPrev = tail->prev;
//
//	free(tail);
//	tailPrev->next = phead;
//	phead->prev = tailPrev;
//
//}




//查找值为x的节点,返回节点地址
LTNode* LTFind(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* cur = phead->next;
	while (cur != phead) {
		if (cur->data == x)
			return cur;

		cur = cur->next;
	}
	return NULL;
}

//在指定节点前插入,与LTFind搭配使用
void LTInsert(LTNode* pos, LTDataType x)
{
	assert(pos);

	LTNode* prev = pos->prev;
	LTNode* newnode = BuyLTNode(x);

	prev->next = newnode;
	newnode->prev = prev;

	newnode->next = pos;
	pos->prev = newnode;
}

// 删除指定节点
void LTErase(LTNode* pos)//可以判断phead是否等于哨兵位
{
	assert(pos);
	LTNode* posPrev = pos->prev;
	LTNode* posNext = pos->next;
	
	posPrev->next = posNext;
	posNext->prev = posPrev;
	free(pos);
}

//销毁链表
void LTDestroy(LTNode** phead)
{
	assert(phead);
	assert(*phead);
	LTNode* cur = (*phead)->next;
	while (cur != *phead) {
		LTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(*phead);
	*phead = NULL;
}

text.c测试代码

#include "List.h"

void test1() {
	/*LTNode* plist;
	LTInit(&plist);*/

	LTNode* plist = LTInit();
	LTPushFront(plist, 1);
	LTPushFront(plist, 2);
	LTPushFront(plist, 3);
	LTPushFront(plist, 4);
	LTPushBack(plist, 888);
	LTPushBack(plist, 888);
	LTPopFront(plist);
	LTPopBack(plist);
	LTNode* pos = LTFind(plist, 888);
	LTInsert(pos, 999);

	LTErase(pos);

	LTPrint(plist);
	LTDestroy(&plist);
}

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java三层架构、表现层-业务层-持久层

三层架构 什么是 Java 三层架构 三层架构是指&#xff1a;视图层view&#xff08;表现层&#xff09;&#xff0c;服务层service&#xff08;业务逻辑层&#xff09;&#xff0c;持久层Dao&#xff08;数据访问层&#xff09;&#xff0c; Java的三层架构是指将Java程序分为三…

会议剪影 | 思腾合力携AI服务器亮相PRCV 2023,并作主题演讲

第六届中国模式识别与计算机视觉大会&#xff08;PRCV 2023&#xff09;于2023年10月13日至15日在厦门国际会议中心酒店举办。本届会议主题为“相约鹭岛&#xff0c;启智未来”。 会议旨在汇聚国内国外模式识别和计算机视觉理论与应用研究的广大科研工作者及工业界同行&#xf…

【LeetCode刷题(数据结构与算法)】:将二叉搜索树转化为排序的双向链表

将一个 二叉搜索树 就地转化为一个 已排序的双向循环链表 对于双向循环列表&#xff0c;你可以将左右孩子指针作为双向循环链表的前驱和后继指针&#xff0c;第一个节点的前驱是最后一个节点&#xff0c;最后一个节点的后继是第一个节点 特别地&#xff0c;我们希望可以 就地 完…

pip 时报错 no such option: --bulid-dir 的解决办法

Pycharm 安装第三方库报错及解决方案——no such option: --build-dir Pycharm 安装第三方库报错及解决方案——no such option: --build-dir 最近在学习路径规划相关内容&#xff0c;在运行GitHub上下载例程时缺少“plotly”库&#xff0c;根据网上查到的安装步骤操作&#x…

计算机组成原理 new08 电路 $\color{red}{Δ}$

文章目录 ALU基本逻辑运算复合逻辑的运算 一位全加器串行加法器串行进位的并行加法器并行进位的并行加法器(全先行进位加法器)这个明天再写。加法电路原理总结ALU和加法器有什么关系加法器原理ALU总结无符号整数/补码加减法加法器标志位的生成补码加减法发运算的溢出判断溢出电…

Python实现一个简单的http服务,Url传参输出html页面

摘要 要实现一个可以接收参数的HTTP服务器&#xff0c;您可以使用Python标准库中的http.server模块。该模块提供了一个简单的HTTP服务器&#xff0c;可以用于开发和测试Web应用程序。 下面是一个示例代码&#xff0c;它实现了一个可以接收参数的HTTP服务器&#xff1a; 代码…

C1N短网址 - 是如何做到行业领先的

今天从技术角度来聊下短网址的一些事情&#xff0c;市面上的短网址发展基本上经历了几个阶段。 短网址发展的几个阶段&#xff1a; 第一阶段&#xff1a;网址缩短&#xff0c;很纯粹的功能&#xff0c;各个大小公司都在做&#xff0c;门槛很低。典型代表&#xff1a;百度短网…

Python+playwright 实现Web UI自动化

实现Web UI自动化 技术&#xff1a;Pythonplaywright 目标&#xff1a;自动打开百度浏览器&#xff0c;并搜索“亚运会 金牌榜” 需安装&#xff1a;Playwright &#xff08;不用安装浏览器驱动&#xff09; # 使用浏览器&#xff0c;并可视化打开 browser playwright.ch…

Power BI 傻瓜入门 3. 选择Power BI的版本

本章内容包括&#xff1a; Excel与Power BI的比较选择Power BI的桌面版和服务版之间的差异了解Microsoft提供的许可选项 挑选正确版本的Power BI可能就像参观世界上最大的糖果店&#xff1a;你可以从许多细微差别的替代品中进行选择。选择可以归结为想要、需要、规模&#xf…

用HFSS仿真平面线圈的电感量

用HFSS工具仿真平面线圈的电感量 平面线圈是指在平面上绕制而成的线圈&#xff0c;如PCB上的电感线圈、无线供电使用的金属丝绕制而成的线圈等。根据线圈的不同形状可将平面线圈分为方形线圈&#xff0c;六角形线圈、八角形线圈、螺旋原型线圈等。 网络上的计算平面线圈电感量…

Tmux:终端复用器的基本使用(二)

相关阅读 Tmuxhttps://blog.csdn.net/weixin_45791458/category_12472796.html?spm1001.2014.3001.5482 上一篇文章列举了一些关于tmux中会话的基本使用方法&#xff0c;但会话并非是tmux的最强大的功能&#xff0c;tmux还能在一个会话中创建多个窗口(windows)&#xff0c;并…

springboot配置打野sql语句,而不打印结果

application.yml&#xff0c;注释掉mybatis-plus&#xff0c;增加logging&#xff0c;级别debug

【暴力剪枝】CF1708D

https://codeforces.com/contest/1708/problem/D 题意 思路 这样的操作下&#xff0c;数列减的速度是非常快的&#xff0c;也就是说&#xff0c;易出现很多的0&#xff0c;0的操作没啥意义&#xff0c;所以我们要找到第一个 >0 的数对其后的序列进行排序&#xff0c;就能大…

Vue中的路由是如何工作的?Vue Router的基本用法。

在Vue中,路由是用于管理应用程序中不同页面之间导航的机制。 Vue Router是Vue.js官方提供的路由管理器,它通过集成到Vue应用程序中,提供了一种简单而强大的方式来实现单页应用程序(SPA)的路由功能。 以下是Vue Router的基本用法: 1:安装Vue Router: 首先,使用npm或…

二维码智慧门牌管理系统升级解决方案:地图展示

文章目录 前言一、地图展示功能二、其他升级和改进 前言 随着城市的发展和信息化建设的推进&#xff0c;二维码智慧门牌管理系统在社区管理、物流配送、巡检巡查等多个领域发挥着越来越重要的作用。为了更好地满足用户需求&#xff0c;提升管理效率和服务质量&#xff0c;我们…

【JavaEE重点知识归纳】第9节:抽象类和接口

目录 一&#xff1a;抽象类 1.概念 2.语法 3.特性 4.作用 二&#xff1a;接口 1.概念 2.语法 3.接口使用 4.特性 5.实现多个接口 6.接口间的继承 7.Comparable接口 8.Clonable接口 9.抽象类和接口的区别 一&#xff1a;抽象类 1.概念 &#xff08;1&#xff0…

【C++】类型转换(dynamic_cast,const_cast,static_cast,reinterpret_cast)

&#x1f30f;博客主页&#xff1a; 主页 &#x1f516;系列专栏&#xff1a; C ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ &#x1f60d;期待与大家一起进步&#xff01; 文章目录 C语言中的类型转换一、static_cast二、reinterpret_cast三、 const_cast四、 dynamic…

xlive.dll下载安装方法分享,教你快速修复xlive.dll文件

在运行某些应用程序或游戏时&#xff0c;你可能会遭遇到"xlive.dll缺失"错误提示&#xff0c;这可能导致程序无法正常运行。本文将向你介绍一些可行的解决方法教你下载xlive.dll文件&#xff0c;并详细阐述xlive.dll是什么文件以及导致其缺失的原因。 一.理解"x…

2023-10-19 指针与指针的指针,我就不信你脑壳不疼

点击 <C 语言编程核心突破> 快速C语言入门 指针与指针的指针&#xff0c;我就不信你脑壳不疼 前言一、从一个链表实现说起二、指针, 指针的指针, 头疼的来源总结 前言 C实现一个链表&#xff0c;为什么有时候传入指针&#xff0c;有时候传入指针的指针&#xff0c;究竟…

编程接口:eBPF 程序是怎么跟内核进行交互的?

目录 背景 BPF 系统调用 BPF 辅助函数 BPF 映射 BPF 类型格式 (BTF) 小结 背景 用高级语言开发的 eBPF 程序&#xff0c;需要首先编译为 BPF 字节码&#xff0c;然后借助 bpf 系统调用加载到内核中&#xff0c;最后再通过性能监控等接口与具体的内核事件进行绑定。这样&…