07.FreeRTOS列表与列表项
文章目录
- 07. FreeRTOS列表与列表项
- 1. 列表和列表项的简介
- 2. 列表相关API函数
- 3. 代码验证
07. FreeRTOS列表与列表项
1. 列表和列表项的简介

列表的定义:
typedef struct xLIST
{listFIRST_LIST_INTEGRITY_CHECK_VALUE /* 校验值 */volatile UBaseType_t uxNumberOfItems; /* 列表中列表项的数量 */ListItem_t * configLIST_VOLATILE pxIndex; /* 用于遍历列表 */MiniListItem_t xListEnd; /* 最后一个列表项 */listSECOND_LIST_INTEGRITY_CHECK_VALUE /* 校验值 */
} List_t;

列表项的定义:
struct xLIST_ITEM
{listFIRST_LIST_ITEM_INTEGRITY_CHECK_VALUE /* 用于检测列表项的数据完整性 */configLIST_VOLATILE TickType_t xItemValue; /* 列表项的值 */struct xLIST_ITEM * configLIST_VOLATILE pxNext; /* 下一个列表项 */struct xLIST_ITEM * configLIST_VOLATILE pxPrevious; /* 上一个列表项 */void * pvOwner; /* 列表项的拥有者 */struct xLIST * configLIST_VOLATILE pxContainer; /* 列表项所在列表 */listSECOND_LIST_ITEM_INTEGRITY_CHECK_VALUE /* 用于检测列表项的数据完整性 */
};
typedef struct xLIST_ITEM ListItem_t; /* 重定义成 ListItem_t */

迷你列表项:
struct xMINI_LIST_ITEM
{listFIRST_LIST_ITEM_INTEGRITY_CHECK_VALUE /* 用于检测列表项的数据完整性 */configLIST_VOLATILE TickType_t xItemValue; /* 列表项的值 */struct xLIST_ITEM * configLIST_VOLATILE pxNext; /* 下一个列表项 */struct xLIST_ITEM * configLIST_VOLATILE pxPrevious; /* 上一个列表项 */
};
typedef struct xMINI_LIST_ITEM MiniListItem_t; /* 重定义成 MiniListItem_t */

列表和列表项的关系:
初始状态:
2. 列表相关API函数
| 函数 | 描述 |
|---|---|
| vListInitialise() | 初始化列表 |
| vListInitialiseItem() | 初始化列表项 |
| vListInsertEnd() | 列表末尾插入列表项 |
| vListInsert() | 列表插入列表项 |
| uxListRemove() | 列表移除列表项 |
-
函数vListInitialise()
此函数用于初始化列表,在定义列表之后,需要先对其进行初始化,只有初始化后的列表,才能够正常地被使用。列表初始化的过程,其实就是初始化列表中的成员变量。


-
函数vListInitialiseItem()
此函数用于初始化列表项,在定义列表项之后,也需要先对其进行初始化,只有初始化完的列表项,才能够被正常地使用。列表项初始化的过程,也是初始化列表项中的成员变量。


-
函数vListInsert()
此函数用于将待插入列表的列表项按照列表项值升序排序的顺序,有序地插入到列表中。

插入示意图:
初始状态列表:

插入40后的列表:

插入60后的列表:

插入50后的列表:

