二,ESP8266 GPIO和SPI和定时器和串口(基于Lua脚本语言)

简介: 我们写lua用这个软件http://pan.baidu.com/s/1kVN09cr 密码:pfv7如果点击的时候提示安装,,安装就行,,如果没有提示呢可以,按照下面链接的提示自己安装哈,,,,,或者自己百度http://jingyan.

 

我们写lua用这个软件

http://pan.baidu.com/s/1kVN09cr 密码:pfv7

如果点击的时候提示安装,,安装就行,,如果没有提示呢可以,按照下面链接的提示自己安装哈,,,,,或者自己百度

http://jingyan.baidu.com/article/3c343ff70bc6ea0d377963df.html

 

 让这个灯亮

 

这个灯连接到了GPIO2,低电平点亮

 

gpio.mode(4,gpio.OUTPUT)--输出模式  
gpio.write(4,0)--输出低电平  

这样的话这个灯就亮了

 

可能会有疑问,明明是GPIO2为什么填4

看资料

https://nodemcu.readthedocs.io/en/master/en/modules/gpio/#gpio-module

还有一个简介版的资料

链接:http://pan.baidu.com/s/1pL3FHgf 密码:l2nm

对照表

 

现在下进去

 

可以先复位一下然后再执行操

对了要是经常出现编译或者下载到模块出问题.........一定是串口模块和WIFI模块通信之间有问题.....尽量使用好的杜邦线,,,尽量短,,,,,再者可以考虑换成pl2303或者CP2102等.......因为自己测试的这两种模块比较可靠

 

保存到芯片里面

 

 

现在控制板子上的继电器,改一下IO口就可以了

 

 

 

 

所以呢让继电器吸合

 

gpio.mode(2,gpio.OUTPUT)  
gpio.write(2,1)  

 

关于

 

gpio.read()

引脚是低电平返回0,外部引脚是高电平返回1
如果设置的输出高电平,但是如果拉低了引脚,也是返回0,,,,和控制状态没有关系,只与
当前引脚的实际高低电平有关系
 
 

 

现在看定时器函数

 

第一个参数 一共可以同时使用7个定时器,,id号呢是0-6

第二个参数是延时多少ms执行最后面的函数

第三个参数是设置是只执行一次,,还是调用启动函数时执行一下,还是一直间隔第二个参数的时间执行最后面的函数

就让它每隔1s打印一下串口打印111111

 

 

[html]  view plain copy
  1. gpio.mode(4,gpio.OUTPUT)  
  2. gpio.write(4,1)  
  3.   
  4. function aa()--定义一个函数  
  5.   print("111111")  
  6. end  
  7.   
  8. tmr.alarm(0, 1000, 1, aa)  

0代表用的标号为0的定时器

 

1000代表延时1000ms执行aa

1代表

当然您填tmr.ALARM_AUTO也行,,,,,太长啦.....我还是选择填1

如果填写0,那么会延时1000ms然后打印一下111111,,就是第二个参数

如果还想执行一下就调用tmr.start(0)这个里面的0代表的标号为0的定时器....

当然啦关闭这个定时器tmr.start(0)

感觉第一个参数  tmr.ALARM_SINGLE = -1...只是感觉没有试过

 

 

更方便点呢

 

 

[html]  view plain copy
  1. gpio.mode(4,gpio.OUTPUT)  
  2. gpio.write(4,1)  
  3.   
  4.   
  5. tmr.alarm(1, 1000, 1, function()  
  6.     print("111111")  
  7. end)  


可以看一下我这篇文章

 

http://www.cnblogs.com/yangfengwu/p/6404695.html

 

 

剩余的定时器的方法呢有可能tmr.delay(us) 会用到,,就是延时多少us......

 

看一下串口....

说一下哈...配置串口是一件及其危险的事情,为什么这样说呢!!

我们烧固件是通过串口,把我们编写的.lua文件保存到模块里面也是通过串口

如果说哈模块一上电就执行了我们配置的串口程序,,,,,那么我们想把新写的程序发给模块就可能写不进去.....造成一个bug了,,,,,

解决方案是

一,重新烧写固件

但是呢也有可能重新烧写固件也不能把以前写的.lua文件清除

二,先烧写别的固件,再烧写现在的

比如:可以先烧写一个AT指令的固件,让它运行一下,然后再重新烧写lua开发版本的,但是也有可能不能把以前写的.lua文件清除

三,修改程序保存的偏移量

 

