Arduino环境使用TM1638扩展版

   日期:2020-05-16     浏览:86    评论:0    
核心提示:TM1638深圳天微生产的一种带键盘扫描接口的LED(发光二极管显示器)驱动控制专用IC,内部集成有MCU数字接口、数据锁存器、LED驱动、键盘扫描等电路。使用TM1638的显式模块在市场上非常多,价格也很便宜。某宝10元左右就可以买到。例如下面的数码管+LED+按键模块(可以使用TM1638+led+key进行搜索):本文通过简单示例介绍Arduino环境中使用ESP32开发板驱动TM1638的方法。其中也包含作者本人踩过的坑,希望可以对读者有些帮助。连接UNO D1 R32和TM..php

TM1638深圳天微生产的一种带键盘扫描接口的LED(发光二极管显示器)驱动控制专用IC,内部集成有MCU数字接口、数据锁存器、LED驱动、键盘扫描等电路。使用TM1638的显式模块在市场上非常多,价格也很便宜。某宝10元左右就可以买到。例如下面的数码管+LED+按键模块(可以使用TM1638+led+key进行搜索):

本文通过简单示例介绍Arduino环境中使用ESP32开发板驱动TM1638的方法。其中也包含作者本人踩过的坑,希望可以对读者有些帮助。

 

 

连接UNO D1 R32和TM1638扩展板

下图的连法不是必须,只是后续程序的基础,由于拍照角度,文字和端口位置稍微有些偏差,实际的接法以后面的连接表为准。

连接表

Arduino D1 R32 TM1638 Led Key Board
3V3 VCC
GND GND
IO16 STB
IO17 CLK
IO25 DIO

 

代码说明

首先定义使用的每个引脚。如果读者希望使用其他的引脚,只要代码中的定义和前面的接续方式一致即可。:

const int strobe = 16;
const int clk = 17;
const int dio = 25;

从连接方式应该可以看出,数据线只有一根,也就是说所有的数据都是以串行方式传送的。第一段代码就是以串行方式写入数据的函数,内容就是从低位开始的移位和输出操作。

void writeByte(int value)     //write a byte.
{
  unsigned char i;
  pinMode(dio,OUTPUT);
  for(i=0;i<8;i++)
  {
    digitalWrite(clk,LOW);
    if(value&0x01){
      digitalWrite(dio,HIGH);
    }
    else{
      digitalWrite(dio,LOW);
    }
    value>>=1;
    digitalWrite(clk,HIGH);
  }
  digitalWrite(clk,LOW);
}

发送命令的函数就是在发送数据操作的前后有增加了strobe端的控制。

void sendCommand(uint8_t value)
{
  digitalWrite(strobe, LOW);
  writeByte(value);
  digitalWrite(strobe, HIGH);
}

初始化TM1638。首先发送两个命令:一个是设定数码管和LED的亮度;另一个是设定地址自动增加模式,这样可以简化连续/接受数据时的操作。最后TM1638的所有寄存器。

void reset()
{
  sendCommand(0x8b); //set light,0x88-0x8f
  sendCommand(0x40); // set auto increment mode
  digitalWrite(strobe, LOW);
  writeByte(0xc0);   // set starting address to 0
  for(uint8_t i = 0; i < 16; i++)
  {
    writeByte(0x00);
  }
  digitalWrite(strobe, HIGH);
}

接下来是数据读入函数,内容是从DIO端口按位接受数据并合成为一个字节。

int readByte(void)          //read a byte.
{
  unsigned char i;
  int value=0;;
  pinMode(dio,INPUT);//set input mode.
  for(i=0;i<8;i++)
  {
    value>>=1;
    digitalWrite(clk,LOW);
    delay(1);                           //if CUP is veryfast.
    if(digitalRead(dio)==HIGH)
      value|=0x80;
    digitalWrite(clk,HIGH);
  }
  digitalWrite(clk,LOW);
  pinMode(dio,OUTPUT);//set output mode.
  return value;
}

