非常实用,推荐一种面向对象思维的单片机程序框架

举报
小麦大叔 发表于 2021/12/05 00:42:13 2021/12/05
【摘要】 点击上方“小麦大叔”,选择“置顶/星标公众号” 福利干货,第一时间送达 大家好,我是小麦,今天分享一篇单片机程序框架的文章。 程序架构重要性 很多人尤其是初学者在写代码的时候往往都是想一点写一点,最开始没有一个整体的规划,导致后面代码越写越乱,bug不断。 最终代码跑起来看似没有问题(有可能也真的没有问题),但是系统的可...

点击上方“小麦大叔”,选择“置顶/星标公众号”

福利干货,第一时间送达

大家好,我是小麦,今天分享一篇单片机程序框架的文章。

程序架构重要性

很多人尤其是初学者在写代码的时候往往都是想一点写一点,最开始没有一个整体的规划,导致后面代码越写越乱,bug不断。

最终代码跑起来看似没有问题(有可能也真的没有问题),但是系统的可扩展性很差,添加一个功能的时候会浪费大量的时间,甚至导致整个代码的崩溃。

所以,在一个项目开始的时候多花一些时间在代码的架构设计上是十分有必要的代码架构确定好了之后你会发现敲代码的时候会特别快,并且在后期调试的时候也不会像无头苍蝇一样胡乱找问题。当然,调试也是一门技术。

在学习实时操作系统的过程中,发现实时操作系统框架与个人的业务代码之间的耦合性就非常低,都是只需要将业务代码通过一定的接口函数注册好后就交给操作系统托管了,十分方便。

但是操作系统的调度过于复杂,这里就使用操作系统的思维方式来重构这个时间片轮询框架。实现该框架的完全解耦,用户只需要包含头文件,并且在使用过程中不需要改动已经写好的库文件。

Demo

首先来个demo,该demo是使用电脑开两个线程:一个线程模拟单片机的定时器中断产生时间片轮询个时钟,另一个线程则模拟主函数中一直运行的时间片轮询调度程序。


   
  1. #include <thread>
  2. #include <stdio.h>
  3. #include <windows.h>
  4. #include "timeslice.h"
  5. // 创建5个任务对象
  6. TimesilceTaskObj task_1, task_2, task_3, task_4, task_5;
  7. // 具体的任务函数
  8. void task1_hdl()
  9. {
  10. printf(">> task 1 is running ...\n");
  11. }
  12. void task2_hdl()
  13. {
  14. printf(">> task 2 is running ...\n");
  15. }
  16. void task3_hdl()
  17. {
  18. printf(">> task 3 is running ...\n");
  19. }
  20. void task4_hdl()
  21. {
  22. printf(">> task 4 is running ...\n");
  23. }
  24. void task5_hdl()
  25. {
  26. printf(">> task 5 is running ...\n");
  27. }
  28. // 初始化任务对象,并且将任务添加到时间片轮询调度中
  29. void task_init()
  30. {
  31. timeslice_task_init(&task_1, task1_hdl, 1, 10);
  32. timeslice_task_init(&task_2, task2_hdl, 2, 20);
  33. timeslice_task_init(&task_3, task3_hdl, 3, 30);
  34. timeslice_task_init(&task_4, task4_hdl, 4, 40);
  35. timeslice_task_init(&task_5, task5_hdl, 5, 50);
  36. timeslice_task_add(&task_1);
  37. timeslice_task_add(&task_2);
  38. timeslice_task_add(&task_3);
  39. timeslice_task_add(&task_4);
  40. timeslice_task_add(&task_5);
  41. }
  42. // 开两个线程模拟在单片机上的运行过程
  43. void timeslice_exec_thread()
  44. {
  45. while (true)
  46. {
  47. timeslice_exec();
  48. }
  49. }
  50. void timeslice_tick_thread()
  51. {
  52. while (true)
  53. {
  54. timeslice_tick();
  55. Sleep(10);
  56. }
  57. }
  58. int main()
  59. {
  60. task_init();
  61. printf(">> task num: %d\n", timeslice_get_task_num());
  62. printf(">> task len: %d\n", timeslice_get_task_timeslice_len(&task_3));
  63. timeslice_task_del(&task_2);
  64. printf(">> delet task 2\n");
  65. printf(">> task 2 is exist: %d\n", timeslice_task_isexist(&task_2));
  66. printf(">> task num: %d\n", timeslice_get_task_num());
  67. timeslice_task_del(&task_5);
  68. printf(">> delet task 5\n");
  69. printf(">> task num: %d\n", timeslice_get_task_num());
  70. printf(">> task 3 is exist: %d\n", timeslice_task_isexist(&task_3));
  71. timeslice_task_add(&task_2);
  72. printf(">> add task 2\n");
  73. printf(">> task 2 is exist: %d\n", timeslice_task_isexist(&task_2));
  74. timeslice_task_add(&task_5);
  75. printf(">> add task 5\n");
  76. printf(">> task num: %d\n", timeslice_get_task_num());
  77. printf("\n\n========timeslice running===========\n");
  78. std::thread thread_1(timeslice_exec_thread);
  79. std::thread thread_2(timeslice_tick_thread);
  80. thread_1.join();
  81. thread_2.join();
  82. return 0;
  83. }

