博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
利用 ProtoThreads实现Arduino多线程处理(2)
阅读量:5774 次
发布时间:2019-06-18

本文共 3182 字,大约阅读时间需要 10 分钟。

转载请注明:@小五义QQ群:64770604

感谢小V分享给大家的博文。

       我在做产品设计的课题的时候,小五义推荐我使用Protothread这个库来进行编写,研究了之后应用于自己的设计上效果还不错,应小五义的请写了这个Protothread的介绍,谈不上懂,就浅浅谈一谈我的理解,帮助大家应用,如果有错误的,欢迎指教。

     Protothread(以下简称PT),是牛逼哄哄的来自瑞典皇家啥啥啥的Adam Dunkels大师编写的用于协程的一个库,你说协程是啥,简单的理解就是类似多任务处理器。当你开始做有关硬件C语言的编程之后,你会发现你的核心芯片不是无敌的,资源是有限的,这个时候合理安排资源就极为重要了,对于8位16位的单片机来说,用真正的多任务处理几乎是无法承受的,这种时候就需要PT的出马了。

        举个例子,你有两个触发端,分别是你老妈和你老婆,当你老婆生气的时候,你要跪键盘10秒,你老妈叫你的时候,你就要去给她捶背15秒,问题是你的脑子只有8位或者16位,一个时间点只能做一件事情。现在你老婆生气了,你就跪在键盘上,用最基础的delay函数数数:1秒,2秒,3秒……这时候你老妈叫你给她捶背,可是你听不见,因为你在数数,如果你老妈耐心好还好,等你跪完再叫你给她捶背,如果她老人家耐心差,叫一遍就不叫了,那可就完蛋了,直接跟你老婆开干,当然你唯一收获的就是不用跪了,准备完蛋吧。

        PT的有一个能力就是你跪下之后,不用你数数,它帮你数数,然后你膝盖就保持跪的状态,然后双手给你老妈捶背,10秒到后,它跟你讲:“诶,气管炎,10秒到了。”这个时候你想不想起来看你自己了,反正老婆给的任务完成了。

        当然这只是一个很狭窄的例子,不要狭隘的理解为PT只是给你计数,timer只是PT的一个方便的应用,这是使PT提醒的原因不是计时多久,而是一个条件,条件是数完十秒,所以说PT很灵活,你可以给把触发端给PT,比如你为了向老婆展示你的诚心,就一直跪着键盘,这时候不是叫PT数数,告诉PT,如果老妈叫你,就提醒你去敲背,这样都是可以的。

        原理大致如此,然后上几个常用的宏定义吧:

PT_INIT(pt)   初始化任务变量,只在初始化函数中执行一次就行 

PT_BEGIN(pt)  启动任务处理,放在函数开始处 
PT_END(pt)   结束任务,放在函数的最后 
PT_WAIT_UNTIL(pt, condition) 等待某个条件(条件可以为时钟或其它变量,IO等)成立,否则直接退出本函数,下一次进入本函数就直接跳到这个地方判断 

        想要真正的理解PT,自然是需要看懂源代码的,我不会那么无聊给你一个一个罗列,我就给你看几个精髓的,

#define PT_BEGIN(pt) switch((pt)->lc) {case 0;#define PT_WAIT_UNTIL(pt,condition)   (pt)->lc = __LINE__;   case __LINE__:                                       if(!(condition))  return

        啊哈哈哈哈,是不是没看懂!!!我一开始也是这样的!!!如果你看懂了你现在真的可以离开了,我没啥可以告诉你的了。。。

     如果你一头雾水,心生怯意,那我来给你打一级强心针吧!