代码具体体现:
void vListInsert( List_t * const pxList,ListItem_t * const pxNewListItem ) {ListItem_t * pxIterator;//* 获取列表项的数值依据数值升序排列 */const TickType_t xValueOfInsertion = pxNewListItem->xItemValue;/* Only effective when configASSERT() is also defined, these tests may catch* the list data structures being overwritten in memory. They will not catch* data errors caused by incorrect configuration or use of FreeRTOS. *///* 检查参数是否正确 */listTEST_LIST_INTEGRITY( pxList );listTEST_LIST_ITEM_INTEGRITY( pxNewListItem );/* Insert the new list item into the list, sorted in xItemValue order.** If the list already contains a list item with the same item value then the* new list item should be placed after it. This ensures that TCBs which are* stored in ready lists (all of which have the same xItemValue value) get a* share of the CPU. However, if the xItemValue is the same as the back marker* the iteration loop below will not end. Therefore the value is checked* first, and the algorithm slightly modified if necessary. *///* 如果待插入列表项的值为最大值 */ if( xValueOfInsertion == portMAX_DELAY ){//* 插入的位置为列表 xListEnd 前面 */ pxIterator = pxList->xListEnd.pxPrevious;}else{/* *** NOTE ************************************************************ If you find your application is crashing here then likely causes are* listed below. In addition see https://www.FreeRTOS.org/FAQHelp.html for* more tips, and ensure configASSERT() is defined!* https://www.FreeRTOS.org/a00110.html#configASSERT** 1) Stack overflow -* see https://www.FreeRTOS.org/Stacks-and-stack-overflow-checking.html* 2) Incorrect interrupt priority assignment, especially on Cortex-M* parts where numerically high priority values denote low actual* interrupt priorities, which can seem counter intuitive. See* https://www.FreeRTOS.org/RTOS-Cortex-M3-M4.html and the definition* of configMAX_SYSCALL_INTERRUPT_PRIORITY on* https://www.FreeRTOS.org/a00110.html* 3) Calling an API function from within a critical section or when* the scheduler is suspended, or calling an API function that does* not end in "FromISR" from an interrupt.* 4) Using a queue or semaphore before it has been initialised or* before the scheduler has been started (are interrupts firing* before vTaskStartScheduler() has been called?).* 5) If the FreeRTOS port supports interrupt nesting then ensure that* the priority of the tick interrupt is at or below* configMAX_SYSCALL_INTERRUPT_PRIORITY.**********************************************************************///*遍历列表中的列表项,找到插入的位置for( pxIterator = ( ListItem_t * ) &( pxList->xListEnd ); pxIterator->pxNext->xItemValue <= xValueOfInsertion; pxIterator = pxIterator->pxNext ) /*lint !e826 !e740 !e9087 The mini list structure is used as the list end to save RAM. This is checked and valid. *//*lint !e440 The iterator moves to a different value, not xValueOfInsertion. */{/* There is nothing to do here, just iterating to the wanted* insertion position. */}}//* 将待插入的列表项插入指定位置 */pxNewListItem->pxNext = pxIterator->pxNext;pxNewListItem->pxNext->pxPrevious = pxNewListItem;pxNewListItem->pxPrevious = pxIterator;pxIterator->pxNext = pxNewListItem;/* Remember which list the item is in. This allows fast removal of the* item later. *///* 更新待插入列表项所在列表 */ pxNewListItem->pxContainer = pxList;//* 更新列表中列表项的数量 */ ( pxList->uxNumberOfItems )++; } -
函数vListInsertEnd()
此函数用于将待插入列表的列表项插入到列表 pxIndex 指针指向列表项的前面,是一种无序的插入方法。


插入示意图:
插入前:

插入30后的列表项:

插入前:

插入30后的列表项:

-
函数uxListRemove()
此函数用于将列表项从列表项所在列表中移除.


移除列表项2示意图:

3. 代码验证
本实验主要实现FreeRTOS的列表项的插入与删除,定义三个任务函数,开始任务用于创建其他任务;任务一用于LED灯闪烁,提示系统正常工作;任务二用于进行列表项的插入与删除。
-
函数入口:
用于创建开始任务并开启任务调度
/*函数入口*/ void freertos_Dynamic_Create(void) {lcd_show_string(10, 10, 220, 32, 32, "STM32", RED);lcd_show_string(10, 47, 220, 24, 24, "Task Create&Delete", LIGHTGREEN);lcd_show_string(15, 80, 110, 16, 16, "Task1: 000", GREEN);lcd_show_string(135, 80, 110, 16, 16, "Task2: 000", GREEN);xTaskCreate((TaskFunction_t ) start_task, //指向任务函数的指针(char * ) "start_task", //任务名称(configSTACK_DEPTH_TYPE) START_TASK_STACK_SIZE,//任务堆栈大小,字节为单位(void * ) NULL, //传递给任务函数的参数(UBaseType_t ) START_TASK_PRIO, //任务优先级(TaskHandle_t * ) &start_task_handler //任务句柄:任务控制块);vTaskStartScheduler(); //开启任务调度 } -
开始任务:
用于创建任务一和任务二
void start_task(void* pvParamter) {taskENTER_CRITICAL(); // 进入临界区 xTaskCreate((TaskFunction_t ) task1, //指向任务函数的指针(char * ) "task1", //任务名称(configSTACK_DEPTH_TYPE) TASK1_TASK_STACK_SIZE, //任务堆栈大小,字节为单位(void * ) NULL, //传递给任务函数的参数(UBaseType_t ) TASK1_TASK_PRIO, //任务优先级(TaskHandle_t * ) &task1_task_handler //任务句柄:任务控制块);xTaskCreate((TaskFunction_t ) task2, //指向任务函数的指针(char * ) "task2", //任务名称(configSTACK_DEPTH_TYPE) TASK2_TASK_STACK_SIZE, //任务堆栈大小,字节为单位(void * ) NULL, //传递给任务函数的参数(UBaseType_t ) TASK2_TASK_PRIO, //任务优先级(TaskHandle_t * ) &task2_task_handler //任务句柄:任务控制块); vTaskDelete(NULL);taskEXIT_CRITICAL(); // 退出临界区 } -
任务一:
实现LED0每500ms翻转一次
void task1(void* pvParamter) {uint32_t task1_num = 0;while(1){lcd_show_xnum(71, 80, ++task1_num, 3, 16, 0x80, GREEN);LED0_TOGGLE();vTaskDelay(500);} } -
任务二:
进行列表和列表项的操作
-
初始化列表
vListInitialise(&TestList); //初始化列表 vListInitialiseItem(&ListItem1); //初始化列表项1 vListInitialiseItem(&ListItem2); //初始化列表项2 vListInitialiseItem(&ListItem3); //初始化列表项3ListItem1.xItemValue = 40; ListItem2.xItemValue = 60; ListItem3.xItemValue = 50;- 列表初始化
vListInitialise(&TestList);- 功能:初始化一个列表
TestList。 - 作用:
TestList是一个 FreeRTOS 列表的头部结构体,列表在初始化后将会处于空状态,准备好用于插入或管理列表项。
- 列表项初始化
vListInitialiseItem(&ListItem1); vListInitialiseItem(&ListItem2); vListInitialiseItem(&ListItem3);- 功能:初始化三个列表项
ListItem1、ListItem2和ListItem3。 - 作用:每个列表项结构体在初始化后将会被设置为一个空的列表项,即这些列表项还未插入到任何列表中,且它们的前后指针将指向自身。
- 设置列表项值
ListItem1.xItemValue = 40; ListItem2.xItemValue = 60; ListItem3.xItemValue = 50;- 功能:为每个列表项设置一个值,
xItemValue。 - 作用:
xItemValue是 FreeRTOS 列表项中的一个成员,用于存储与列表项相关的值。这个值可以用来对列表项进行排序或者优先级排序。通常,在 FreeRTOS 中,列表项的值越小,优先级越高。
-
打印列表和其他列表项的地址
/* 第二步:打印列表和其他列表项的地址 */ printf("/**************第二步:打印列表和列表项的地址**************/\r\n"); printf("项目\t\t\t地址\r\n"); printf("TestList\t\t0x%p\t\r\n", &TestList); printf("TestList->pxIndex\t0x%p\t\r\n", TestList.pxIndex); printf("TestList->xListEnd\t0x%p\t\r\n", (&TestList.xListEnd)); printf("ListItem1\t\t0x%p\t\r\n", &ListItem1); printf("ListItem2\t\t0x%p\t\r\n", &ListItem2); printf("ListItem3\t\t0x%p\t\r\n", &ListItem3); printf("/**************************结束***************************/\r\n\r\n");-
打印列表和列表项地址:
printf("TestList\t\t0x%p\t\r\n", &TestList);- 功能:打印
TestList列表头部结构体的地址。 - 解释:
&TestList是TestList的内存地址,这个地址指向整个列表结构体。
- 功能:打印
-
打印
TestList的pxIndex成员的地址:printf("TestList->pxIndex\t0x%p\t\r\n", TestList.pxIndex);- 功能:打印
TestList中pxIndex成员的地址。 - 解释:
pxIndex是TestList列表结构体中的一个成员,通常指向当前列表项的索引。这里打印的是该成员的值,实际上是TestList中pxIndex成员所指向的地址。
- 功能:打印
-
打印
TestList的xListEnd成员的地址:printf("TestList->xListEnd\t0x%p\t\r\n", (&TestList.xListEnd));- 功能:打印
TestList中xListEnd成员的地址。 - 解释:
xListEnd是TestList列表结构体中的一个成员,表示列表的结束位置。这里打印的是该成员的地址。
- 功能:打印
-
打印列表项地址:
printf("ListItem1\t\t0x%p\t\r\n", &ListItem1); printf("ListItem2\t\t0x%p\t\r\n", &ListItem2); printf("ListItem3\t\t0x%p\t\r\n", &ListItem3);- 功能:打印三个列表项的内存地址。
- 解释:
&ListItem1、&ListItem2和&ListItem3分别是这三个列表项的内存地址,用于调试和验证这些结构体的存储位置。
-
实验结果:


-
-
列表项1插入列表
/* 第三步:列表项1插入列表 */ printf("/*****************第三步:列表项1插入列表******************/\r\n"); vListInsert((List_t* )&TestList, /* 列表 */(ListItem_t*)&ListItem1); /* 列表项 */ printf("项目\t\t\t\t地址\r\n"); printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext)); printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext)); printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious)); printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious)); printf("/**************************结束***************************/\r\n\r\n");插入操作
vListInsert((List_t*)&TestList, (ListItem_t*)&ListItem1);- 功能:将
ListItem1插入到TestList列表中。 - 解释:
vListInsert是 FreeRTOS 提供的函数,用于将一个列表项插入到指定的列表中。此操作会将ListItem1插入到TestList中的正确位置,通常是按照xItemValue的顺序。
打印列表项的指针状态
打印输出语句的目的是验证插入操作后的列表状态。
printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext));- 功能:打印
TestList列表的xListEnd成员的pxNext指针的地址。 - 解释:
pxNext指向列表末尾的下一个列表项。在插入操作后,xListEnd的pxNext应该指向插入的列表项(ListItem1)。
printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext));- 功能:打印
ListItem1的pxNext指针的地址。 - 解释:
pxNext是ListItem1的next指针。在插入操作后,ListItem1的pxNext应该指向xListEnd(即列表的末尾)。
printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious));- 功能:打印
TestList列表的xListEnd成员的pxPrevious指针的地址。 - 解释:
pxPrevious指向列表末尾的前一个列表项。在插入操作后,xListEnd的pxPrevious应该指向ListItem1。
printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious));- 功能:打印
ListItem1的pxPrevious指针的地址。 - 解释:
pxPrevious是ListItem1的previous指针。在插入操作后,ListItem1的pxPrevious应该指向xListEnd(即列表的末尾)。
综合分析
通过这段代码,你可以验证
ListItem1是否成功插入到TestList列表中。插入操作完成后,应该能看到以下情况:TestList->xListEnd->pxNext应该指向ListItem1。ListItem1->pxNext应该指向xListEnd。TestList->xListEnd->pxPrevious应该指向ListItem1。ListItem1->pxPrevious应该指向xListEnd。
实验结果


