看了很多的wifi时钟,于是就做了一个掌上的mini时钟,我没有安装esp8266的库文件和板级支持包,所以就打算直接用at指令来访问服务器,并从接收到的数据中获取当前时间。经测量大概3天误差在±10毫秒以内,然后系统每24小时自动进行校准时间,减少误差。最终效果还算理想。
硬件部分:
- arduino nano :体型较小,比较便宜
- esp8266_01s :01s相对于01在接线上可以不用把EN接高电平
- TM1637 :与其它四位数码管相比使用I2c接口,所需引脚大量减少
- usb转ttl:用于wifi模块的调试
软件:
1.arduino IDE
2.需要安装TM1637库文件
3.我并未安装arduino上的esp8266库文件,因此wifi模块使用at指令来进行调试的,具体操作在下面展示
库安装
打开arduino IDE再按快捷键 Crtrl + Shift + I 调出库管理,再
在输入栏上写上TM1637,点击安装。
连线部分
esp8266_01s | arduino |
---|---|
3.3v | 3.3v |
GND | GND |
RX | TX |
TX | RX |
TM1637 | arduino |
---|---|
3.3v | 3.3v |
GND | GND |
CLK | 11 |
DIO | 12 |
AT指令调试:
wifi模块接上usb转ttl,打开串口调试助手输入:
AT 若返回“ok”则表示能正常通讯,然后依次输入以下代码
(输出时在最后加上回车换行):
AT+CWMODE=3 配置成 AP+STA模式
AT+RST 重启生效
AT+CIFSR 查询设备IP
AT+CWJAP=“ssid”,“password” 连接你手机的热点“ssid”为热点的名称,"password"为密码。
AT+CIPSTART=“TCP”,“183.230.40.33”,80l连接的服务器ip 183.230.40.33/80
AT+CIPMODE=1开启透传模式
AT+CIPSEND开始透传
现在wifi模块就已经连上服务端了,这时我们在串口随便发送任意字符,都会收到服务器返回给我们的信息,这时只要把里面的时间给读取出来就行了。
这里我发送了一个’A’,服务器给我返回了带有时间的消息包。但这里的时间是美国时间,所以需要在小时上加个8小时的时差就行了。
wifi模块上电自动连接到服务器
如果我们想把wifi模块给配置好
AT+SAVETRANSLINK=1,“183.230.40.33”,80,“TCP”,100开机自动连接并进入透传。
输入这行命令这样就不用每次上电都重新发送AT指令了。
esp8266_01s获取时间
String receive = ""; //读取串口收到的字符
char info[380] = {0}; //char* 类型临时接收缓冲区
int hour_shi;
int hour_ge;
int min_shi;
int min_ge;
int sec_shi;
int sec_ge;void setup() {Serial.begin(115200); //初始化WIFI串口Serial.print("a");delay(1000);//等待连接 while (Serial.available() > 0){receive += char(Serial.read());//将串口读取的值存到字符串receive中delay(2);}strcpy(info, receive.c_str()); //String转换成char*Serial.println(info);char ret[10] = {0}; //时间存入缓冲区strncpy( ret, strstr(info, "GMT") - 9, 9 ); //将字符“GMT”前的9个字符放到ret字符数组中hour_shi = ret[0] - '0';hour_ge = ret[1] - '0';min_shi = ret[3] - '0';min_ge = ret[4] - '0';sec_shi = ret[6] - '0';sec_ge = ret[7] - '0';hour_ge += 8;//时差处理,在小时上加8,返回北京时间if(hour_ge>10){if(hour_shi==0){hour_ge = hour_ge%10;hour_shi = 1;}else//(hour_shi==1){hour_ge = hour_ge % 10;hour_shi = 2;if(hour_ge>4){hour_ge -= 4;hour_shi = 0;}}}else{hour_ge = hour_ge % 10;if(hour_shi==0){hour_shi = 1;}else{hour_shi = 2;}}Serial.print(hour_shi);//串口打印获取的时间Serial.print(hour_ge);Serial.print(":");Serial.print(min_shi);Serial.print(min_ge);Serial.print(":");Serial.print(sec_shi);Serial.println(sec_ge);delay(1000);
}
void loop()
{}
运行成功后
使用millis()函数,使程序不断进行时间累加
unsigned long time;
int hour_shi;
int hour_ge;
int min_shi;
int min_ge;
int sec_shi;
int sec_ge;void showTime(unsigned long t);
void(*restart)(void) = 0; //软重启
void adjustTime(long sec = 0);void setup() {Serial.begin(115200);
}void loop() {time = millis();//返回开始运行当前程序时的毫秒数。这个数字在大约50天后溢出,即回到零。 showTime(time/1000); adjustTime(); //默认不校对 否则sec秒校对一次时间(一天进行一次校准(86400))
}void showTime(unsigned long t)
{//网络时间矫正精确到秒t += (sec_shi*10 + sec_ge + 1); //把格式全部转换成分钟unsigned long sum_min = t/60 + ((hour_shi*10 + hour_ge)*60 + (min_shi*10 + min_ge));unsigned long temp_hour = sum_min/60%24;unsigned long temp_min = sum_min%60;unsigned long temp_sec = t%60;int h_0 = temp_hour/10;int h_1 = temp_hour%10;int m_2 = temp_min/10;int m_3 = temp_min%10;Serial.print(h_0);Serial.print(h_1);Serial.print(":");Serial.print(m_2);Serial.print(m_3);Serial.print(":");Serial.print(temp_sec/10);Serial.println(temp_sec%10);
}void adjustTime(long sec)
{if (sec != 0 && time/1000 > sec) //0表示不校对 否则sec秒校对一次 restart(); //校对时间(重启)
}
在将代码进行整合,就完成大部分了
最后把TM1637的代码在加入最后进行整合就大功告成了
最后再准备用新买的3D打印机给时钟做个外壳就完成了
后面我会陆续发布我做的一些好玩的东西,如果感兴趣的话可以关注一波!
完整文件和代码 提取码:klhk
未经许可请不要转载