当前位置: 首页 > news >正文

7,FreeRTOS列表与列表项的插入删除

一、实验目标


        创建三个动态任务,栈空间大小均为128字。startTask、Task1、Task2。startTask仅运行一次,负责task1、task2任务的创建,startTask任务的删除。Task1负责初始化列表、列表项123,并进行列表项的插入实验与删除实验。Task2负责5S闪烁一次LED0,用于指示系统的运行状态。

注:本实验基于正点原子FreeRTOS教程的学习总结。

  二、实验准备

1.FreeRTOS的Keil动态任务创建与删除程序

2.STM3F407开发板

3.列表与列表项结构体

        列表结构体List_t如下,主要包含两个校验值、列表中包含列表项的数量、用于遍历列表项的指针、末尾列表项。

/** Definition of the type of queue used by the scheduler.*/
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;

        列表项结构体ListItem_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;                  

4.所需API函数介绍(list.c/list.)

        vListInitialise函数用来初始化列表,输入为需要初始化的列表。

void vListInitialise( List_t * const pxList ) PRIVILEGED_FUNCTION;

         vListInitialiseItem函数用来初始化列表项,输入为需要初始化的列表项。

void vListInitialiseItem( ListItem_t * const pxItem ) PRIVILEGED_FUNCTION;

         vListInsert函数用来顺序的将列表项插入列表,输入为列表、列表项。有序插入是将列表项俺早列表项值的大小,从小到大排序插入。

void vListInsert( List_t * const pxList,ListItem_t * const pxNewListItem ) PRIVILEGED_FUNCTION;

         vListInsertEnd函数用来无序的将列表项插入列表,输入为列表、列表项。无序插入是将所需插入的列表项插入到当前列表项前一个。

void vListInsertEnd( List_t * const pxList,ListItem_t * const pxNewListItem ) PRIVILEGED_FUNCTION;

        uxListRemove函数用于删除列表项,输入为待删除的列表项。

UBaseType_t uxListRemove( ListItem_t * const pxItemToRemove ) PRIVILEGED_FUNCTION;

 三、代码编写

3.1修改Task1的任务函数内容

        Task1分为五步。第一步初始化列表与列表项,第二步打印列表项和列表的地址,第三步将列表项123顺序插入到列表中,第四步移除列表项2,第五步将列表项2无序插入到列表。

void task1(void *pvParameters)
{/* 第一步:初始化列表和列表项 */vListInitialise(&TestList);                 /* 初始化列表 */vListInitialiseItem(&ListItem1);            /* 初始化列表项1 */vListInitialiseItem(&ListItem2);            /* 初始化列表项2 */vListInitialiseItem(&ListItem3);            /* 初始化列表项3 *//* 第二步:打印列表和其他列表项的地址 */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");/* 第三步:列表项1/2/3插入列表 */printf("\r\n/*****************第三步:列表项1/2/3插入列表******************/\r\n");vListInsert((List_t*    )&TestList,         /* 列表 */(ListItem_t*)&ListItem1);       /* 列表项1 */vListInsert((List_t*    )&TestList,         /* 列表 */(ListItem_t*)&ListItem2);       /* 列表项2 */vListInsert((List_t*    )&TestList,         /* 列表 */(ListItem_t*)&ListItem3);       /* 列表项3 */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");/* 第四步:移除列表项2 */printf("\r\n/*******************第四步:移除列表项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");/* 第五步:列表末尾添加列表项2 */printf("\r\n/****************第五步:列表末尾添加列表项2****************/\r\n");vListInsertEnd((List_t*     )&TestList,     /* 列表 */(ListItem_t* )&ListItem2);   /* 列表项 */printf("\r\n项目\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");while(1){vTaskDelay(10);}
}

 3.2修改Task2的任务函数内容

        Task2实现5S间隔LED0亮灭。

void task2(void *pvParameters)
{while(1){LED0_TOGGLE();vTaskDelay(5000);}
}

四、实验现象

        实验现象如下所示。可以看到在第三步中,列表项123按照列表项值的从小到大顺序排列。

http://www.dtcms.com/a/262819.html

相关文章:

  • docker安装MySQL,创建MySQL容器
  • 认识 Spring AI
  • 根据OS自动加载不同的native库和本地jar包
  • Linux驱动学习day11(定时器)
  • 百度文库智能PPT月访问量超3400万,用户规模翻倍增长
  • demo01:基于 SpringMVC 的用户管理系统
  • AlpineLinux安装部署MongoDB
  • Clickhouse源码分析-TTL执行流程
  • 杂谈-架构时代演进
  • C语言常用转换函数实现原理
  • 50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | ThemeClock(主题时钟)
  • Windows环境下Docker容器化的安装与设置指南
  • 【第二章:机器学习与神经网络概述】04.回归算法理论与实践 -(1)线性回归模型
  • AWS WebRTC:通过shell分析并发启动master后产生的日志文件
  • 御控助力打造物联网实训室,赋能职业教育高质量发展
  • 大模型-分布式推理简介
  • Linux基础环境开发工具apt、vim和gcc/g++
  • STC8H驱动两相四线步进电机
  • 基于llama-factory+ollama+vllm加速大模型训推生产
  • 大数据(4)-spark
  • Windows 开发环境部署指南:WSL、Docker Desktop、Podman Desktop 部署顺序与存储路径迁移指南
  • STM32-第一节-新建工程,GPIO,点亮LED,蜂鸣器
  • GC3910S:一款高性能双通道直流电机驱动芯片
  • 【Wireshark】高级过滤技巧精讲
  • Chromium 136 编译指南 Ubuntu篇:Python环境与开发工具配置(五)
  • 解决VSCode打开最近项目后终端shell不正常的问题
  • TCP 滑动窗口实现机制
  • 颠覆传统加密:微算法科技创新LSQb算法,提升量子图像处理速度
  • 芯谷科技--150KHz 3A PWM 降压型 DC/DC 转换器D1507
  • 【原创】【5】【视频二创工具发布】基于视觉模型+FFmpeg+MoviePy实现短视频自动化二次编辑+多赛道