#define PT_END(pt) }

     怎么样!是不是不敢相信!就一个“}”,所以那些大神跟你讲你PT_BEGIN之后,一定要一个PT_END,知道为啥了吧!因为你看看PT_BEGIN,里面一开始有个“{”。

     其实你懂了PT_WAIT_UNTIL之后,你对PT就算是基本的了解,我们一句一句来,首先pt是啥,pt就是一个struct变量,里面有什么呢?里面就一个lc,lc就是用来记住当前的位置,然后 下次进入函数的时候就直接从当前位置开始的,那又怎么做到从当前位置开始呢!这个时候就是展现Adam牛逼哄哄的时候了,_LINE_闪亮登场,_LINE_的作用就是能取到当前_LINE_所在位置的行数,现在你知道了当前的位置,你怎么到这个位置呢,就用的switch case语句,用static pt记录_LINE_的行数,然后通过switch((pt)->lc)跳到case _LINE_,怎么样,理解了之后是不是感叹Adam机智!

     最后就简单的我下面列一个示例吧,是按照arduino的编译器格式写的,就按照上面说的老妈老婆的例子,我个人不敢给老妈老婆分高下,一个管我一生,一个管我下半身,所以就老婆叫跪就跪10秒,不然不跪,老妈叫敲背就敲背15秒,否则浪费。

#include
//声明protothread#include
//声明pt.timer#define ANGEROFWIFE 1//老婆的愤怒#define MOTHERCALL 2//老母的呼唤#define KNEE 3//遭殃的膝盖#define HAND 4//幸福的双手static struct pt pt1,pt2;PT_timer servotimer1;PT_timer servotimer2;//定义两个计时器void setup(){ pinMode(ANGEROFWIFE,INPUT); pinMode(MOTHERCALL,INPUT); pinMode(KNEE,OUTPUT); pinMode(HAND,OUTPUT);//定义端口输入输出 PT_INIT(&pt1); PT_INIT(&pt2);//两个线程初始化,其实就是pt->lc归零}static int mission1(struct pt *pt){ PT_BEGIN(pt);//线程开始 while(1) { digitalWrite(KNEE,LOW);//老婆不生气我不跪 PT_WAIT_UNTIL(pt,digitalRead(ANGEROFWIFE));//等老婆生气 digitalWrite(KNEE,HIGH);//我擦真生气了,赶紧跪 servotimer1.setTimer(10000);//设定跪10秒,这里单位1为一微妙 PT_WAIT_UNTIL(pt,(servotimer1.Expired()));//当时间溢出,这里是10秒就结束}PT_END(pt);//线程结束}Static int mission2(struct pt *pt){ PT_BEGIN(pt); While(1){ digitalWrite(HAND,LOW); PT_WAIT_UNTIL(pt,digitalRead(MOTHERCALL)); digitalWrite(HAND,HIGH); servotiemr2.setTimer(15000); PT_WAIT_UNTIL(pt,(servotimer2.Expired()));} PT_END(pt);}void loop(){ mission1(&pt1); mission2(&pt2);//循环执行,特步,永不停息}

        上述只是关于PT一个很简单的例子,有一点需要提醒的是,尽量不要在线程里声明变量,如果要用也要用static变量,不然很容易出错也不易察觉,更多的关于protothread详细的源代码和理解可以参照REFERENCE,TV领进门,修行在个人。多多应用,勇于尝试,你就能熟练应用这一神器。

你可能感兴趣的文章
c++的vector容器
查看>>
jsp中启动java服务报错
查看>>
第五周编程总结
查看>>
B. Little Dima and Equation
查看>>
iOS开发数据库篇—SQL
查看>>
转:HTTP/2 新特性浅析
查看>>
centos关闭邮件提醒
查看>>
React Native资料收集
查看>>
jquery javascript获得网页的高度和宽度
查看>>
barrier()函数
查看>>
(十八)js控制台方法
查看>>
VB关键字总结
查看>>
虚拟机类加载机制
查看>>
android代码生成jar包并混淆
查看>>
Java基础2-基本语法
查看>>
SPI总线通信电路设计
查看>>
一个不错的vue项目
查看>>
屏蔽指定IP访问网站
查看>>
[leetcode] 237. Delete Node in a Linked List
查看>>
python学习 第一天
查看>>