您想哈,,我们的.lua程序肯定也在flash里面,,,我烧写固件时把偏移量调大肯定能把我们写的.lua代码给清除掉.要是调整后出现一直发乱码,,恭喜你哈,肯定覆盖到了先前的.lua了...然后呢我再把偏移量调整到0000重新烧写就好啦...

 

四,其实呢根本解决方案是烧写一个空的bin文件把原先的全都清空

这是空的bin文件

链接:http://pan.baidu.com/s/1cHFko2 密码:g6nu

可以看这个,关于各个型号的模块的Flash大小

http://wiki.ai-thinker.com/_media/esp8266/esp8266_module_list.png

如果出现

 

 

 等着格式化文件系统,,就等着哈

 

 

 

好现在做一个程序我发给串口什么数据,串口就回给我什么数据

下面的功能可以实现但是别这样写,我只是来解释函数....否则亲们就按照上面的方法刷固件把

 

 

第一个参数只有"data",,说明是接收串口的数据

第二个参数写0就是说串口接收的数据存在了Revdata里面,,,说一下哈Revdata是一个字符串变量,,接着看后面也会提及...写小于255的数就代表,接收到这个数目才执行uart.write(0,Revdata) ,,,,如果写别的"X" ,就代表接收到字符X后就执行uart.write(0,Revdata) 

第三个参数是函数,串口接收到数据就会调用这个函数,并把接收到的数据存到Revdata里面然后调用uart.write(0,Revdata) 把数据再传回串口,,,参数0是说用串口0,,当然咱们使用的就是串口0,,,.其实还有串口1,,不过呢串口1只有TX引脚引了出来,还有串口2,不过呢

即使uart.write(2,Revdata) 写上串口2,信息还会默认发给串口0

第四个参数写0吧

 

第一种方案这样写

 

 

gpio.mode(4,gpio.OUTPUT)
gpio.write(4,1)


tmr.alarm(1, 2000, 0, function()

    uart.on("data", 0,function(Revdata)

            uart.write(0,Revdata)
            
    end, 0)
    
end)

 

 


程序启动以后等待2s执行下面的函数,,定时器第三个参数写的0,所以这个定时器就关闭了

 

 

1.uart.on("data", 0,function(Revdata)  
2.  
3.            uart.write(0,Revdata)  
4.              
5.    end, 0)  

 

只要一执行这个函数,那么以后串口接收到数据就会执行

 

uart.write(0,Revdata)

 

现在用串口助手测试一下

 

现在我假如修改了程序想重新写入芯片

复位芯片后赶紧点击,,当然有两秒时间

 

 

如果程序小有时候可以直接

 

 

如果程序大点,,可以再复位一下模块然后再...当然假设下载的时间不会超过两秒

 

 

假设程序很大

复位芯片后赶紧点击,,当然有两秒时间

 

然后看这里

 

 

 

只要移除掉init.lua一切都好办

那我先右击准备好点击,,,,然后呢我复位一下模块,然后在两秒之前点击就移除了init.lua

 

 

还有

 

 

也是复位后再点击是,,,移除芯片内部所有文件..........这个随时可以用,,不用像上面似的事先编译

当然这只是在自己配置了串口之后的特殊情况下才会遇到的问题哈

 

做到这里有些人会想,能不能配置收到什么数据就去干点什么

我们就配置收到H 就控制继电器引脚输出高电平,收到L就控制继电器引脚输出低电平

 

print("Relay=1")
print("Relay=0")

 

相当于printf,,,,把里面的字符串内容发送到串口

 

不过呢这个慎用,,因为遇到'\0'就默认发送'\0'以前的数据,,,所以对于 byte类型的 0

它认为是 '\0',,,用它打印字符串还是蛮不错的

 

 

[html]  view plain copy
  1. gpio.mode(4,gpio.OUTPUT)  
  2. gpio.mode(2,gpio.OUTPUT)  
  3. gpio.write(4,1)  
  4.   
  5.   
  6.   
  7.   
  8. tmr.alarm(1, 2000, 0, function()  
  9.   
  10.   
  11.     uart.on("data", 0,function(Revdata)  
  12.   
  13.   
  14.             if Revdata == "H" then  
  15.                gpio.write(2,1)  
  16.                print("Relay=1")  
  17.             end  
  18.               
  19.             if Revdata == "L" then  
  20.                gpio.write(2,0)  
  21.                print("Relay=0")  
  22.             end  
  23.               
  24.             uart.write(0,Revdata) 
  25.               
  26.     end, 0)  
  27.       
  28. end)  