运行结果如下:

488392b185279e0a5fbc2a61a1185a07.png

由以上例子可见,这个框架使用十分方便,甚至可以完全不知道其原理,仅仅通过几个简单的接口就可以迅速创建任务并加入到时间片轮询的框架中,十分好用。

时间片轮询架构

其实该部分主要使用了面向对象的思维,使用结构体作为对象,并使用结构体指针作为参数传递,这样作可以节省资源,并且有着极高的运行效率。

其中最难的部分是侵入式链表的使用,这种链表在一些操作系统内核中使用十分广泛,这里是参考RT-Thread实时操作系统中的侵入式链表实现。

h文件:


   
  1. #ifndef _TIMESLICE_H
  2. #define _TIMESLICE_H
  3. #include "./list.h"
  4. typedef enum {
  5. TASK_STOP,
  6. TASK_RUN
  7. } IsTaskRun;
  8. typedef struct timesilce
  9. {
  10. unsigned int id;
  11. void (*task_hdl)(void);
  12. IsTaskRun is_run;
  13. unsigned int timer;
  14. unsigned int timeslice_len;
  15. ListObj timeslice_task_list;
  16. } TimesilceTaskObj;
  17. void timeslice_exec(void);
  18. void timeslice_tick(void);
  19. void timeslice_task_init(TimesilceTaskObj* obj, void (*task_hdl)(void), unsigned int id, unsigned int timeslice_len);
  20. void timeslice_task_add(TimesilceTaskObj* obj);
  21. void timeslice_task_del(TimesilceTaskObj* obj);
  22. unsigned int timeslice_get_task_timeslice_len(TimesilceTaskObj* obj);
  23. unsigned int timeslice_get_task_num(void);
  24. unsigned char timeslice_task_isexist(TimesilceTaskObj* obj);
  25. #endif

c文件:


   
  1. #include "./timeslice.h"
  2. static LIST_HEAD(timeslice_task_list);
  3. void timeslice_exec()
  4. {
  5. ListObj* node;
  6. TimesilceTaskObj* task;
  7. list_for_each(node, &timeslice_task_list)
  8. {
  9. task = list_entry(node, TimesilceTaskObj, timeslice_task_list);
  10. if (task->is_run == TASK_RUN)
  11. {
  12. task->task_hdl();
  13. task->is_run = TASK_STOP;
  14. }
  15. }
  16. }
  17. void timeslice_tick()
  18. {
  19. ListObj* node;
  20. TimesilceTaskObj* task;
  21. list_for_each(node, &timeslice_task_list)
  22. {
  23. task = list_entry(node, TimesilceTaskObj, timeslice_task_list);
  24. if (task->timer != 0)
  25. {
  26. task->timer--;
  27. if (task->timer == 0)
  28. {
  29. task->is_run = TASK_RUN;
  30. task->timer = task->timeslice_len;
  31. }
  32. }
  33. }
  34. }
  35. unsigned int timeslice_get_task_num()
  36. {
  37. return list_len(&timeslice_task_list);
  38. }
  39. void timeslice_task_init(TimesilceTaskObj* obj, void (*task_hdl)(void), unsigned int id, unsigned int timeslice_len)
  40. {
  41. obj->id = id;
  42. obj->is_run = TASK_STOP;
  43. obj->task_hdl = task_hdl;
  44. obj->timer = timeslice_len;
  45. obj->timeslice_len = timeslice_len;
  46. }
  47. void timeslice_task_add(TimesilceTaskObj* obj)
  48. {
  49. list_insert_before(&timeslice_task_list, &obj->timeslice_task_list);
  50. }
  51. void timeslice_task_del(TimesilceTaskObj* obj)
  52. {
  53. if (timeslice_task_isexist(obj))
  54. list_remove(&obj->timeslice_task_list);
  55. else
  56. return;
  57. }
  58. unsigned char timeslice_task_isexist(TimesilceTaskObj* obj)
  59. {
  60. unsigned char isexist = 0;
  61. ListObj* node;
  62. TimesilceTaskObj* task;
  63. list_for_each(node, &timeslice_task_list)
  64. {
  65. task = list_entry(node, TimesilceTaskObj, timeslice_task_list);
  66. if (obj->id == task->id)
  67. isexist = 1;
  68. }
  69. return isexist;
  70. }
  71. unsigned int timeslice_get_task_timeslice_len(TimesilceTaskObj* obj)
  72. {
  73. return obj->timeslice_len;
  74. }

