1. 前言
近年来,随着电子产品的发展,数字日程表这项应用在人们工作和生活中起到越来越重要的作用。时间对人们来说总是那么宝贵,工作的忙碌性和繁杂性容易使人忘记当前的时间,忘记了要做的事情,当事情不是很重要的时候,这种遗忘无伤大雅。但是,遇上重要事务,一时的耽误可能酿成大祸。
因此从人们的日常生活到公司办公,从台式电脑到便携式智能手机,都要求标配上日程表的作用。人们要求随时随地都能快速准确的提醒当前事务,并且要求日程表能够更直观、更可靠、更便宜。这种要求催生了新型日程表的产生。除此之外,由于对社会责任的更多承担,人们要求所设计的产品能够产生尽量少的垃圾、能够消耗尽量少的能量。因此人们对日程表的又有了体积小、功耗低的要求。
2. 功能需求
2.1 硬件部分
整个项目在正点原子stm32f103mini开发板环境下进行。开发板主控是stm32f103rct6。
项目模块由一些部分组成:
(1)2.8寸tft触摸屏负责显示和进行交互;
(2)pcf8563t用作时钟计时,并把日期和时间显示在屏幕上。
(3)时间采用24小时制显示,上位机要支持设备端RTC日期及时间信息更新.
(4)DHT11温湿度传感器检测环境信息,并把信息显示在液晶屏幕上;
(5)使用esp8266作WiFi模块与手机app进行通信;
(6)单片机将接收到的内容存储在w25Q64内,同时可以在屏幕上指定位置将内容显示出来。内容包括具体日程的文字内容以及日程开始、结束的时间;
(7)使用蜂鸣器以及通过syn6288语音芯片合成的语音信息,通过喇叭播报实现提醒功能。在日程开始、结束前五分钟提前(这个提前提醒的时间要可以修改)通过蜂鸣器以及喇叭发出语音提示。
(8)显示屏(横屏显示)上应包含:
基础的日期、时间、温湿度信息显示,屏幕主体部分通过列表的方式显示从手机app端接收到的日程内容;屏幕上设置一个触屏按钮,按下该按钮是可以跳当前进行中的,或者还未开始的下一项即将开始的日程。
日程显示部分:因为2.8寸显示屏的空间有限,所以同屏范围内只显示一到两个日程的具体内容。需要显示出日程的文字内容,开始和结束的时间。当日程时间即将开始以及即将结束时触发语音提示;日程结束之后要从当前显示位置上清除,同时删除w25Q64上存储的信息。同时显示下一个待开始的日程。
显示的文本部分要求能够支持显示16和24大小的,包括中文字符在内的所有字符(无法兼容就做成24大小即可)。从手机端发送的中文文本信息在在屏幕上显示的同时要存储在w25q64内。并且单片机终端上要支持存储最少十五条日程内容。单片机要能识别具体日程的时间信息,根据时间排序,同时判断日程是否过时,过时的日程直接删除,删除和跳过日程不需要触发提示。
(9)语音提示内容:
1:您有待开始的日程,请注意时间。(如果可以实现将手机端输入的日程内容(主要是汉字)读出来,那么此句改为:下一项日程:XXX即将开始,请注意时间。(XXX内即为日程内容))
2:当前日程即将结束
3:连接成功(成功连上app时播报)
4:连接失败
5:日程已设置(单片机接收到手机上发送的日程内容。)
(10)每句提示播报前蜂鸣器响一声,响完后停顿一秒再播报。
2.2 软件部分
软件部分主要就是手机上的控制app,手机的app包括可以选择添加日程的按钮,可以输入信息的文本框,可以将文本框内的内容发送到单片机的按钮。同时要可以在app上查看单片机终端的已录入的日程内容,以及加入一个可以删除已录入内容的按钮。文本框分为三部分,一部分输入文本内容(两到八个汉字字符左右的长度即可。)输入开始时间的文本框,输入结束时间的文本框。
2.3 功能总结
(1)STM32采用正点原子mini板。正点原子的2.8寸tft触摸屏
(2)pcf8563t用作时钟计时,并把日期和时间显示在屏幕上
(3)DHT11温湿度传感器检测环境信息,并把信息显示在液晶屏幕上
(4)使用esp8266作WiFi模块与手机app进行通信;
(5)w25Q64 烧录字库,存放字库,存放日程提醒信息
实现思路: 将W25Q64安装FATFS文件系统,方便数据存放读取,读写日程信息,字库信息。
(6)syn6288语音芯片合成的语音信息,通过喇叭播报实现提醒功能
(7)开发手机APP输入提醒日程,单片机接收到手机上发送的日程内容。
单片机将接收到的内容存储在w25Q64内,同时可以在屏幕上指定位置将内容显示出来。内容包括具体日程的文字内容以及日程开始、结束的时间;日程信息采用文件形式存储,修改、读写都针对文件进行操作。
3. 软件运行效果
软件打开之后先输入设备端的IP地址和端口,连接成功之后就可以进行功能操作。
软件上有日程表查看页面(也就是主页面)、日志页面、新增日程提醒页面。
软件分为windows桌面版本和Android手机版本,下面演示的截图以windows桌面版本为例。
软件采用QT设计,Qt Creator是跨平台的 Qt IDE, Qt Creator 是 Qt 被 Nokia 收购后推出的一款新的轻量级集成开发环境(IDE)。此 IDE 能够跨平台运行,支持的系统包括 Linux(32 位及 64 位)、Mac OS X 以及 Windows。
Qt Creator官网下载地址:Download Qt | Embedded System | Real Time Embedded Systems | Qt
QT所有版本下载地址:Index of /archive/qt
QT环境搭建,入门开发专栏: https://blog.csdn.net/xiaolong1126626497/category_11400392.html
(1)日程表查看页面(也就是主页面),查看日程提醒事件,点击更新日程按钮
,可以从设备端获取最新的数据过来。
(2)日志页面用来查看软件与设备间交互的过程,可以调试了解发送的数据是否正常。
(3)新增日程提醒页面
在这个页面上可以填入提醒的事件内容,输入提醒的起始时间、结束时间,提前提醒的时,输入完毕后,点击新增提醒事件
按钮,就可以将数据发送给设备端,并且在主页面添加数据显示。
(4)Android手机运行效果
4. 通信协议
设备端与软件上位机之间数据交互的的协议:
(1)软件上位机对STM32发送:#update 让STM32发送当前存储的所有日程数据过来STM32向上位机返回的数据格式:$update,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)$update,2022/02/22 13:15,2022/02/23 12:17,吃饭,5(2)给STM32发送一条日程数据过去格式:$add,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)$add,2022/02/22 13:15,2022/02/23 12:17,吃饭,5(3)给STM32发送校准时间 *20220222131338(4)删除STM32上存储的日程数据$del,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)$del,2022/02/22 13:15,2022/02/23 12:17,吃饭,5
5. 测试流程总结
设备端采用ESP8266与上位机进行通信,ESP8266上电初始化为AP+TCP服务器模式,设置固定端口号。
采用电脑或者手机运行APP测试之前,先搜索ESP8266创建的WIFI热点连接上,然后打开软件,在软件里输入ESP8266服务器的IP地址和端口号点击连接,连接成功之后就可以与设备端进行交互。
如果没有设备端,也可以采用网络调试助手与上位机之前交互,测试功能。
6. 硬件部分
6.1 硬件实物
板子的串口正常提示:
(1)上电提示
(2) 更新事件提示
(3)SD卡上生成的文件
6.2 外设硬件连线
(1) ESP8266 WIFIPB10--->ESP8266-RXPB11--->ESP8266-TX3.3v--->VCCGND---->GND(2) SYN6628PA2(TX)---SYN6628-RXPA3(RX)---SYN6628-TX3.3v---->VCCGND----->GND(3) DHT11 温湿度传感器PA5 ---->DHT11-OUT3.3v---->VCCGND----->GND 剩下的用的硬件是开发板本身自带--正点原子STM32F1战舰V3开发板,硬件连接详情看原理图接口。
6.3 字库创建
6.4 SD卡上存放的字库文件
7. 设备端核心代码及实现思路
如果需要整个工程直接使用,可以去这里获取: https://download.csdn.net/download/xiaolong1126626497/85892788
7.1 字库读取
目前设备端LCD屏字库存放在SD卡上,通过fatfs文件系统读取字模进行显示,这样做的优点: 更换字库方便,直接把SD卡拔出来放在电脑上拷贝字库即可。
核心代码如下:
void NT35310_DisplayGBKData(u32 x,u32 y,u32 size,u8 *p,u16 c1,u16 c2){ FIL fp; UINT br; u8 L,H; u32 Addr; u16 font_size=size/8*size; //字体占用的点阵码字节大小 u8 *buff=NULL;H=*p;L=*(p+1);if(L<0x7f)L=L-0x40;else L=L-0x41;H=H-0x81;Addr=(190*H+L)*font_size; //中文在字库里的偏移量buff=malloc(font_size); //使用的堆空间if(buff==NULL)return;switch(size){case 16:if(f_open(&fp,"0:SYSTEM/FONT/GBK16-H.DZK",FA_READ)!=FR_OK) { printf("f_open error.\r\n"); }f_lseek(&fp,Addr);f_read(&fp,buff,font_size,&br);f_close(&fp); break;case 24: f_open(&fp,"0:SYSTEM/FONT/GBK24-H.DZK",FA_READ);f_lseek(&fp,Addr);f_read(&fp,buff,font_size,&br);f_close(&fp);break;case 32:break;}//显示中文NT35310_DisplayData(x,y,size,size,buff,c1,c2);//释放空间free(buff);}void NT35310_DisplayData(u32 x,u32 y,u32 w,u32 h,u8 *p,u16 c1,u16 c2){ u16 i,j,x0=x; u8 data; u16 colortemp=POINT_COLOR; for(i=0;i<w/8*h;i++) //取出的模型总字节数 { data=p[i]; //取出数组里一个字节的数据 for(j=0;j<8;j++) { if(data&0x80)Draw_Point(x0,y,c1); //字体颜色 else { Draw_Point(x0,y,c2); //背景颜色 } data<<=1; //继续判断下一位 x0++; //继续画下一个点 } if(x0-x==w) //判断是否需要换行 { x0=x;//横坐标归位 y++; //纵坐标自增 } } POINT_COLOR=colortemp; }
7.2 解析ESP8266数据
主函数里通过轮询方式检测,ESP8266是否收到上位机的命令,收到之后进行解析处理
核心代码如下:
//ESP8266 WIFI 返回的数据if(USART3_RX_FLAG){USART3_RX_BUFFER[USART3_RX_CNT]='\0';printf("%s",USART3_RX_BUFFER);//解析WIFI返回的数据//如果是校准RTC时间 +IPD,0,15:*20220304220552if(strstr((char*)USART3_RX_BUFFER,":*")){ printf("校准时间.\r\n"); rtc_time_update((char*)USART3_RX_BUFFER);}//如果是请求更新提醒else if(strstr((char*)USART3_RX_BUFFER,"#update")){printf("请求更新事件.\r\n");update_enev();}//如果是新增提醒//+IPD,0,49:$add,2022/03/04 21:56,2022/03/04 21:56,else if(strstr((char*)USART3_RX_BUFFER,"$add")){ printf("新增提醒事件.\r\n"); add_enev((char*)USART3_RX_BUFFER);}//如果是删除某个提醒//+IPD,0,49:$del,2022/03/04 21:56,2022/03/04 21:56,水水水水,0else if(strstr((char*)USART3_RX_BUFFER,"$del")){ printf("删除某个提醒.\r\n");del_enev((char*)USART3_RX_BUFFER);}USART3_RX_CNT=0;USART3_RX_FLAG=0;}
7.3 向SD卡存放事件信息
事件提醒都是存放在SD卡上,以文件的形式存放,上面封装的几个函数里,主要是就是读写文件。
核心代码:
/*函数功能:从buf里面得到第cnt个逗号所在的位置返 回 值:0~254,代表逗号所在位置的偏移.255,代表不存在第cnt个逗号*/u8 GetCommaOffset(char *buf,u8 cnt){char *p=buf;while(cnt){if(*buf==',')cnt--;buf++;}return buf-p; //计算偏移量}/*写文件*/void FATFS_Write(const TCHAR *FileName,const char *WriteBuff){ FIL fp; FRESULT res; UINT cnt; //存放写入成功的数量 /*1. 创建文件*/ res=f_open(&fp,FileName, FA_WRITE | FA_CREATE_ALWAYS); if(res!=0) { printf("%s文件创建失败!\n",FileName); return; } /*2. 写入数据*/ res=f_write(&fp,WriteBuff,strlen(WriteBuff),&cnt); if(res!=0) { printf("%s文件写入失败!\n",FileName); return; } /*3. 关闭文件*/ f_close(&fp); printf("%s文件创建成功,成功写入:%d\n",FileName,cnt);}//提取指定逗号位置的数据void GetCommaOffsetBuff(char *buf_in,char *buf_out,u8 cnt){while (cnt){if (*buf_in == ',')cnt--;if (*buf_in == '\0')break;buf_in++;}while (*buf_in != ',' && *buf_in != '\0'){*buf_out=*buf_in;buf_in++;buf_out++;}*buf_out = '\0';}//字符串替换//sub1替换前 sub2替换后字符void StringSubstitution(char *p,char sub1,char sub2){while (*p!='\0'){if (*p == sub1){*p = sub2;}p++;}}//新增提醒 //+IPD,0,49:$add,2022/03/04 21:56,2022/03/04 21:56,void add_enev(char *p){ char buf_out[20];u8 offset = 0;char *p2;offset = GetCommaOffset(p, 3);p2 = p + offset;printf("%d,%s\r\n", offset, p2);//提取提醒的起始时间 例如:2022/03/04 21:56GetCommaOffsetBuff(p,buf_out,3);printf("buf_out1=%s\r\n",buf_out);StringSubstitution(buf_out,'/','-');StringSubstitution(buf_out, ' ','-');StringSubstitution(buf_out, ':','-');printf("buf_out2=%s\r\n", buf_out);strcat(buf_out,".ev");printf("buf_out3=%s\r\n", buf_out);FATFS_Write(buf_out, p2); //更新事件列表 update_event_list();}//删除某个提醒 //+IPD,0,49:$del,2022/03/04 21:56,2022/03/04 21:56,水水水水,0void del_enev(char *p){ char buf_out[20]; char dir_file_path[50];//提取提醒的起始时间 例如:2022/03/04 21:56GetCommaOffsetBuff(p, buf_out, 3);printf("buf_out1=%s\n", buf_out);StringSubstitution(buf_out, '/', '-');StringSubstitution(buf_out, ' ', '-');StringSubstitution(buf_out, ':', '-');printf("buf_out2=%s\n", buf_out);strcat(buf_out, ".ev");printf("buf_out3=%s\n", buf_out); //拼接目录名称 sprintf(dir_file_path,"0:/%s",buf_out); //删除文件 if(f_unlink(dir_file_path)==FR_OK) { printf("%s\r\n删除成功.",dir_file_path); } else { printf("%s\r\n删除失败.",dir_file_path); } //更新事件列表 update_event_list();}//向APP终端更新提醒//+IPD,0,7:#update/*STM32向上位机返回的数据格式:$update,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)$update,2022/02/22 13:15,2022/02/23 12:17,吃饭,5*/u8 update_enev(void){// u8 buff[]="$update,2022/02/22 13:15,2022/02/23 12:17,吃饭,5";// ESP8266_ServerSendData(0,buff,strlen((char*)buff)); char *path="0:/"; //目录位置 DIR dir; FIL file; //文件指针 FRESULT res; FILINFO fno; //存放读取的文件信息 char *abs_path=NULL; char cmd[]="$update,"; //请求的命令头 char out_buff[100]; UINT cnt; strcpy(out_buff,cmd); /*1. 打开目录*/ res=f_opendir(&dir,path); if(res!=FR_OK)return res; /*2. 循环读取目录*/ while(1) { res=f_readdir(&dir,&fno); if(fno.fname[0] == 0 || res!=0)break; //printf("文件名称: %s,文件大小: %ld 字节\r\n",fno.fname,fno.fsize); /*过滤目录*/ if(strstr(fno.fname,".ev")) { //申请存放文件名称的长度 abs_path=malloc(strlen(path)+strlen(fno.fname)+1); if(abs_path==NULL)break; strcpy(abs_path,path); strcat(abs_path,"/"); strcat(abs_path,fno.fname); //读取文件数据 f_open(&file,abs_path,FA_READ); f_read(&file,out_buff+strlen(cmd),100,&cnt); free(abs_path); printf("abs_path=%s,读取:%d\r\n",abs_path,cnt); out_buff[cnt+strlen(cmd)]='\0'; //发送给上位机 ESP8266_ServerSendData(0,(u8*)out_buff,strlen((char*)out_buff)); printf("发送:%s\r\n",out_buff); delay_ms(100); } } /*3. 关闭目录*/ f_closedir(&dir); return 0;}//提取数据,存放到全局事件结构体里//参数: i 索引值 buf_out 源数据内容void ExtractData(int i,char *buf_out){//std_to_sec(u16 syear, u8 smon, u8 sday, u8 hour, u8 min)char buf_num[50]; //格式化处理StringSubstitution(buf_out, '/', ',');StringSubstitution(buf_out, ' ', ',');StringSubstitution(buf_out, ':', ',');printf("buf_out=%s\r\n",buf_out);//2022-03-05-10-11-2022-03-05-10-11-打酱油-5//1. 提取起始时间GetCommaOffsetBuff(buf_out,buf_num,0);event_buff[i].s_year = atoi(buf_num);printf("起始-年: %d,%s\r\n", event_buff[i].s_year, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 1);event_buff[i].s_month = atoi(buf_num);printf("起始-月: %d,%s\r\n", event_buff[i].s_month, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 2);event_buff[i].s_date = atoi(buf_num);printf("起始-日: %d,%s\r\n", event_buff[i].s_date, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 3);event_buff[i].s_hour = atoi(buf_num);printf("起始-时: %d,%s\r\n", event_buff[i].s_hour, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 4);event_buff[i].s_min = atoi(buf_num);printf("起始-分: %d,%s\r\n", event_buff[i].s_min, buf_num);//2. 提取结束时间GetCommaOffsetBuff(buf_out, buf_num, 5);event_buff[i].e_year = atoi(buf_num);printf("结束-年: %d,%s\r\n", event_buff[i].e_year, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 6);event_buff[i].e_month = atoi(buf_num);printf("结束-月: %d,%s\r\n", event_buff[i].e_month, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 7);event_buff[i].e_date = atoi(buf_num);printf("结束-日: %d,%s\r\n", event_buff[i].e_date, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 8);event_buff[i].e_hour = atoi(buf_num);printf("结束-时: %d,%s\r\n", event_buff[i].e_hour, buf_num);GetCommaOffsetBuff(buf_out, buf_num, 9);event_buff[i].e_min = atoi(buf_num);printf("结束-分: %d,%s\r\n", event_buff[i].e_min, buf_num);//3. 提取事件内容GetCommaOffsetBuff(buf_out, buf_num,10);strcpy((char*)event_buff[i].event_buff, buf_num);printf("事件内容:%s\r\n", event_buff[i].event_buff);//4. 提前提醒时间GetCommaOffsetBuff(buf_out, buf_num, 11);event_buff[i].time_min_advance = atoi(buf_num);printf("提前提醒时间:%d分钟\r\n", event_buff[i].time_min_advance);//5. 设置标志位event_buff[i].flag = 1;}//最多15个事件提醒struct Event event_buff[15];//更新事件列表u8 update_event_list(void){ char *path="0:/"; //目录位置 DIR dir; FIL file; //文件指针 FRESULT res; FILINFO fno; //存放读取的文件信息 char *abs_path=NULL; char out_buff[100]; UINT cnt; int index=0; //数组索引 //先将数组全部清0 memset(event_buff,0,sizeof(event_buff)); /*1. 打开目录*/ res=f_opendir(&dir,path); if(res!=FR_OK)return res; /*2. 循环读取目录*/ while(1) { res=f_readdir(&dir,&fno); if(fno.fname[0] == 0 || res!=0)break; //printf("文件名称: %s,文件大小: %ld 字节\r\n",fno.fname,fno.fsize); /*过滤目录*/ if(strstr(fno.fname,".ev")) { //申请存放文件名称的长度 abs_path=malloc(strlen(path)+strlen(fno.fname)+1); if(abs_path==NULL)break; strcpy(abs_path,path); strcat(abs_path,"/"); strcat(abs_path,fno.fname); //读取文件数据 f_open(&file,abs_path,FA_READ); f_read(&file,out_buff,100,&cnt); free(abs_path); out_buff[cnt]='\0'; //添加结束符 printf("abs_path=%s,读取:%d,%s\r\n",abs_path,cnt,out_buff); //解析数据存放到数组里 //2022/03/05 10:11,2022/03/05 10:11,打酱油,5 ExtractData(index++,out_buff); if(index>=15)break; //最大存放15个事件 } } /*3. 关闭目录*/ f_closedir(&dir); return 0;}
7.4 时间更新
RTC时间更新,通过上位机发送时间进行更新。
核心代码:
void rtc_time_update(char *buff){ char *time; /*判断是否收到客户端发来的数据 */ char *p=strstr((char*)buff,"+IPD"); if(p!=NULL) //正常数据格式: +IPD,0,7:LED1_ON +IPD,0表示第0个客户端 7:LED1_ON表示数据长度与数据 { /*解析上位机发来的数据*/ p=strstr((char*)buff,":"); if(p!=NULL) { p+=1; //向后偏移1个字节 if(*p=='*') //设置RTC时间 { p+=1; //向后偏移,指向正确的时间 time=p; calendar.w_year=(time[0]-48)*1000+(time[1]-48)*100+(time[2]-48)*10+(time[3]-48)*1; calendar.w_month=(time[4]-48)*10+(time[5]-48)*1; calendar.w_date=(time[6]-48)*10+(time[7]-48)*1; calendar.hour=(time[8]-48)*10+(time[9]-48)*1; calendar.min=(time[10]-48)*10+(time[11]-48)*1; calendar.sec=(time[12]-48)*10+(time[13]-48)*1; RTC_Set(calendar.w_year, calendar.w_month, calendar.w_date, calendar.hour, calendar.min, calendar.sec); printf("时间设置成功:%s\r\n",buff); } } }}
7.5 LCD屏几个主要页面
主循环里通过轮询按键,检测是否需要切换显示页面。
目前设计有3个主页面+N个事件显示页面。
(1)页面1: 模拟电子时钟页面
(2)页面2:日历显示页面
(3)页面3-N : 待办事件显示页面
7.6 RTC时钟
RTC开启了秒中断,在秒中断里绘制模拟时钟页面,更新当前的系统时间。
7.7 事件时间判断页面
主函数里使用定时器1,以10秒的频率,判断待办事件时间是否到达,是否需要语音播报。
/*函数功能: 定时器1的更新中断服务函数*/#include "app.h"#include "rtc.h"#include "syn6628.h"u32 time_ev_cnt=0;void TIM1_UP_IRQHandler(void){ int i=0; u32 t1=0; u32 t2=0; u32 t3=0; char dir_file_path[50]; if(TIM1->SR&1<<0) { TIM1->SR&=~(1<<0); } time_ev_cnt++; //10秒时间 if(time_ev_cnt>=2) { time_ev_cnt=0; printf("10秒时间到达.\r\n"); for(i=0;i<15;i++) { if(event_buff[i].flag==0)break; //当前时间 t1=std_to_sec(calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour,calendar.min); //提醒开始时间 t2=std_to_sec(event_buff[i].s_year,event_buff[i].s_month,event_buff[i].s_date,event_buff[i].s_hour,event_buff[i].s_min); //提醒结束时间 t3=std_to_sec(event_buff[i].e_year,event_buff[i].e_month,event_buff[i].e_date,event_buff[i].s_hour,event_buff[i].e_min); //加上提前提醒的分钟 t2+=event_buff[i].time_min_advance*60; //判断有没有时间满足需求,满足就语音提示 if(t1>=t2 && t1<=t3) { //时间达到,语音播报 SYN6288_Speech("您有待开始的日程,请注意"); //如果需要播放具体的内容,直接播放结构体里的成员即可 } //如果时间超过,就可以删除事件 if(t1>=t3) { //删除SD卡上的这个文件 //拼接路径 sprintf(dir_file_path,"0:/%d-%d-%d-%d-%d.ev",event_buff[i].s_year,event_buff[i].s_month,event_buff[i].s_date,event_buff[i].s_hour,event_buff[i].s_min); if(f_unlink(dir_file_path)==FR_OK) { printf("%s\r\n删除成功.",dir_file_path); //更新事件列表 update_event_list(); } else { printf("%s\r\n删除失败.",dir_file_path); } } } }}