这里作者遇到了本文开头时提到的那个坑:显示都正常,只是读按钮函数只能检测到S1到S4,S5到S8无论如何也检测不到。后来经过分析觉得应该是发出clk之后,马上去读数据,由于间隔太短导致一个字节中的后面几位不能正确读出(别问为什么,我真不知道),因此在这个地方增加的一个1ms等待,就所有都OK了。原因可能是CPU太快,也可能是连线状态不好,也可能是芯片本身的问题。总之加了个1毫秒的等待就一切都好了。这个问题无论是国外还是国内的资料都没提到。

实际上可能不需要1ms,但是没有找到更短的delay函数,先凑合着用。

读按键操作会调用这个readByte函数,读出4个字节并合成一个字节。

int readButtons(void)
{
  int buttons = 0;
  digitalWrite(strobe, LOW);
  pinMode(dio, OUTPUT);
  shiftOut(dio, clk, LSBFIRST, 0x42);
  pinMode(dio, INPUT);
  for (uint8_t i = 0; i < 4; i++)
  {
    int v = readByte() << i;
    buttons |= v;
  }
  pinMode(dio, OUTPUT);
  digitalWrite(strobe, HIGH);
  return buttons;
}

指定位置设定LED状态。LED的地址在0xC1开始的奇数地址。

void setLed(int value, int position)
{
  pinMode(dio, OUTPUT);
  sendCommand(0x44);
  digitalWrite(strobe, LOW);
  writeByte(0xC1 + (position << 1));
  writeByte(value);
  digitalWrite(strobe, HIGH);
}

在数码管上显示数字。数码管的地址在0xC0开始的偶数地址。本函数将数字按照10进制位拆分并显示在数码管上。

void showNumber(int number)
{
  const int digital[]={0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F};
  int pos = 7;
  while(pos >=0){
    digitalWrite(strobe, LOW);
    writeByte(0xc0 + pos * 2);
    if(number > 0){
      int dig = number % 10;
      writeByte(digital[dig]);
      number /= 10;
    }
    else{
      writeByte(0);
    }
    digitalWrite(strobe, HIGH);
    pos--;
  }
}

buttonLoop读出8个按键状态并显示在对应的LED上。

void buttonLoop()
{
  int buttons = readButtons();
 
  for(int position = 0; position < 8; position++)
  {
    int mask = 0x1 << position;
    setLed(buttons & mask ? 1 : 0, position);
  }
}

setup和loop就简单了。

void setup()
{
  pinMode(strobe, OUTPUT);
  pinMode(clk, OUTPUT);
  pinMode(dio, OUTPUT);
  reset();
}

loop函数根据使用数码管显示循环变量同时读按键并显示到LED上。

void loop()
{
  for(int i = 0;i < 10000000; i++)
  {
    showNumber(i);
    buttonLoop();
  }
}

国外的网页使用shiftIn和shiftOut函数实现数据的写入和读出,这两个函数和readByte/writeByte是一回事。如果不是因为加那个1ms等待,shiftIn/shiftOut也很好。

 

动作状态

 

https://github.com/xueweiguo/EmbeddedCpp/blob/master/Arduino/TM1638/images/running.gif

 

参考资料

UNO D1 R32开发环境准备:

https://github.com/xueweiguo/EmbeddedCpp/blob/master/Arduino/ESP32/Startup/ESP32Startup.md

USING A TM1638 BASED BOARD WITH ARDUINO:

https://blog.3d-logic.com/2015/01/10/using-a-tm1638-based-board-with-arduino/

TM1638芯片资料

http://www.titanmec.com/index.php/project/download/id/532.html

 

原文链接:

https://github.com/xueweiguo/EmbeddedCpp/blob/master/Arduino/TM1638/TM1638.md

 

 

觉得本文有帮助?欢迎点赞并分享给更多的人。

阅读更多更新文章,请关注微信公众号【面向对象思考】

 
打赏
 本文转载自:网络 
所有权利归属于原作者,如文章来源标示错误或侵犯了您的权利请联系微信13520258486
更多>最近资讯中心
更多>最新资讯中心
0相关评论

推荐图文
推荐资讯中心
点击排行
最新信息
新手指南
采购商服务
供应商服务
交易安全
关注我们
手机网站:
新浪微博:
微信关注:

13520258486

周一至周五 9:00-18:00
(其他时间联系在线客服)

24小时在线客服