- 功能:将
-
列表项2插入列表
printf("/*****************第四步:列表项2插入列表******************/\r\n"); vListInsert((List_t*)&TestList, (ListItem_t*)&ListItem2); /* 插入 ListItem2 */ printf("项目\t\t\t\t地址\r\n"); printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext)); printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext)); printf("ListItem2->pxNext\t\t0x%p\r\n", (ListItem2.pxNext)); printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious)); printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious)); printf("ListItem2->pxPrevious\t\t0x%p\r\n", (ListItem2.pxPrevious)); printf("/**************************结束***************************/\r\n\r\n");这段代码的目的是将
ListItem2插入到TestList列表中,并打印出插入操作后的列表状态,以便检查和验证列表结构是否正确更新。具体来说,这里主要关注插入后列表项的连接情况。下面是详细的分析:代码功能
/* 第四步:列表项2插入列表 */ printf("/*****************第四步:列表项2插入列表******************/\r\n"); vListInsert((List_t*)&TestList, (ListItem_t*)&ListItem2); /* 插入 ListItem2 到 TestList 列表 */ printf("项目\t\t\t\t地址\r\n"); printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext)); printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext)); printf("ListItem2->pxNext\t\t0x%p\r\n", (ListItem2.pxNext)); printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious)); printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious)); printf("ListItem2->pxPrevious\t\t0x%p\r\n", (ListItem2.pxPrevious)); printf("/**************************结束***************************/\r\n\r\n");插入操作
vListInsert((List_t*)&TestList, (ListItem_t*)&ListItem2);- 功能:将
ListItem2插入到TestList列表中。 - 解释:此操作将
ListItem2插入到TestList列表中的正确位置,通常是按照xItemValue的顺序。在插入之前,ListItem1已经在列表中,ListItem2会根据它的xItemValue值决定它的插入位置。
打印列表项的指针状态
打印输出语句的目的是检查
ListItem2插入后列表项之间的连接状态:printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext));- 功能:打印
TestList的xListEnd成员的pxNext指针的地址。 - 解释:插入
ListItem2后,xListEnd的pxNext应该指向ListItem2,因为ListItem2将成为列表的新的末尾项。
printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext));- 功能:打印
ListItem1的pxNext指针的地址。 - 解释:插入
ListItem2后,ListItem1的pxNext应该指向ListItem2,因为ListItem2将跟在ListItem1之后。
printf("ListItem2->pxNext\t\t0x%p\r\n", (ListItem2.pxNext));- 功能:打印
ListItem2的pxNext指针的地址。 - 解释:插入
ListItem2后,ListItem2的pxNext应该指向xListEnd,即列表的末尾。
printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious));- 功能:打印
TestList的xListEnd成员的pxPrevious指针的地址。 - 解释:插入
ListItem2后,xListEnd的pxPrevious应该指向ListItem2,因为ListItem2现在是列表的最后一个有效项。
printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious));- 功能:打印
ListItem1的pxPrevious指针的地址。 - 解释:插入
ListItem2后,ListItem1的pxPrevious应该指向xListEnd,因为ListItem1之前的项是ListItem2。
printf("ListItem2->pxPrevious\t\t0x%p\r\n", (ListItem2.pxPrevious));- 功能:打印
ListItem2的pxPrevious指针的地址。 - 解释:插入
ListItem2后,ListItem2的pxPrevious应该指向ListItem1,因为ListItem1是ListItem2的前一个项。
综合分析
在插入
ListItem2后,TestList->xListEnd->pxNext应该指向ListItem1。ListItem1->pxNext应该指向ListItem2。ListItem2->pxNext应该指向xListEnd。TestList->xListEnd->pxPrevious应该指向ListItem2。ListItem1->pxPrevious应该指向xListEnd。ListItem2->pxPrevious应该指向ListItem1。
实验结果