底层侵入式双向链表

该链表是linux内核中使用十分广泛,也十分经典,其原理具体可以参考文章:

https://www.cnblogs.com/skywang12345/p/3562146.html

h文件:


   
  1. #ifndef _LIST_H
  2. #define _LIST_H
  3. #define offset_of(type, member) (unsigned long) &((type*)0)->member
  4. #define container_of(ptr, type, member) ((type *)((char *)(ptr) - offset_of(type, member)))
  5. typedef struct list_structure
  6. {
  7. struct list_structure* next;
  8. struct list_structure* prev;
  9. } ListObj;
  10. #define LIST_HEAD_INIT(name) {&(name), &(name)}
  11. #define LIST_HEAD(name) ListObj name = LIST_HEAD_INIT(name)
  12. void list_init(ListObj* list);
  13. void list_insert_after(ListObj* list, ListObj* node);
  14. void list_insert_before(ListObj* list, ListObj* node);
  15. void list_remove(ListObj* node);
  16. int list_isempty(const ListObj* list);
  17. unsigned int list_len(const ListObj* list);
  18. #define list_entry(node, type, member) \
  19. container_of(node, type, member)
  20. #define list_for_each(pos, head) \
  21. for (pos = (head)->next; pos != (head); pos = pos->next)
  22. #define list_for_each_safe(pos, n, head) \
  23. for (pos = (head)->next, n = pos->next; pos != (head); \
  24. pos = n, n = pos->next)
  25. #endif

c文件:


   
  1. #include "list.h"
  2. void list_init(ListObj* list)
  3. {
  4. list->next = list->prev = list;
  5. }
  6. void list_insert_after(ListObj* list, ListObj* node)
  7. {
  8. list->next->prev = node;
  9. node->next = list->next;
  10. list->next = node;
  11. node->prev = list;
  12. }
  13. void list_insert_before(ListObj* list, ListObj* node)
  14. {
  15. list->prev->next = node;
  16. node->prev = list->prev;
  17. list->prev = node;
  18. node->next = list;
  19. }
  20. void list_remove(ListObj* node)
  21. {
  22. node->next->prev = node->prev;
  23. node->prev->next = node->next;
  24. node->next = node->prev = node;
  25. }
  26. int list_isempty(const ListObj* list)
  27. {
  28. return list->next == list;
  29. }
  30. unsigned int list_len(const ListObj* list)
  31. {
  32. unsigned int len = 0;
  33. const ListObj* p = list;
  34. while (p->next != list)
  35. {
  36. p = p->next;
  37. len++;
  38. }
  39. return len;
  40. }

到此,一个全新的,完全解耦的,十分方便易用时间片轮询框架完成。

—— The End ——

推荐好文  点击蓝色字体即可跳转

 状态机的三种骚操作,值得你了解

☞ 推荐一个直接用于项目开发的PID库!很好用,很稳定e77e72ec023c95761280c0de01e3c147.gif

☞ 这14种嵌入式实时系统,你用过哪些?

☞ 推荐一款我私藏已久的串口示波神器

欢迎转发、留言、点赞、分享给你的朋友,感谢您的支持!

点击上方名片关注公众号

分享 💬  点赞 👍  在看 ❤️ 

以“三连”行动支持优质内容!

文章来源: great.blog.csdn.net,作者:小麦大叔,版权归原作者所有,如需转载,请联系作者。

原文链接:great.blog.csdn.net/article/details/120407781

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。