现在用串口调试助手测试

 

 

 

 

 

 

 

 

好现在换一下命令,配置收到++H 就控制继电器引脚输出高电平,收到++L就控制继电器引脚输出低电平

如果按照下面写竟然不管来了

 

[html]  view plain copy
  1. gpio.mode(4,gpio.OUTPUT)  
  2. gpio.mode(2,gpio.OUTPUT)  
  3. gpio.write(4,1)  
  4.   
  5.   
  6. tmr.alarm(1, 2000, 0, function()  
  7.   
  8.     uart.on("data", 0,function(Revdata)  
  9.   
  10.             if Revdata == "++H" then  
  11.                gpio.write(2,1)  
  12.                print("Relay=1")  
  13.             end  
  14.               
  15.             if Revdata == "++L" then  
  16.                gpio.write(2,0)  
  17.                print("Relay=0")  
  18.             end  
  19.               
  20.             uart.write(0,Revdata) -- unregister callback function  
  21.               
  22.     end, 0)  
  23.       
  24. end)  



 

现在发送的时候多加一个加号

也就是+++H    +++L

用串口调试助手来调试,,ESPlorer 这个软件的串口有些问题

 

 

您会发现居然这样可以

告诉您原因

其实是因为串口先接收了一个+ 然后又接收的++H 所以可以控制

所以先前的++H  是串口先接收了一个+  然后又接收了 +H  所以不能控制了

不信的话可以打印一下,,修改为下面的程序

 

[html]  view plain copy
  1. gpio.mode(4,gpio.OUTPUT)  
  2. gpio.mode(2,gpio.OUTPUT)  
  3. gpio.write(4,1)  
  4.   
  5.   
  6. tmr.alarm(1, 2000, 0, function()  
  7.   
  8.     uart.on("data", 0,function(Revdata)  
  9.   
  10.             if Revdata == "++H" then  
  11.                gpio.write(2,1)  
  12.                print("Relay=1")  
  13.             else  
  14.                print(Revdata)----LOOK  LOOK  LOOK      
  15.             end  
  16.               
  17.             if Revdata == "++L" then  
  18.                gpio.write(2,0)  
  19.                print("Relay=0")  
  20.             end  
  21.               
  22.             uart.write(0,Revdata) -- unregister callback function  
  23.               
  24.     end, 0)  
  25.       
  26. end)  



 

其实

这地方写0,就代表了串口接收到1个数据就会进入中断函数function(Revdata)

解决方法

 

我是不愿意使用控制接收到多少个字节,或者加入一个标志,,,,还记得AT指令要加换行不......那是因为"\r".接收到换行....

我的做法...我用定时器做空闲检测

关于Lua的部分语法可以看,,其实了解就行,,当时自己以为需要把lua学的很好才能用lua开发8266,,,最后才知道只需要了解些语法就好了,,剩下的看8266的API文档看怎么使用那些函数就行.....

 

http://www.cnblogs.com/yangfengwu/p/6357838.html

http://www.cnblogs.com/yangfengwu/p/6358444.html

http://www.cnblogs.com/yangfengwu/p/6366428.html

http://www.cnblogs.com/yangfengwu/p/6376098.html

 

 

[html]  view plain copy
  1. gpio.mode(4,gpio.OUTPUT)  
  2. gpio.mode(2,gpio.OUTPUT)  
  3. gpio.write(4,1)  
  4.   
  5.   
  6. ReadData=""  
  7. ReadDataCopy=""  
  8. ReadCnt=0  
  9. ReadCntt =0
  10.   
  11. tmr.alarm(2, 5, 1, function()  
  12.   
  13.     if  ReadCnt ~= 0 then  
  14.   
  15.         if  ReadCnt == ReadCntt then  
  16.             ReadCnt = 0  
  17.             ReadCntt = 0  
  18.             ReadDataCopy = ReadData  
  19.             ReadData = ""  
  20.             
  21.             if  ReadDataCopy == "++H" then  
  22.                 gpio.write(2,1)  
  23.             end  
  24.             
  25.             if  ReadDataCopy == "++L" then  
  26.                 gpio.write(2,0)  
  27.             end  
  28.             
  29.         else  
  30.             ReadCntt = ReadCnt      
  31.         end  
  32.          
  33.     end  
  34.      
  35. end)  
  36.   
  37.   
  38.   
  39. tmr.alarm(1, 2000, 0, function()  
  40.   
  41.     uart.on("data", 0,function(Revdata)  
  42.   
  43.            ReadData = ReadData..Revdata  
  44.            ReadCnt = ReadCnt + 1   
  45.     end, 0)  
  46.       
  47. end)  