- 功能:将
-
列表项3插入列表
在插入
ListItem3到TestList列表中的过程中,我们需要分析其如何影响列表的结构。以下是详细的分析,包括插入ListItem3后预期的指针状态。列表状态分析
printf("/*****************第五步:列表项3插入列表******************/\r\n"); vListInsert((List_t*)&TestList, (ListItem_t*)&ListItem3); /* 插入 ListItem3 */ printf("项目\t\t\t\t地址\r\n"); printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext)); printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext)); printf("ListItem2->pxNext\t\t0x%p\r\n", (ListItem2.pxNext)); printf("ListItem3->pxNext\t\t0x%p\r\n", (ListItem3.pxNext)); printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious)); printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious)); printf("ListItem2->pxPrevious\t\t0x%p\r\n", (ListItem2.pxPrevious)); printf("ListItem3->pxPrevious\t\t0x%p\r\n", (ListItem3.pxPrevious)); printf("/**************************结束***************************/\r\n\r\n");结果分析:
-
TestList->xListEnd->pxNext- 功能:打印
TestList的xListEnd成员的pxNext指针的地址。 - 预期结果:指向
ListItem1。在插入ListItem3后,xListEnd的pxNext应指向列表的第一个有效项ListItem1。
- 功能:打印
-
ListItem1->pxNext- 功能:打印
ListItem1的pxNext指针的地址。 - 预期结果:指向
ListItem3。ListItem1的pxNext应指向ListItem3,因为ListItem3被插入在ListItem1和ListItem2之间。
- 功能:打印
-
ListItem2->pxNext- 功能:打印
ListItem2的pxNext指针的地址。 - 预期结果:指向
TestList->xListEnd。ListItem2是列表的最后一个有效项,因此它的pxNext应指向xListEnd。
- 功能:打印
-
ListItem3->pxNext- 功能:打印
ListItem3的pxNext指针的地址。 - 预期结果:指向
ListItem2。ListItem3被插入在ListItem1和ListItem2之间,因此它的pxNext应指向ListItem2。
- 功能:打印
-
TestList->xListEnd->pxPrevious- 功能:打印
TestList的xListEnd成员的pxPrevious指针的地址。 - 预期结果:指向
ListItem2。xListEnd的pxPrevious应指向列表的最后一个有效项,即ListItem2。
- 功能:打印
-
ListItem1->pxPrevious- 功能:打印
ListItem1的pxPrevious指针的地址。 - 预期结果:指向
xListEnd。在插入ListItem3后,ListItem1的pxPrevious应指向xListEnd,因为ListItem1是第一个有效项。
- 功能:打印
-
ListItem2->pxPrevious- 功能:打印
ListItem2的pxPrevious指针的地址。 - 预期结果:指向
ListItem3。ListItem2的pxPrevious应指向ListItem3,因为ListItem3是ListItem2的前一个项。
- 功能:打印
-
ListItem3->pxPrevious- 功能:打印
ListItem3的pxPrevious指针的地址。 - 预期结果:指向
ListItem1。ListItem3被插入在ListItem1和ListItem2之间,因此它的pxPrevious应指向ListItem1。
- 功能:打印
总结
在插入
ListItem3后,TestList->xListEnd->pxNext应指向ListItem1。ListItem1->pxNext应指向ListItem3。ListItem3->pxNext应指向ListItem2。ListItem2->pxNext应指向xListEnd。TestList->xListEnd->pxPrevious应指向ListItem2。ListItem1->pxPrevious应指向xListEnd。ListItem3->pxPrevious应指向ListItem1。ListItem2->pxPrevious应指向ListItem3.
实验结果


-
-
移除列表项2
printf("/*******************第六步:移除列表项2********************/\r\n"); uxListRemove((ListItem_t* )&ListItem2); /* 移除列表项 */ printf("项目\t\t\t\t地址\r\n"); printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext)); printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext)); printf("ListItem3->pxNext\t\t0x%p\r\n", (ListItem3.pxNext)); printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious)); printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious)); printf("ListItem3->pxPrevious\t\t0x%p\r\n", (ListItem3.pxPrevious)); printf("/**************************结束***************************/\r\n\r\n");在调用
uxListRemove后,ListItem2将被从列表中断开,列表的其他项应保持连接。移除
ListItem2后的列表状态在移除
ListItem2后,ListItem3和ListItem1之间的连接将直接建立,ListItem2将不再存在于列表中。具体指针更新如下:-
TestList->xListEnd->pxNext- 功能:打印
TestList的xListEnd成员的pxNext指针的地址。 - 预期结果:指向
ListItem1。xListEnd的pxNext应该仍指向列表的第一个有效项ListItem1。
- 功能:打印
-
ListItem1->pxNext- 功能:打印
ListItem1的pxNext指针的地址。 - 预期结果:指向
ListItem3。
- 功能:打印
-
ListItem3->pxNext- 功能:打印
ListItem3的pxNext指针的地址。 - 预期结果:指向
TestList->xListEnd。在ListItem2被移除后,ListItem3的pxNext应指向xListEnd。
- 功能:打印
-
TestList->xListEnd->pxPrevious- 功能:打印
TestList的xListEnd成员的pxPrevious指针的地址。 - 预期结果:指向
ListItem3。xListEnd的pxPrevious应指向列表的最后一个有效项,即ListItem3,因为ListItem2已被移除。
- 功能:打印
-
ListItem1->pxPrevious- 功能:打印
ListItem1的pxPrevious指针的地址。 - 预期结果:指向
TestList->xListEnd。ListItem1的pxPrevious应指向xListEnd,因为ListItem1是列表中的第一个有效项。
- 功能:打印
-
ListItem3->pxPrevious- 功能:打印
ListItem3的pxPrevious指针的地址。 - 预期结果:指向
ListItem1。ListItem3的pxPrevious应指向ListItem1,因为ListItem3是ListItem1的下一个有效项。
- 功能:打印
总结
在移除
ListItem2后,TestList->xListEnd->pxNext应指向ListItem1。ListItem1->pxNext应指向ListItem3。ListItem3->pxNext应指向xListEnd。TestList->xListEnd->pxPrevious应指向ListItem3。ListItem1->pxPrevious应指向xListEnd。ListItem3->pxPrevious应指向ListItem1。
实验结果
-
-