空闲中断..............
可以先看一下自己51的程序

 

串口中断接收数据

定时器做检测

 

 

 

可以参考的文章

http://www.cnblogs.com/yangfengwu/p/6746403.html  这个较早,后期改进了

http://www.cnblogs.com/yangfengwu/p/7341056.html  在文章中找哈

http://www.cnblogs.com/yangfengwu/p/6921832.html  在文章中找哈

 

现在说一下SPI,,其实呢为什么说spi呢,,,,咱的模块最终99.99%都会设计成,发给模块串口的数据自动转发到网络,,,,,模块网络接收的数据自动转发到模块的串口对吧!!!!

那么如果想灵活的配置模块要么通过串口,要么通过其它方式比如说SPI方式....串口刚才说了主要用于用户传输数据使用.....那么我们自己写的灵活配置模块的程序也需要通过一种通信方式告诉模块哈...当然也可以选择IIC,或者自己控制模块的引脚自己编一个

感觉这篇说的够多的了,下一篇再说SPI,关键是自己去做,,自己去做,,自己去做.........

 

目录
相关文章
|
5月前
|
数据挖掘 Linux 数据处理
探索Linux下的Lua命令:轻量级脚本语言在数据处理和分析中的应用
**探索Linux上的Lua:轻量级脚本语言用于数据处理。Lua通过命令行解释器执行,适用于游戏开发、数据分析及自动化。特点包括小巧、高效、可扩展和动态类型。使用`lua`或`luajit`,配合-e、-l、-i参数执行脚本或互动模式。示例:执行`hello.lua`脚本打印"Hello, Lua!"。最佳实践涉及版本兼容、性能优化、使用C API、测试和文档编写。**
|
Ubuntu 网络协议 Linux
【Lua基础入门】解密世界上最快的脚本语言
【Lua基础入门】解密世界上最快的脚本语言
1792 1
|
存储 NoSQL Java
Lua高性能脚本语言快速入门
Lua高性能脚本语言快速入门
251 0
|
XML 存储 Java
【Lua基础 第1章】初识Lua脚本语言、数据类型、全局变量、关键字的使用
初识Lua脚本语言、数据类型、全局变量、关键字的使用
157 0
【Lua基础 第1章】初识Lua脚本语言、数据类型、全局变量、关键字的使用
|
存储 JavaScript 编译器
Lua脚本语言——Lua脚本基础语法
Lua脚本语言——Lua脚本基础语法
682 0
Lua脚本语言——Lua脚本基础语法
(一)Lua脚本语言入门
今天开始自己的Lua语言学习,Lua脚本语言,是介于应用程序和开发其应用程序的底层编程语言之间,,它很方便调用其它语言,它只是在载入时对其进行编译,而不像我们写的单片机程序是预编译的,先编译好然后写入单片机,它只是在调用时编译,所以脚本语言的灵活性很高,,,,,对于自己为什么又开始学习脚本语言了,,因为需要了   .
1315 0
|
C语言 数据安全/隐私保护
(二)Lua脚本语言入门
上一篇文章忘了插入代码了,方便粘贴复制...... 函数 对于c语言就是  void aa()//c语言是用void { print("这是一个函数") } Lua就变成了 function aa()--定义一个函数 print("这是一个函数") end aa()--调用函数   function aa(str)--定义一个函数 print("传进来的是".
1090 0
|
1月前
|
缓存 分布式计算 NoSQL
大数据-43 Redis 功能扩展 Lua 脚本 对Redis扩展 eval redis.call redis.pcall
大数据-43 Redis 功能扩展 Lua 脚本 对Redis扩展 eval redis.call redis.pcall
29 2
|
5月前
|
消息中间件 NoSQL Java
Redis系列学习文章分享---第六篇(Redis实战篇--Redis分布式锁+实现思路+误删问题+原子性+lua脚本+Redisson功能介绍+可重入锁+WatchDog机制+multiLock)
Redis系列学习文章分享---第六篇(Redis实战篇--Redis分布式锁+实现思路+误删问题+原子性+lua脚本+Redisson功能介绍+可重入锁+WatchDog机制+multiLock)
227 0
|
2月前
|
存储 JSON Ubuntu
如何使用 Lua 脚本进行更复杂的网络请求,比如 POST 请求?
如何使用 Lua 脚本进行更复杂的网络请求,比如 POST 请求?