-
列表末尾添加列表项2
打印输出分析
printf("/****************第七步:列表末尾添加列表项2****************/\r\n"); vListInsertEnd((List_t*)&TestList, (ListItem_t*)&ListItem2); /* 添加 ListItem2 到末尾 */ printf("项目\t\t\t\t地址\r\n"); printf("TestList->pxIndex\t\t0x%p\r\n", TestList.pxIndex); printf("TestList->xListEnd->pxNext\t0x%p\r\n", (TestList.xListEnd.pxNext)); printf("ListItem1->pxNext\t\t0x%p\r\n", (ListItem1.pxNext)); printf("ListItem2->pxNext\t\t0x%p\r\n", (ListItem2.pxNext)); printf("ListItem3->pxNext\t\t0x%p\r\n", (ListItem3.pxNext)); printf("TestList->xListEnd->pxPrevious\t0x%p\r\n", (TestList.xListEnd.pxPrevious)); printf("ListItem1->pxPrevious\t\t0x%p\r\n", (ListItem1.pxPrevious)); printf("ListItem2->pxPrevious\t\t0x%p\r\n", (ListItem2.pxPrevious)); printf("ListItem3->pxPrevious\t\t0x%p\r\n", (ListItem3.pxPrevious)); printf("/************************实验结束***************************/\r\n");解释每个打印语句的含义:
-
TestList->pxIndex- 功能:打印
TestList的pxIndex成员的地址。 - 预期结果:应指向
ListItem1。pxIndex通常指向列表中的第一个有效项。
- 功能:打印
-
TestList->xListEnd->pxNext- 功能:打印
TestList的xListEnd成员的pxNext指针的地址。 - 预期结果:应指向
ListItem1。xListEnd的pxNext应指向列表的第一个有效项。
- 功能:打印
-
ListItem1->pxNext- 功能:打印
ListItem1的pxNext指针的地址。 - 预期结果:应指向
ListItem3。ListItem1的pxNext应指向ListItem3。
- 功能:打印
-
ListItem2->pxNext- 功能:打印
ListItem2的pxNext指针的地址。 - 预期结果:应指向
TestList->xListEnd。因为ListItem2被重新添加到列表的末尾,它的pxNext应指向xListEnd。
- 功能:打印
-
ListItem3->pxNext- 功能:打印
ListItem3的pxNext指针的地址。 - 预期结果:应指向
ListItem2。ListItem3的pxNext应指向ListItem2,因为ListItem2被添加到ListItem3的后面。
- 功能:打印
-
TestList->xListEnd->pxPrevious- 功能:打印
TestList的xListEnd成员的pxPrevious指针的地址。 - 预期结果:应指向
ListItem2。xListEnd的pxPrevious应指向列表的最后一个有效项,即ListItem2。
- 功能:打印
-
ListItem1->pxPrevious- 功能:打印
ListItem1的pxPrevious指针的地址。 - 预期结果:应指向
TestList->xListEnd。因为ListItem1是列表中的第一个有效项,它的pxPrevious应指向xListEnd。
- 功能:打印
-
ListItem2->pxPrevious- 功能:打印
ListItem2的pxPrevious指针的地址。 - 预期结果:应指向
ListItem3。因为ListItem2被添加到列表的末尾,它的pxPrevious应指向ListItem3。
- 功能:打印
-
ListItem3->pxPrevious- 功能:打印
ListItem3的pxPrevious指针的地址。 - 预期结果:应指向
ListItem1。因为ListItem3现在是ListItem2的前一个项,它的pxPrevious应指向ListItem1。
总结
在将
ListItem2添加到列表末尾后,TestList->pxIndex应指向ListItem1。TestList->xListEnd->pxNext应指向ListItem1。ListItem1->pxNext应指向ListItem3。ListItem2->pxNext应指向xListEnd。ListItem3->pxNext应指向ListItem2。TestList->xListEnd->pxPrevious应指向ListItem2。ListItem1->pxPrevious应指向xListEnd。ListItem2->pxPrevious应指向ListItem3。ListItem3->pxPrevious应指向ListItem1。
- 功能:打印
实验结果:


-
相关文章:
07.FreeRTOS列表与列表项
文章目录 07. FreeRTOS列表与列表项1. 列表和列表项的简介2. 列表相关API函数3. 代码验证 07. FreeRTOS列表与列表项 1. 列表和列表项的简介 列表的定义: typedef struct xLIST {listFIRST_LIST_INTEGRITY_CHECK_VALUE /* 校验值 */volatile UBaseType_t uxN…...
餐饮业油烟净化器安装势在必行,切勿侥幸
我最近分析了餐饮市场的油烟净化器等产品报告,解决了餐饮业厨房油腻的难题,更加方便了在餐饮业和商业场所有需求的小伙伴们。 随着环保法规的日益严格和公众环保意识的提升,餐饮业油烟排放问题成为社会关注的焦点。油烟不仅影响环境质量&am…...
SpringBoot集成阿里百炼大模型 原子的学习日记Day01
文章目录 概要下一章SpringBoot集成阿里百炼大模型(多轮对话) 原子的学习日记Day02 整体架构流程技术名词解释集成步骤1,选择大模型以及获取自己的api-key(前面还有一步开通服务就没有展示啦!)2,…...
【网络编程】网络原理(一)
系列文章目录 1、 初识网络 2、网络编程的基础使用(一) 文章目录 系列文章目录前言一、端口号的使用二、UDP报文学习1.报文格式2.MD5算法 总结 前言 在前文中,主要对UDP和TCP协议有了简单的了解,而这两种协议是负责传输层的内容…...
鲁班上门维修安装系统源码开发之功能模式
鲁班上门维修安装系统在当今的趋势呈现出显著的增长与创新。随着物联网、智能家居的普及,以及消费者对便捷、高效生活方式的追求,鲁班上门维修安装系统凭借其多渠道预约、智能派单、在线支付与费用明细透明等优势,赢得了市场的广泛认可。 …...
图数据处理的新时代:阿里FraphCompute与蚂蚁金服TuGraph对比综述
目录 前言 阿里FraphCompute与蚂蚁金服TuGraph的主要特性和功能的比较: 阿里FraphCompute与蚂蚁金服TuGraph在不同应用场景分析对比: 阿里FraphCompute与蚂蚁金服TuGraph未来趋势的对比: FraphCompute与TuGraph详解 缺点劣势深入比较 前言…...
InnoDB引擎下SQL的执行流程
SQL执行流程 连接器 客户端连接驱动与mysql连接池连接 半双工通信传入客户端的sql 查询缓存(8.0之后没有) 删除原因 如果每次查询条件不同导致命中率低没有命中缓存 创建新缓存在创建缓存的时候会添加表级锁缓存更新需要批量失效 sql解析器 对传入的sql 词法分析 分解成各种t…...
Java小白入门到实战应用教程-重写和重载
引言 在上一节中我们学习了面向对象中的继承,然后在那一节中我们提到了一个知识点叫做:重写。 通过上节的代码样例我们也观察到了,重写是发生在子类和父类的这种继承关系中。 继承的特点就是提取所有子类共有的属性和方法,但是…...
微力同步如何安装使用并使用内网穿透配置公网地址远程访问
文章目录 1.前言2. 微力同步网站搭建2.1 微力同步下载和安装2.2 微力同步网页测试2.3 内网穿透工具安装 3.本地网页发布3.1 Cpolar云端设置3.2 Cpolar本地设置 4. 公网访问测试5. 结语 1.前言 私有云盘作为云存储概念的延伸,虽然谈不上多么新颖,但是其广…...
nginx负载聚能
一、负载均衡 早期的网站流量和业务功能都比较简单,单台服务器足以满足基本的需求, 但是随着互联网的发展,业务流量越来越大并且业务逻辑也跟着越来越复 杂,单台服务器的性能及单点故障问题就凸显出来了,因此需要多台服…...
Python进阶 JSON数据,pyecharts制图
目录 json数据格式的转换 什么是json json本质 注意 pyecharts快速入门 画一个最简单的折线图 使用全局配置选项优化折线图 总结 json数据格式的转换 什么是json 一种轻量级的数据交换格式,可以按json指定的格式去组织和封装数据 json本质 带有特定格式的…...
polyglot,一个有趣的 Python 库!
更多资料获取 📚 个人网站:ipengtao.com 大家好,今天为大家分享一个有趣的 Python 库 - polyglot。 Github地址:https://github.com/aboSamoor/polyglot 在处理多语言文本时,解析和翻译不同语言的文本数据是一个常见…...
4.3.语言模型
语言模型 假设长度为 T T T的文本序列中的词元依次为 x 1 , x 2 , ⋯ , x T x_1,x_2,\cdots,x_T x1,x2,⋯,xT。 于是, x T x_T xT( 1 ≤ t ≤ T 1\le t\le T 1≤t≤T) 可以被认为是文本序列在时间步 t t t处的观测或标签。 在给定这样的文本…...
(学习总结10)C++类和对象1
C类和对象1 一、类的定义1.类定义格式2.访问限定符3. 类域 二、实例化1.实例化概念2.对象大小 三、this指针四、C和C语言实现Stack对比 以下代码环境在 VS2022。 一、类的定义 1.类定义格式 class 为定义类的关键字,Stack 为类的名字, { } 中为类的主体…...
进击大数据系列(一):Hadoop 基本概念与生态介绍
进击大数据系列(一):Hadoop 基本概念与生态介绍-腾讯云开发者社区-腾讯云 Hadoop 简介-CSDN博客 hadoop-common-3.2.1.jar hadoop-mapreduce-client-core-3.2.1.jar hadoop-hdfs-3.2.1.jar hadoop-core 依赖之间关系...
评价类算法--模糊综合评价算法模型
一.概述 二.经典集合和模糊集合的基本概念 经常采用向量表示法来进行表示 三.隶属函数的三种确定方法 其中,梯形分布用得最多 四.应用:模糊综合评价 对应一个指定的评语: 选择一个方案:...
哪些系统需要按照等保2.0进行定级?
等保2.0适用的系统类型 根据等保2.0的要求,需要进行定级的系统主要包括但不限于以下几类: 基础信息网络:包括互联网、内部网络、虚拟专用网络等。云计算平台/系统:涵盖公有云、私有云、混合云等多种部署模式的云服务平台。大数据…...
自注意力和位置编码
一、自注意力 1、给定一个由词元组成的输入序列x1,…,xn, 其中任意xi∈R^d(1≤i≤n)。 该序列的自注意力输出为一个长度相同的序列 y1,…,yn,其中: 2、自注意力池化层将xi当作key,value,query来…...
“文件夹提示无法访问?高效数据恢复策略全解析“
一、现象解析:文件夹为何提示无法访问? 在日常使用电脑的过程中,不少用户可能会突然遇到文件夹提示“无法访问”的尴尬情况。这一提示不仅阻断了对重要文件的即时访问,还可能预示着数据丢失的风险。造成这一现象的原因多种多样&a…...
结构开发笔记(一):外壳IP防水等级与IP防水铝壳体初步选型
若该文为原创文章,转载请注明原文出处 本文章博客地址:https://hpzwl.blog.csdn.net/article/details/140928101 长沙红胖子Qt(长沙创微智科)博文大全:开发技术集合(包含Qt实用技术、树莓派、三维、OpenCV…...
如何将联系人从 iPhone 转移到 Android
从 iPhone 换到 Android 手机时,你可能需要保留重要的数据,例如通讯录。好在,将通讯录从 iPhone 转移到 Android 手机非常简单,你可以从本文中学习 6 种可靠的方法,确保随时保持连接,不错过任何信息。 第 1…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...
【python异步多线程】异步多线程爬虫代码示例
claude生成的python多线程、异步代码示例,模拟20个网页的爬取,每个网页假设要0.5-2秒完成。 代码 Python多线程爬虫教程 核心概念 多线程:允许程序同时执行多个任务,提高IO密集型任务(如网络请求)的效率…...
佰力博科技与您探讨热释电测量的几种方法
热释电的测量主要涉及热释电系数的测定,这是表征热释电材料性能的重要参数。热释电系数的测量方法主要包括静态法、动态法和积分电荷法。其中,积分电荷法最为常用,其原理是通过测量在电容器上积累的热释电电荷,从而确定热释电系数…...
JVM虚拟机:内存结构、垃圾回收、性能优化
1、JVM虚拟机的简介 Java 虚拟机(Java Virtual Machine 简称:JVM)是运行所有 Java 程序的抽象计算机,是 Java 语言的运行环境,实现了 Java 程序的跨平台特性。JVM 屏蔽了与具体操作系统平台相关的信息,使得 Java 程序只需生成在 JVM 上运行的目标代码(字节码),就可以…...
Web中间件--tomcat学习
Web中间件–tomcat Java虚拟机详解 什么是JAVA虚拟机 Java虚拟机是一个抽象的计算机,它可以执行Java字节码。Java虚拟机是Java平台的一部分,Java平台由Java语言、Java API和Java虚拟机组成。Java虚拟机的主要作用是将Java字节码转换为机器代码&#x…...
解读《网络安全法》最新修订,把握网络安全新趋势
《网络安全法》自2017年施行以来,在维护网络空间安全方面发挥了重要作用。但随着网络环境的日益复杂,网络攻击、数据泄露等事件频发,现行法律已难以完全适应新的风险挑战。 2025年3月28日,国家网信办会同相关部门起草了《网络安全…...
【java】【服务器】线程上下文丢失 是指什么
目录 ■前言 ■正文开始 线程上下文的核心组成部分 为什么会出现上下文丢失? 直观示例说明 为什么上下文如此重要? 解决上下文丢失的关键 总结 ■如果我想在servlet中使用线程,代码应该如何实现 推荐方案:使用 ManagedE…...
Java设计模式:责任链模式
一、什么是责任链模式? 责任链模式(Chain of Responsibility Pattern) 是一种 行为型设计模式,它通过将请求沿着一条处理链传递,直到某个对象处理它为止。这种模式的核心思想是 解耦请求的发送者和接收者,…...
数据可视化交互
目录 【实验目的】 【实验原理】 【实验环境】 【实验步骤】 一、安装 pyecharts 二、下载数据 三、实验任务 实验 1:AQI 横向对比条形图 代码说明: 运行结果: 实验 2:AQI 等级分布饼图 实验 3:多城市 AQI…...
