u8 OLED_GRAM[128][8]; void OLED_Refresh_Gram(void) { u8 i,n; for(i=0;i<8;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //???(0~7) OLED_WR_Byte (0x00,OLED_CMD); //??? OLED_WR_Byte (0x10,OLED_CMD); //??? for(n=0;n<128;n++)OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA); } } //?OLED??? //dat:???/?? //cmd:??/??? 0,???;1,???; void OLED_WR_Byte(u8 dat,u8 cmd) { u8 i; if(cmd) OLED_RS_Set(); else OLED_RS_Clr(); for(i=0;i<8;i++) { OLED_SCLK_Clr(); if(dat&0x80) OLED_SDIN_Set(); else OLED_SDIN_Clr(); OLED_SCLK_Set(); dat<<=1; } OLED_RS_Set(); } //开启OLED显示 void OLED_Display_On(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC?? OLED_WR_Byte(0X14,OLED_CMD); //DCDC ON OLED_WR_Byte(0XAF,OLED_CMD); //DISPLAY ON } //关闭OLED显示 void OLED_Display_Off(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC?? OLED_WR_Byte(0X10,OLED_CMD); //DCDC OFF OLED_WR_Byte(0XAE,OLED_CMD); //DISPLAY OFF } //清屏函数 void OLED_Clear(void) { u8 i,n; for(i=0;i<8;i++)for(n=0;n<128;n++)OLED_GRAM[n][i]=0X00; OLED_Refresh_Gram();//??? } //画点 //x:0~127 //y:0~63 //t:1 填充 0,清空 void OLED_DrawPoint(u8 x,u8 y,u8 t) { u8 pos,bx,temp=0; if(x>127||y>63)return;//???. pos=7-y/8; bx=y%8; temp=1<<(7-bx); if(t)OLED_GRAM[x][pos]|=temp; else OLED_GRAM[x][pos]&=~temp; } //在指定位置显示字符,包括部分字符 //x:0~127 //y:0~63 //mode:0,反白显示;1,正常显示 //size:选择字体 16/12 void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size,u8 mode) { u8 temp,t,t1; u8 y0=y; chr=chr-’ ';//??? for(t=0;t<size;t++) { if(size==12)temp=oled_asc2_1206[chr][t]; //??1206?? else temp=oled_asc2_1608[chr][t]; //??1608?? for(t1=0;t1<8;t1++) { if(temp&0x80)OLED_DrawPoint(x,y,mode); else OLED_DrawPoint(x,y,!mode); temp<<=1; y++; if((y-y0)==size) { y=y0; x++; break; } } } } //m^n函数 u32 oled_pow(u8 m,u8 n) { u32 result=1; while(n–)result*=m; return result; } //显示两个数字 //x,y :起点的坐标 //len :数字位数 //size:字体大小 //mode: 0,填充模式;1,叠加模式 //num:数字(0~4294967295); //void OLED_ShowNumber(u8 x,u8 y,u32 num,u8 len,u8 size) //{ // u8 t,temp; // u8 enshow=0; // for(t=0;t<len;t++) // { // temp=(num/oled_pow(10,len-t-1))%10; // if(enshow==0&&t<(len-1)) // { // if(temp==0) // { // OLED_ShowChar(x+(size/2)*t,y,’ ',size,1); // continue; // }else enshow=1; // } // OLED_ShowChar(x+(size/2)*t,y,temp+‘0’,size,1); // } //} void OLED_ShowNumber(u8 x, u8 y, int32_t num, u8 len, u8 size) { u8 t, temp; u8 enshow = 0; if (num < 0) { OLED_ShowChar(x, y, ‘-’, size, 1); num = -num; } x += size / 2; for (t = 0; t < len; t++) { temp = (num / oled_pow(10, len - t - 1)) % 10; if (enshow == 0 && t < (len - 1)) { if (temp == 0) { OLED_ShowChar(x + (size / 2) * t, y, ’ ', size, 1); continue; } else enshow = 1; } OLED_ShowChar(x + (size / 2) * t, y, temp + ‘0’, size, 1); } } //显示字符串 //x,y:起点坐标 //*p:字符串起始地址 //用16字体 void OLED_ShowString(u8 x,u8 y,const u8 *p) { #define MAX_CHAR_POSX 122 #define MAX_CHAR_POSY 58 while(*p!=‘\0’) { if(x>MAX_CHAR_POSX){x=0;y+=16;} if(y>MAX_CHAR_POSY){y=x=0;OLED_Clear();} OLED_ShowChar(x,y,*p,12,1); x+=8; p++; } } //显示浮点数字 //x,y :起点坐标 //value :要显示的值 //decimalPlaces,小数点后的位数, // 显示浮点数函数 void OLED_ShowFloat(float value, uint8_t decimalPlaces, uint8_t x, uint8_t y) { char buffer[16]; // ????????????,??????? snprintf(buffer, sizeof(buffer), "%.*f", decimalPlaces, value); // ?????????? for (uint8_t i = 0; i < strlen(buffer); i++) { OLED_ShowChar(x + i * 8, y, buffer[i], 16, 1); } } //???OLED void OLED_Init(void) { OLED_RST_Clr(); HAL_Delay(100); OLED_RST_Set(); OLED_WR_Byte(0xAE,OLED_CMD); //???? OLED_WR_Byte(0xD5,OLED_CMD); //????????,???? OLED_WR_Byte(80,OLED_CMD); //[3:0],????;[7:4],???? OLED_WR_Byte(0xA8,OLED_CMD); //?????? OLED_WR_Byte(0X3F,OLED_CMD); //??0X3F(1/64) OLED_WR_Byte(0xD3,OLED_CMD); //?????? OLED_WR_Byte(0X00,OLED_CMD); //???0 OLED_WR_Byte(0x40,OLED_CMD); //??????? [5:0],??. OLED_WR_Byte(0x8D,OLED_CMD); //????? OLED_WR_Byte(0x14,OLED_CMD); //bit2,??/?? OLED_WR_Byte(0x20,OLED_CMD); //???????? OLED_WR_Byte(0x02,OLED_CMD); //[1:0],00,?????;01,?????;10,?????;??10; OLED_WR_Byte(0xA1,OLED_CMD); //??????,bit0:0,0->0;1,0->127; OLED_WR_Byte(0xC0,OLED_CMD); //??COM????;bit3:0,????;1,????? COM[N-1]->COM0;N:???? OLED_WR_Byte(0xDA,OLED_CMD); //??COM?????? OLED_WR_Byte(0x12,OLED_CMD); //[5:4]?? OLED_WR_Byte(0x81,OLED_CMD); //????? OLED_WR_Byte(0xEF,OLED_CMD); //1~255;??0X7F (????,????) OLED_WR_Byte(0xD9,OLED_CMD); //??????? OLED_WR_Byte(0xf1,OLED_CMD); //[3:0],PHASE 1;[7:4],PHASE 2; OLED_WR_Byte(0xDB,OLED_CMD); //??VCOMH ???? OLED_WR_Byte(0x30,OLED_CMD); //[6:4] 000,0.65*vcc;001,0.77*vcc;011,0.83*vcc; OLED_WR_Byte(0xA4,OLED_CMD); //??????;bit0:1,??;0,??;(??/??) OLED_WR_Byte(0xA6,OLED_CMD); //??????;bit0:1,????;0,???? OLED_WR_Byte(0xAF,OLED_CMD); //???? OLED_Clear(); } /* OLED.c Created on: May 22, 2024 Author: 30985 / 上面的代码为keil5+cubemx环境中6针oled显示的代码,编写显示中文的函数,汉字通过取模软件得到,如下面所示{0x00,0x02,0x82,0x82,0x82,0x82,0x82,0xFE,0x82,0x82,0x82,0x82,0x82,0x02,0x00,0x00}, {0x40,0x40,0x40,0x40,0x40,0x40,0x40,0x7F,0x40,0x40,0x40,0x40,0x40,0x40,0x40,0x00},/“王”,0*/

时间: 2025-05-29 18:05:34 浏览: 21
要实现显示中文的函数,首先需要准备一个中文字库。字库可以是通过取模软件生成的点阵数据(如题目中提到的 `{0x00, 0x02, ...}`)。接下来我们将编写一个函数来显示这些汉字。 以下是完整的代码实现: ```c // 中文字符显示函数 // x: 起始横坐标 (0~127) // y: 起始纵坐标 (0~63) // hanzi_data: 汉字的点阵数据数组 (假设为16x16点阵) void OLED_ShowChinese(u8 x, u8 y, const u8* hanzi_data) { u8 i, j; for (i = 0; i < 16; i++) { // 遍历每一行 for (j = 0; j < 2; j++) { // 每行有2个字节的数据 u8 byte_data = hanzi_data[i * 2 + j]; // 获取当前字节数据 for (u8 bit = 0; bit < 8; bit++) { // 遍历每个字节中的8位 if (byte_data & (0x80 >> bit)) { // 如果该位为1,则点亮对应的像素 OLED_DrawPoint(x + j * 8 + bit, y + i, 1); } else { // 如果该位为0,则清空对应的像素 OLED_DrawPoint(x + j * 8 + bit, y + i, 0); } } } } } ``` ### 上述代码解释 1. **参数说明**: - `x` 和 `y` 是汉字在屏幕上的起始坐标。 - `hanzi_data` 是指向汉字点阵数据的指针。例如,对于“王”字,它的点阵数据为 `{0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x7F, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x00}`。 2. **逐行绘制**: - 外层循环 `for (i = 0; i < 16; i++)` 遍历汉字的每一行。 - 内层循环 `for (j = 0; j < 2; j++)` 遍历每行的两个字节。 - 最内层循环 `for (u8 bit = 0; bit < 8; bit++)` 遍历每个字节的8位。 3. **点亮或清空像素**: - 如果某一位为1,则调用 `OLED_DrawPoint(x + j * 8 + bit, y + i, 1)` 点亮对应的像素。 - 如果某一位为0,则调用 `OLED_DrawPoint(x + j * 8 + bit, y + i, 0)` 清空对应的像素。 --- ### 示例调用 假设我们要在坐标 `(10, 10)` 显示汉字“王”,其点阵数据为 `{0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x7F, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x00}`,可以这样调用: ```c const u8 hanzi_wang[] = {0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x7F, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x00}; OLED_ShowChinese(10, 10, hanzi_wang); ``` --- ### 注意事项 1. **点阵大小**: - 上述代码假设汉字点阵为16x16。如果使用其他尺寸的点阵(如12x12),需要调整代码逻辑。 2. **字库存储**: - 可以将所有汉字的点阵数据存储在一个全局数组中,并通过索引访问特定汉字的数据。 3. **性能优化**: - 如果频繁显示汉字,可以将常用汉字的点阵数据缓存到内存中,减少重复计算。 --- ###
阅读全文

相关推荐

u8 OLED_GRAM[128][8]; void OLED_Refresh_Gram(void) { u8 i,n; for(i=0;i<8;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //?????(0~7) OLED_WR_Byte (0x00,OLED_CMD); //?????????? OLED_WR_Byte (0x10,OLED_CMD); //?????????? for(n=0;n<128;n++)OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA); } } //?OLED??????? //dat:??????/?? //cmd:??/???? 0,????;1,????; void OLED_WR_Byte(u8 dat,u8 cmd) { u8 i; if(cmd) OLED_RS_Set(); else OLED_RS_Clr(); for(i=0;i<8;i++) { OLED_SCLK_Clr(); if(dat&0x80) OLED_SDIN_Set(); else OLED_SDIN_Clr(); OLED_SCLK_Set(); dat<<=1; } OLED_RS_Set(); } //开启OLED显示 void OLED_Display_On(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC?? OLED_WR_Byte(0X14,OLED_CMD); //DCDC ON OLED_WR_Byte(0XAF,OLED_CMD); //DISPLAY ON } //关闭OLED显示 void OLED_Display_Off(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC?? OLED_WR_Byte(0X10,OLED_CMD); //DCDC OFF OLED_WR_Byte(0XAE,OLED_CMD); //DISPLAY OFF } //清屏函数 void OLED_Clear(void) { u8 i,n; for(i=0;i<8;i++)for(n=0;n<128;n++)OLED_GRAM[n][i]=0X00; OLED_Refresh_Gram();//???? } //画点 //x:0~127 //y:0~63 //t:1 填充 0,清空 void OLED_DrawPoint(u8 x,u8 y,u8 t) { u8 pos,bx,temp=0; if(x>127||y>63)return;//?????. pos=7-y/8; bx=y%8; temp=1<<(7-bx); if(t)OLED_GRAM[x][pos]|=temp; else OLED_GRAM[x][pos]&=~temp; } //在指定位置显示字符,包括部分字符 //x:0~127 //y:0~63 //mode:0,反白显示;1,正常显示 //size:选择字体 16/12 void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size,u8 mode) { u8 temp,t,t1; u8 y0=y; chr=chr-' ';//??????? for(t=0;t<size;t++) { if(size==12)temp=oled_asc2_1206[chr][t]; //??1206?? else temp=oled_asc2_1608[chr][t]; //??1608?? for(t1=0;t1<8;t1++) { if(temp&0x80)OLED_DrawPoint(x,y,mode); else OLED_DrawPoint(x,y,!mode); temp<<=1; y++; if((y-y0)==size) { y=y0; x++; break; } } } } //m^n函数 u32 oled_pow(u8 m,u8 n) { u32 result=1; while(n--)result*=m; return result; } //显示两个数字 //x,y :起点的坐标 //len :数字位数 //size:字体大小 //mode: 0,填充模式;1,叠加模式 //num:数字(0~4294967295); //void OLED_ShowNumber(u8 x,u8 y,u32 num,u8 len,u8 size) //{ // u8 t,temp; // u8 enshow=0; // for(t=0;t<len;t++) // { // temp=(num/oled_pow(10,len-t-1))%10; // if(enshow==0&&t<(len-1)) // { // if(temp==0) // { // OLED_ShowChar(x+(size/2)*t,y,' ',size,1); // continue; // }else enshow=1; // } // OLED_ShowChar(x+(size/2)*t,y,temp+'0',size,1); // } //} void OLED_ShowNumber(u8 x, u8 y, int32_t num, u8 len, u8 size) { u8 t, temp; u8 enshow = 0; if (num < 0) { OLED_ShowChar(x, y, '-', size, 1); num = -num; } x += size / 2; for (t = 0; t < len; t++) { temp = (num / oled_pow(10, len - t - 1)) % 10; if (enshow == 0 && t < (len - 1)) { if (temp == 0) { OLED_ShowChar(x + (size / 2) * t, y, ' ', size, 1); continue; } else enshow = 1; } OLED_ShowChar(x + (size / 2) * t, y, temp + '0', size, 1); } } //显示字符串 //x,y:起点坐标 //*p:字符串起始地址 //用16字体 void OLED_ShowString(u8 x,u8 y,const u8 *p) { #define MAX_CHAR_POSX 122 #define MAX_CHAR_POSY 58 while(*p!='\0') { if(x>MAX_CHAR_POSX){x=0;y+=16;} if(y>MAX_CHAR_POSY){y=x=0;OLED_Clear();} OLED_ShowChar(x,y,*p,12,1); x+=8; p++; } } //显示浮点数字 //x,y :起点坐标 //value :要显示的值 //decimalPlaces,小数点后的位数, // 显示浮点数函数 void OLED_ShowFloat(float value, uint8_t decimalPlaces, uint8_t x, uint8_t y) { char buffer[16]; // ????????????,??????? snprintf(buffer, sizeof(buffer), "%.*f", decimalPlaces, value); // ?????????? for (uint8_t i = 0; i < strlen(buffer); i++) { OLED_ShowChar(x + i * 8, y, buffer[i], 16, 1); } } //???OLED void OLED_Init(void) { OLED_RST_Clr(); HAL_Delay(100); OLED_RST_Set(); OLED_WR_Byte(0xAE,OLED_CMD); //???? OLED_WR_Byte(0xD5,OLED_CMD); //????????,???? OLED_WR_Byte(80,OLED_CMD); //[3:0],????;[7:4],???? OLED_WR_Byte(0xA8,OLED_CMD); //?????? OLED_WR_Byte(0X3F,OLED_CMD); //??0X3F(1/64) OLED_WR_Byte(0xD3,OLED_CMD); //?????? OLED_WR_Byte(0X00,OLED_CMD); //???0 OLED_WR_Byte(0x40,OLED_CMD); //??????? [5:0],??. OLED_WR_Byte(0x8D,OLED_CMD); //????? OLED_WR_Byte(0x14,OLED_CMD); //bit2,??/?? OLED_WR_Byte(0x20,OLED_CMD); //???????? OLED_WR_Byte(0x02,OLED_CMD); //[1:0],00,?????;01,?????;10,?????;??10; OLED_WR_Byte(0xA1,OLED_CMD); //??????,bit0:0,0->0;1,0->127; OLED_WR_Byte(0xC0,OLED_CMD); //??COM????;bit3:0,????;1,????? COM[N-1]->COM0;N:???? OLED_WR_Byte(0xDA,OLED_CMD); //??COM?????? OLED_WR_Byte(0x12,OLED_CMD); //[5:4]?? OLED_WR_Byte(0x81,OLED_CMD); //????? OLED_WR_Byte(0xEF,OLED_CMD); //1~255;??0X7F (????,????) OLED_WR_Byte(0xD9,OLED_CMD); //??????? OLED_WR_Byte(0xf1,OLED_CMD); //[3:0],PHASE 1;[7:4],PHASE 2; OLED_WR_Byte(0xDB,OLED_CMD); //??VCOMH ???? OLED_WR_Byte(0x30,OLED_CMD); //[6:4] 000,0.65*vcc;001,0.77*vcc;011,0.83*vcc; OLED_WR_Byte(0xA4,OLED_CMD); //??????;bit0:1,??;0,??;(??/??) OLED_WR_Byte(0xA6,OLED_CMD); //??????;bit0:1,????;0,???? OLED_WR_Byte(0xAF,OLED_CMD); //???? OLED_Clear(); } /* * OLED.c * * Created on: May 22, 2024 * Author: 30985 */ 根据上面的代码,编写一个显示中文的函数

void OLED_Refresh_Gram(void) { u8 i,n; for(i=0;i<8;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //Set page address (0~7) //ʨ׃ҳַ֘è0~7é OLED_WR_Byte (0x00,OLED_CMD); //Set the display location - column low address //ʨ׃Дʾλ׃j֍ַ֘ OLED_WR_Byte (0x10,OLED_CMD); //Set the display location - column height address //ʨ׃Дʾλ׃jַٟ֘ for(n=0;n<128;n++)OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA); } } void OLED_Refresh_Line(void) { u8 i,n; for(i=0;i<2;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //Set page address (0~7) //ʨ׃ҳַ֘è0~7é OLED_WR_Byte (0x00,OLED_CMD); //Set the display location - column low address //ʨ׃Дʾλ׃j֍ַ֘ OLED_WR_Byte (0x10,OLED_CMD); //Set the display location - column height address //ʨ׃Дʾλ׃jַٟ֘ for(n=0;n<128;n++)OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA); } } /************************************************************************** Function: Refresh the OLED screen Input : Dat: data/command to write, CMD: data/command flag 0, represents the command;1, represents data Output : none گ˽٦ŜúвOLEDдɫһٶؖޚ ɫࠚӎ˽údat:Ҫдɫք˽ߝ/ļ®ìcmd:˽ߝ/ļ®Ҫ־ 0,ҭʾļ®;1,ҭʾ˽ߝ ׵ܘ ֵúϞ **************************************************************************/ void OLED_WR_Byte(u8 dat,u8 cmd) { u8 i; if(cmd) OLED_RS_Set(); else OLED_RS_Clr(); for(i=0;i<8;i++) { OLED_SCLK_Clr(); if(dat&0x80) OLED_SDIN_Set(); else OLED_SDIN_Clr(); OLED_SCLK_Set(); dat<<=1; } OLED_RS_Set(); } /************************************************************************** Function: Turn on the OLED display Input : none Output : none گ˽٦ŜúߪǴOLEDДʾ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Display_On(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC command //SET DCDCļ® OLED_WR_Byte(0X14,OLED_CMD); //DCDC ON OLED_WR_Byte(0XAF,OLED_CMD); //DISPLAY ON } /************************************************************************** Function: Turn off the OLED display Input : none Output : none گ˽٦ŜúژҕOLEDДʾ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Display_Off(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC command //SET DCDCļ® OLED_WR_Byte(0X10,OLED_CMD); //DCDC OFF OLED_WR_Byte(0XAE,OLED_CMD); //DISPLAY OFF } /************************************************************************** Function: Screen clear function, clear the screen, the entire screen is black, and did not light up the same Input : none Output : none گ˽٦Ŝúȥǁگ˽,ȥΪǁ,ֻٶǁώۚɫքìۍû֣һҹ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Clear(void) { u8 i,n; for(i=0;i<8;i++)for(n=0;n<128;n++)OLED_GRAM[n][i]=0X00; OLED_Refresh_Gram(); //Update the display //ټтДʾ } /************************************************************************** Function: Draw point Input : x,y: starting coordinate;T :1, fill,0, empty Output : none گ˽٦Ŝúۭ֣ ɫࠚӎ˽úx,y :ǰ֣ظҪ; t:1,ͮԤ,0,ȥࠕ ׵ܘ ֵúϞ **************************************************************************/ void OLED_DrawPoint(u8 x,u8 y,u8 t) { u8 pos,bx,temp=0; if(x>127||y>63)return;//ӬԶ׶Χ‹. pos=7-y/8; bx=y%8; temp=1<<(7-bx); if(t)OLED_GRAM[x][pos]|=temp; else OLED_GRAM[x][pos]&=~temp; } /************************************************************************** Function: Displays a character, including partial characters, at the specified position Input : x,y: starting coordinate;Len: The number of digits;Size: font size;Mode :0, anti-white display,1, normal display Output : none گ˽٦Ŝú՚ָ֨λ׃Дʾһٶؖػ,Ѽ(ҿؖؖػ ɫࠚӎ˽úx,y :ǰ֣ظҪ; len :˽ؖքλ˽; size:ؖͥճС; mode:0,״їДʾ,1,ֽӣДʾ ׵ܘ ֵúϞ **************************************************************************/ void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size,u8 mode) { u8 temp,t,t1; u8 y0=y; chr=chr-' '; //Get the offset value //փսƫӆ۳քֵ for(t=0;t<size;t++) { if(size==12)temp=oled_asc2_1206[chr][t]; //Invoke 1206 font //ַԃ1206ؖͥ else temp=oled_asc2_1608[chr][t]; //Invoke the 1608 font //ַԃ1608ؖͥ for(t1=0;t1<8;t1++) { if(temp&0x80)OLED_DrawPoint(x,y,mode); else OLED_DrawPoint(x,y,!mode); temp<<=1; y++; if((y-y0)==size) { y=y0; x++; break; } } } } /************************************************************************** Function: Find m to the NTH power Input : m: base number, n: power number Output : none گ˽٦ŜúȳmքnՎ׽քگ˽ ɫࠚӎ˽úmú֗˽ìnúՎ׽˽ ׵ܘ ֵúϞ **************************************************************************/ u32 oled_pow(u8 m,u8 n) { u32 result=1; while(n--)result*=m; return result; } /************************************************************************** Function: Displays 2 numbers Input : x,y: starting coordinate;Len: The number of digits;Size: font size;Mode: mode, 0, fill mode, 1, overlay mode;Num: value (0 ~ 4294967295); Output : none گ˽٦ŜúДʾ2ٶ˽ؖ ɫࠚӎ˽úx,y :ǰ֣ظҪ; len :˽ؖքλ˽; size:ؖͥճС; mode:ģʽ, 0,ͮԤģʽ, 1,־ݓģʽ; num:˽ֵ(0~4294967295); ׵ܘ ֵúϞ **************************************************************************/ void OLED_ShowNumber(u8 x,u8 y,u32 num,u8 len,u8 size) { u8 t,temp; u8 enshow=0; for(t=0;t<len;t++) { temp=(num/oled_pow(10,len-t-1))%10; if(enshow==0&&t<(len-1)) { if(temp==0) { OLED_ShowChar(x+(size/2)*t,y,' ',size,1); continue; }else enshow=1; } OLED_ShowChar(x+(size/2)*t,y,temp+'0',size,1); } } /************************************************************************** Function: Display string Input : x,y: starting coordinate;*p: starting address of the string Output : none گ˽٦ŜúДʾؖػԮ ɫࠚӎ˽úx,y :ǰ֣ظҪ; *p:ؖػԮǰʼַ֘ ׵ܘ ֵúϞ **************************************************************************/ void OLED_ShowString(u8 x,u8 y,const u8 *p) { #define MAX_CHAR_POSX 122 #define MAX_CHAR_POSY 58 while(*p!='\0') { if(x>MAX_CHAR_POSX){x=0;y+=16;} if(y>MAX_CHAR_POSY){y=x=0;OLED_Clear();} OLED_ShowChar(x,y,*p,12,1); x+=8; p++; } } void OLED_ShowString16(u8 x,u8 y,const u8 *p) { #define MAX_CHAR_POSX 122 #define MAX_CHAR_POSY 58 while(*p!='\0') { if(x>MAX_CHAR_POSX){x=0;y+=16;} if(y>MAX_CHAR_POSY){y=x=0;OLED_Clear();} OLED_ShowChar(x,y,*p,16,1); x+=8; p++; } } /************************************************************************** Function: Initialize the OLED Input : none Output : none گ˽٦ŜúԵʼۯOLED ɫࠚӎ˽: Ϟ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; // 启用 GPIOD 时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE); // 配置 PD11~PD14 为推挽输出 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11 | GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_Init(GPIOD, &GPIO_InitStructure); // OLED 复位 OLED_RST_Clr(); delay_ms(100); // 延时 100ms OLED_RST_Set(); // OLED 寄存器初始化 OLED_WR_Byte(0xAE, OLED_CMD); // 关闭显示 OLED_WR_Byte(0xD5, OLED_CMD); // 设置时钟分频 OLED_WR_Byte(80, OLED_CMD); // 推荐值:80 OLED_WR_Byte(0xA8, OLED_CMD); // 设置驱动路数 OLED_WR_Byte(0x3F, OLED_CMD); // 默认 1/64 OLED_WR_Byte(0xD3, OLED_CMD); // 设置显示偏移 OLED_WR_Byte(0x00, OLED_CMD); // 默认 0 OLED_WR_Byte(0x40, OLED_CMD); // 设置起始行 OLED_WR_Byte(0x8D, OLED_CMD); // 充电泵 OLED_WR_Byte(0x14, OLED_CMD); // 开启充电泵 OLED_WR_Byte(0x20, OLED_CMD); // 内存地址模式 OLED_WR_Byte(0x02, OLED_CMD); // 页地址模式 OLED_WR_Byte(0xA1, OLED_CMD); // 段重定义 OLED_WR_Byte(0xC0, OLED_CMD); // COM 扫描方向 OLED_WR_Byte(0xDA, OLED_CMD); // COM 硬件配置 OLED_WR_Byte(0x12, OLED_CMD); // 默认配置 OLED_WR_Byte(0x81, OLED_CMD); // 对比度设置 OLED_WR_Byte(0xEF, OLED_CMD); // 默认值(可调整) OLED_WR_Byte(0xD9, OLED_CMD); // 预充电周期 OLED_WR_Byte(0xF1, OLED_CMD); // 默认值 OLED_WR_Byte(0xDB, OLED_CMD); // VCOMH 电压 OLED_WR_Byte(0x30, OLED_CMD); // 默认值 OLED_WR_Byte(0xA4, OLED_CMD); // 全局显示开启 OLED_WR_Byte(0xA6, OLED_CMD); // 正常显示模式 OLED_WR_Byte(0xAF, OLED_CMD); // 开启显示 OLED_Clear(); // 清屏 }

u8 OLED_GRAM[128][8]; /************************************************************************** Function: Refresh the OLED screen Input : none Output : none گ˽٦ŜúˢтOLEDǁĻ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Refresh_Gram(void) { u8 i,n; for(i=0;i<8;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //Set page address (0~7) //ʨ׃ҳַ֘è0~7é OLED_WR_Byte (0x00,OLED_CMD); //Set the display location - column low address //ʨ׃Дʾλ׃j֍ַ֘ OLED_WR_Byte (0x10,OLED_CMD); //Set the display location - column height address //ʨ׃Дʾλ׃jַٟ֘ for(n=0;n<128;n++)OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA); } } void OLED_Refresh_Line(void) { u8 i,n; for(i=0;i<2;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //Set page address (0~7) //ʨ׃ҳַ֘è0~7é OLED_WR_Byte (0x00,OLED_CMD); //Set the display location - column low address //ʨ׃Дʾλ׃j֍ַ֘ OLED_WR_Byte (0x10,OLED_CMD); //Set the display location - column height address //ʨ׃Дʾλ׃jַٟ֘ for(n=0;n<128;n++)OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA); } } /************************************************************************** Function: Refresh the OLED screen Input : Dat: data/command to write, CMD: data/command flag 0, represents the command;1, represents data Output : none گ˽٦ŜúвOLEDдɫһٶؖޚ ɫࠚӎ˽údat:Ҫдɫք˽ߝ/ļ®ìcmd:˽ߝ/ļ®Ҫ־ 0,ҭʾļ®;1,ҭʾ˽ߝ ׵ܘ ֵúϞ **************************************************************************/ void OLED_WR_Byte(u8 dat,u8 cmd) { u8 i; if(cmd) OLED_RS_Set(); else OLED_RS_Clr(); for(i=0;i<8;i++) { OLED_SCLK_Clr(); if(dat&0x80) OLED_SDIN_Set(); else OLED_SDIN_Clr(); OLED_SCLK_Set(); dat<<=1; } OLED_RS_Set(); } /************************************************************************** Function: Turn on the OLED display Input : none Output : none گ˽٦ŜúߪǴOLEDДʾ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Display_On(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC command //SET DCDCļ® OLED_WR_Byte(0X14,OLED_CMD); //DCDC ON OLED_WR_Byte(0XAF,OLED_CMD); //DISPLAY ON } /************************************************************************** Function: Turn off the OLED display Input : none Output : none گ˽٦ŜúژҕOLEDДʾ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Display_Off(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC command //SET DCDCļ® OLED_WR_Byte(0X10,OLED_CMD); //DCDC OFF OLED_WR_Byte(0XAE,OLED_CMD); //DISPLAY OFF } /************************************************************************** Function: Screen clear function, clear the screen, the entire screen is black, and did not light up the same Input : none Output : none گ˽٦Ŝúȥǁگ˽,ȥΪǁ,ֻٶǁώۚɫքìۍû֣һҹ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Clear(void) { u8 i,n; for(i=0;i<8;i++)for(n=0;n<128;n++)OLED_GRAM[n][i]=0X00; OLED_Refresh_Gram(); //Update the display //ټтДʾ } /************************************************************************** Function: Draw point Input : x,y: starting coordinate;T :1, fill,0, empty Output : none گ˽٦Ŝúۭ֣ ɫࠚӎ˽úx,y :ǰ֣ظҪ; t:1,ͮԤ,0,ȥࠕ ׵ܘ ֵúϞ **************************************************************************/ void OLED_DrawPoint(u8 x,u8 y,u8 t) { u8 pos,bx,temp=0; if(x>127||y>63)return;//ӬԶ׶Χ‹. pos=7-y/8; bx=y%8; temp=1<<(7-bx); if(t)OLED_GRAM[x][pos]|=temp; else OLED_GRAM[x][pos]&=~temp; } /************************************************************************** Function: Displays a character, including partial characters, at the specified position Input : x,y: starting coordinate;Len: The number of digits;Size: font size;Mode :0, anti-white display,1, normal display Output : none گ˽٦Ŝú՚ָ֨λ׃Дʾһٶؖػ,Ѽ(ҿؖؖػ ɫࠚӎ˽úx,y :ǰ֣ظҪ; len :˽ؖքλ˽; size:ؖͥճС; mode:0,״їДʾ,1,ֽӣДʾ ׵ܘ ֵúϞ **************************************************************************/ void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size,u8 mode) { u8 temp,t,t1; u8 y0=y; chr=chr-' '; //Get the offset value //փսƫӆ۳քֵ for(t=0;t<size;t++) { if(size==12)temp=oled_asc2_1206[chr][t]; //Invoke 1206 font //ַԃ1206ؖͥ else temp=oled_asc2_1608[chr][t]; //Invoke the 1608 font //ַԃ1608ؖͥ for(t1=0;t1<8;t1++) { if(temp&0x80)OLED_DrawPoint(x,y,mode); else OLED_DrawPoint(x,y,!mode); temp<<=1; y++; if((y-y0)==size) { y=y0; x++; break; } } } } /************************************************************************** Function: Find m to the NTH power Input : m: base number, n: power number Output : none گ˽٦ŜúȳmքnՎ׽քگ˽ ɫࠚӎ˽úmú֗˽ìnúՎ׽˽ ׵ܘ ֵúϞ **************************************************************************/ u32 oled_pow(u8 m,u8 n) { u32 result=1; while(n--)result*=m; return result; } /************************************************************************** Function: Displays 2 numbers Input : x,y: starting coordinate;Len: The number of digits;Size: font size;Mode: mode, 0, fill mode, 1, overlay mode;Num: value (0 ~ 4294967295); Output : none گ˽٦ŜúДʾ2ٶ˽ؖ ɫࠚӎ˽úx,y :ǰ֣ظҪ; len :˽ؖքλ˽; size:ؖͥճС; mode:ģʽ, 0,ͮԤģʽ, 1,־ݓģʽ; num:˽ֵ(0~4294967295); ׵ܘ ֵúϞ **************************************************************************/ void OLED_ShowNumber(u8 x,u8 y,u32 num,u8 len,u8 size) { u8 t,temp; u8 enshow=0; for(t=0;t<len;t++) { temp=(num/oled_pow(10,len-t-1))%10; if(enshow==0&&t<(len-1)) { if(temp==0) { OLED_ShowChar(x+(size/2)*t,y,' ',size,1); continue; }else enshow=1; } OLED_ShowChar(x+(size/2)*t,y,temp+'0',size,1); } } /************************************************************************** Function: Display string Input : x,y: starting coordinate;*p: starting address of the string Output : none گ˽٦ŜúДʾؖػԮ ɫࠚӎ˽úx,y :ǰ֣ظҪ; *p:ؖػԮǰʼַ֘ ׵ܘ ֵúϞ **************************************************************************/ void OLED_ShowString(u8 x,u8 y,const u8 *p) { #define MAX_CHAR_POSX 122 #define MAX_CHAR_POSY 58 while(*p!='\0') { if(x>MAX_CHAR_POSX){x=0;y+=16;} if(y>MAX_CHAR_POSY){y=x=0;OLED_Clear();} OLED_ShowChar(x,y,*p,12,1); x+=8; p++; } } void OLED_ShowString16(u8 x,u8 y,const u8 *p) { #define MAX_CHAR_POSX 122 #define MAX_CHAR_POSY 58 while(*p!='\0') { if(x>MAX_CHAR_POSX){x=0;y+=16;} if(y>MAX_CHAR_POSY){y=x=0;OLED_Clear();} OLED_ShowChar(x,y,*p,16,1); x+=8; p++; } } /************************************************************************** Function: Initialize the OLED Input : none Output : none گ˽٦ŜúԵʼۯOLED ɫࠚӎ˽: Ϟ ׵ܘ ֵúϞ **************************************************************************/ void OLED_Init(void) { // 1. 关闭显示 OLED_WR_Byte(0xAE, OLED_CMD); // 2. 硬件复位 OLED_RST_Clr(); delay_ms(10); OLED_RST_Set(); delay_ms(50); // 3. 重新配置GPIO GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11 | GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; GPIO_Init(GPIOD, &GPIO_InitStructure); // 4. 精简初始化序列 const uint8_t init_seq[] = { 0xAE, // 关闭显示 0xD5, 0x80, // 时钟分频 0xA8, 0x3F, // 多路复用 0xD3, 0x00, // 显示偏移 0x40, // 起始行 0x8D, 0x14, // 开启充电泵 0x20, 0x02, // 页地址模式 0xA1, // 段重定向 0xC8, // COM扫描方向 0xDA, 0x12, // COM配置 0x81, 0xEF, // 对比度 0xD9, 0xF1, // 预充电 0xDB, 0x30, // VCOMH 0xA4, // 关闭全局显示 0xA6 // 正常显示模式 }; // 直接发送所有命令(无错误检查) for(uint8_t i=0; i<sizeof(init_seq); ) { OLED_WR_Byte(init_seq[i++], OLED_CMD); // 保留关键命令的固定延时结构(但移除条件判断) if(i==7 || i==9 || i==11) delay_ms(10); } // 5. 清屏并开启显示 OLED_Clear(); OLED_WR_Byte(0xAF, OLED_CMD); }#define SDA_IN() {GPIOB->MODER&=~(3<<(11*2));GPIOB->MODER|=0<<11*2;} //PB5ˤɫģʽ #define SDA_OUT() {GPIOB->MODER&=~(3<<(11*2));GPIOB->MODER|=1<<11*2;} //PB5ˤԶģʽ //IOәطگ˽ #define IIC_SCL PBout(10) //SCL #define IIC_SDA PBout(11) //SDA #define READ_SDA PBin(11) //ˤɫSDA #ifndef I2C_Direction_Transmitter #define I2C_Direction_Transmitter ((uint8_t)0x00) #endif #ifndef I2C_Direction_Receiver #define I2C_Direction_Receiver ((uint8_t)0x01) #endif enum { I2C_ACK, I2C_NACK };void I2C_GPIOInit(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);//ʹŜGPIOBʱד GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10|GPIO_Pin_11; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;//ǕͨˤԶģʽ GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//ΆάˤԶ GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//ʏ- GPIO_Init(GPIOB, &GPIO_InitStructure);//Եʼۯ IIC_SCL=1; IIC_SDA=1; } /************************************************************************** Function: Simulate IIC start signal Input : none Output : none گ˽٦ŜúģŢIICǰʼхۅ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void I2C_Start(void) { SDA_OUT(); //sdaПˤԶ IIC_SDA=1; if(!READ_SDA)return ; IIC_SCL=1; delay_us(1); IIC_SDA=0;//START:when CLK is high,DATA change form high to low if(READ_SDA)return ; delay_us(1); IIC_SCL=0;//ǯסI2C؜Пì׼Ѹע̍ܲޓ˕˽ߝ return ; } /************************************************************************** Function: Analog IIC end signal Input : none Output : none گ˽٦ŜúģŢIICޡ˸хۅ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void I2C_Stop(void) { SDA_OUT();//sdaПˤԶ IIC_SCL=0; IIC_SDA=0;//STOP:when CLK is high DATA change form low to high delay_us(1); IIC_SCL=1; IIC_SDA=1;//ע̍I2C؜Пޡ˸хۅ delay_us(1); } bool I2C_WaiteForAck(void) { u8 ucErrTime=0; SDA_IN(); //SDAʨ׃Ϊˤɫ IIC_SDA=1; delay_us(1); IIC_SCL=1; delay_us(1); while(READ_SDA) { ucErrTime++; if(ucErrTime>50) { I2C_Stop(); return 0; } delay_us(1); } IIC_SCL=0;//ʱדˤԶ0 return 1; } /************************************************************************** Function: IIC response Input : none Output : none گ˽٦ŜúIICӦհ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void I2C_Ack(void) { IIC_SCL=0; SDA_OUT(); IIC_SDA=0; delay_us(1); IIC_SCL=1; delay_us(1); IIC_SCL=0; } /************************************************************************** Function: IIC don't reply Input : none Output : none گ˽٦ŜúIICһӦհ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void I2C_NAck(void) { IIC_SCL=0; SDA_OUT(); IIC_SDA=1; delay_us(1); IIC_SCL=1; delay_us(1); IIC_SCL=0; } bool I2C_WriteOneBit(uint8_t DevAddr, uint8_t RegAddr, uint8_t BitNum, uint8_t Data) { uint8_t Dat; Dat =I2C_ReadOneByte(DevAddr, RegAddr); Dat = (Data != 0) ? (Dat | (1 << BitNum)) : (Dat & ~(1 << BitNum)); I2C_WriteOneByte(DevAddr, RegAddr, Dat); return true; } bool I2C_WriteBits(uint8_t DevAddr, uint8_t RegAddr, uint8_t BitStart, uint8_t Length, uint8_t Data) { uint8_t Dat, Mask; Dat = I2C_ReadOneByte(DevAddr, RegAddr); Mask = (0xFF << (BitStart + 1)) | 0xFF >> ((8 - BitStart) + Length - 1); Data <<= (8 - Length); Data >>= (7 - BitStart); Dat &= Mask; Dat |= Data; I2C_WriteOneByte(DevAddr, RegAddr, Dat); return true; } /************************************************************************** Function: IIC sends a bit Input : none Output : none گ˽٦ŜúIICע̍һٶλ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ void I2C_WriteByte(uint8_t Data) { u8 t; SDA_OUT(); IIC_SCL=0;//-֍ʱדߪʼ˽ߝԫˤ for(t=0;t<8;t++) { IIC_SDA=(Data&0x80)>>7; Data<<=1; delay_us(1); IIC_SCL=1; delay_us(1); IIC_SCL=0; delay_us(1); } } u8 I2C_WriteOneByte(uint8_t DevAddr, uint8_t RegAddr, uint8_t Data) { I2C_Start(); I2C_WriteByte(DevAddr | I2C_Direction_Transmitter); I2C_WaiteForAck(); I2C_WriteByte(RegAddr); I2C_WaiteForAck(); I2C_WriteByte(Data); I2C_WaiteForAck(); I2C_Stop(); return 1; } bool I2C_WriteBuff(uint8_t DevAddr, uint8_t RegAddr, uint8_t Num, uint8_t *pBuff) { uint8_t i; if(0 == Num || NULL == pBuff) { return false; } I2C_Start(); I2C_WriteByte(DevAddr | I2C_Direction_Transmitter); I2C_WaiteForAck(); I2C_WriteByte(RegAddr); I2C_WaiteForAck(); for(i = 0; i < Num; i ++) { I2C_WriteByte(*(pBuff + i)); I2C_WaiteForAck(); } I2C_Stop(); return true; } /************************************************************************** Function: IIC reads a bit Input : none Output : none گ˽٦ŜúIICׁȡһٶλ ɫࠚӎ˽úϞ ׵ܘ ֵúϞ **************************************************************************/ uint8_t I2C_ReadByte(uint8_t Ack) { uint8_t i, RecDat = 0; SDA_IN(); for(i = 0; i < 8; i ++) { // I2C_SCL_Clr(); IIC_SCL=0; delay_us(1); // I2C_SCL_Set(); IIC_SCL=1; RecDat <<= 1; if(READ_SDA) RecDat |= 0x01; else RecDat &= ~0x01; delay_us(1); } if(I2C_ACK == Ack) I2C_Ack(); else I2C_NAck(); return RecDat; } uint8_t I2C_ReadOneByte(uint8_t DevAddr, uint8_t RegAddr) { uint8_t TempVal = 0; I2C_Start(); I2C_WriteByte(DevAddr | I2C_Direction_Transmitter); I2C_WaiteForAck(); I2C_WriteByte(RegAddr); I2C_WaiteForAck(); I2C_Start(); I2C_WriteByte(DevAddr | I2C_Direction_Receiver); I2C_WaiteForAck(); TempVal = I2C_ReadByte(I2C_NACK); I2C_Stop(); return TempVal; } bool I2C_ReadBuff(uint8_t DevAddr, uint8_t RegAddr, uint8_t Num, uint8_t *pBuff) { uint8_t i; if(0 == Num || NULL == pBuff) { return false; } I2C_Start(); I2C_WriteByte(DevAddr | I2C_Direction_Transmitter); I2C_WaiteForAck(); I2C_WriteByte(RegAddr); I2C_WaiteForAck(); I2C_Start(); I2C_WriteByte(DevAddr | I2C_Direction_Receiver); I2C_WaiteForAck(); for(i = 0; i < Num; i ++) { if((Num - 1) == i) { *(pBuff + i) = I2C_ReadByte(I2C_NACK); } else { *(pBuff + i) = I2C_ReadByte(I2C_ACK); } } I2C_Stop(); return true; }重写基于STM32F407的6针OLED 初始化

#include "stm32f10x.h" // Device header #include "Delay.h" #include "OLED.h" #include "Serial.h" #include "Key.h" #include "mpuiic.h" #include "mpu6050.h"//MPU6050驱动库 #include "inv_mpu.h"//MPU6050计步驱动库 #include "inv_mpu_dmp_motion_driver.h"//MPU6050计步,计时驱动库 #include <math.h> /*----------------各引脚请看Readme.txt------------------*/ u8 keynum,mode; int main(void) { Serial_Init(); MPU_IIC_Init(); MPU_Init(); OLED_Init(); Key_Init(); while(mpu_dmp_init()) //mpu6050_dmp初始化, { Delay_ms(100); //初始化需要时间 } while (1) { if(mode==0) { //初始界面时才显示 dmp_getwalk(); //运行取步函数,函数内有OLED_Show dmp_svm(); //运行跌倒函数,函数内有OLED_Show } Key_control(); //按钮控制mode OLED_Show(); //屏幕显示内容 Delay_ms(50); } }这段代码是mpu6050测摔倒和计步数的主函数,其他函数进行了分装,我想把下面测dht11温度湿度的传感器放在上面的主函数里令它在pb1的按键按下时切换屏幕显示温湿度,使他可以再0.96寸的oled屏幕上显示具体数据,要求能正常运行和显示,采用c语言标准库进行编写,要具体代码#include "stm32f10x.h" // Device header #include "Delay.h" #include "OLED.h" #include "DHT11.h" int main() { OLED_Init(); uint8_t buffer[5]; float h, t; OLED_ShowCC_F16x16(2, 1, 2); // 温 OLED_ShowCC_F16x16(2, 3, 4); // 度 OLED_ShowChar(2, 5, ':'); OLED_ShowCC_F16x16(3, 1, 3); // 湿 OLED_ShowCC_F16x16(3, 3, 4); // 度 OLED_ShowChar(3, 5, ':'); while(1) { if(DHT_ReadData(buffer) == 0) { h = buffer[0] + buffer[1] / 10.0; t = buffer[2] + buffer[3] / 10.0; OLED_ShowFNum(2, 6, t, 3, 1); OLED_ShowCC_F16x16(2, 11, 0); // ℃ OLED_ShowFNum(3, 6, h, 3, 1); OLED_ShowCC_F16x16(3, 11, 1); // % } else { OLED_Clear(); OLED_ShowString(2, 6, "ERROR"); } } }

/* * 项目名称:STM32 OLED显示系统(含字模与IIC测试) * 功能描述: * 1. 在128×64 OLED屏上显示学号、日期及汉字信息 * 2. 通过IIC总线驱动OLED和AT24C02 * 3. LED周期性闪烁指示系统运行状态 * 4. 支持BMP字模数据解析与显示 * * 硬件依赖: * - STM32F103系列MCU * - 128×64 OLED显示屏(4线IIC接口) * - AT24C02 EEPROM(IIC接口) * - LED指示灯(推挽输出模式) * * 核心知识点: * 1. OLED显示原理:点阵字模数据存储与扫描显示 * 2. IIC总线协议:时序控制与多设备共用总线 * 3. 字模生成:BMP图片转C语言数组的方法 * 4. 系统初始化:外设时钟配置与接口初始化 * 5. 状态指示:LED闪烁的软件定时器实现 * * 字模数据说明: * - picture_tab数组存储128×64像素的BMP字模 * - 每个字节对应8个像素(高位→左像素,低位→右像素) * - 0xFF表示像素全亮,0x00表示全灭,按行存储 */ #include "led.h" #include "delay.h" #include "key.h" #include "sys.h" #include "24cxx.h" #include "OLED4PINiic.h" /* ------------------- BMP字模数据区 ------------------- */ /* * 字模生成信息: * - 源文件路径:C:\Documents and Settings\Administrator\桌面\新建文件夹 (2)\logo.bmp字模 * - 分辨率:128×64像素(宽×高) * - 存储格式:按行扫描,每行16字节(128像素/8=16字节) * - 显示方向:从左到右,从上到下 */ const unsigned char picture_tab[] = { // 顶部边框像素数据(全亮边框) 0xFF,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x20,0x44,0x08,0x00,0x18,0x48,0x28,0xC8,0x08,0x28,0x48,0x18,0x00, 0x40,0x40,0xFC,0x40,0x40,0xFC,0x00,0x00,0xF8,0x00,0x00,0xFC,0x00,0x40,0x40,0xA0, 0x90,0x88,0x84,0x88,0x90,0x20,0x40,0x40,0x00,0x00,0x40,0x44,0xD8,0x20,0xF0,0xAC, 0xA8,0xE8,0xB8,0xA8,0xE0,0x00,0x00,0x00,0xC0,0x7C,0x54,0x54,0x54,0x54,0x54,0x54, 0x7C,0x40,0x40,0x00,0x00,0xF0,0x90,0x90,0x90,0xFC,0x90,0x90,0x90,0xF0,0x00,0x00, 0x00,0x80,0x88,0x88,0x88,0x88,0x88,0xE8,0xA8,0x98,0x8C,0x88,0x80,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x04,0x3E,0x01,0x10,0x11,0x09,0x05,0x3F,0x05,0x09,0x11,0x11,0x00, 0x08,0x18,0x0F,0x24,0x14,0x0F,0x00,0x00,0x0F,0x00,0x00,0x3F,0x00,0x20,0x22,0x2A, 0x32,0x22,0x3F,0x22,0x32,0x2A,0x22,0x20,0x00,0x00,0x20,0x10,0x0F,0x10,0x28,0x24, 0x23,0x20,0x2F,0x28,0x2A,0x2C,0x00,0x30,0x0F,0x04,0x3D,0x25,0x15,0x15,0x0D,0x15, 0x2D,0x24,0x24,0x00,0x00,0x07,0x04,0x04,0x04,0x1F,0x24,0x24,0x24,0x27,0x20,0x38, 0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x3F,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0xFF, 0xFF,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x81,0x41,0x21,0x21,0x61,0x01,0x01,0x21,0xE1,0xE1,0x01,0xE1,0xE1, 0x21,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x21,0xE1,0x21,0x21,0x21,0x61,0x01,0x01, 0x21,0x21,0xE1,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0xC1,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x21,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x0F,0x10,0x20,0x24,0x1C,0x04,0x00,0x20,0x3F,0x01,0x3E,0x01,0x3F, 0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x3F,0x22,0x22,0x27,0x30,0x00,0x00, 0x20,0x20,0x3F,0x20,0x20,0x00,0x00,0x1E,0x25,0x25,0x25,0x16,0x00,0x00,0x1E,0x21, 0x21,0x21,0x13,0x00,0x01,0x01,0x1F,0x21,0x21,0x00,0x00,0x00,0x21,0x3F,0x22,0x21, 0x01,0x00,0x00,0x1E,0x21,0x21,0x21,0x1E,0x00,0x21,0x3F,0x22,0x01,0x01,0x3E,0x20, 0x00,0x21,0x21,0x3F,0x20,0x20,0x00,0x00,0x1E,0x21,0x21,0x21,0x13,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0xF0,0x08,0x04,0x04,0x04,0x0C,0x00,0xF0,0x08,0x04,0x04,0x08,0xF0, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0x80,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x04,0xFC,0x04,0x00, 0x00,0x00,0x00,0x00,0x0C,0x04,0xFC,0x04,0x0C,0x00,0x04,0xFC,0x04,0x04,0x08,0xF0, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x81,0x82,0x84,0x84,0x84,0x82,0x80,0x81,0x82,0x84,0x84,0x82,0x81, 0x80,0x80,0x86,0x86,0x80,0x80,0x80,0x80,0x80,0x85,0x83,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x84,0x87,0x84,0x84, 0x84,0x86,0x80,0x80,0x80,0x84,0x87,0x84,0x80,0x80,0x84,0x87,0x84,0x84,0x82,0x81, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0xFF }; /* ------------------- EEPROM测试数据 ------------------- */ /* * 测试字符串:用于AT24C02写入验证 * - 字符串内容:"Explorer STM32F4 IIC TEST" * - 存储方式:通过IIC总线写入EEPROM指定地址 * - 长度定义:使用sizeof自动计算字节数 */ const u8 TEXT_Buffer[] = {"Explorer STM32F4 IIC TEST"}; #define SIZE sizeof(TEXT_Buffer) /* ------------------- 主函数实现 ------------------- */ int main(void) { const unsigned char *point; // 字模数据指针 u16 i = 0; // 循环计数器 /* 系统初始化流程 * 优先级:延时→中断→通信→外设→显示 */ delay_init(); // 初始化SysTick延时函数 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); // 配置中断优先级分组 uart_init(115200); // 初始化USART1(波特率115200) LED_Init(); // 初始化LED控制引脚 // KEY_Init(); // 按键初始化(代码中未调用,已注释) AT24CXX_Init(); // 初始化IIC总线(OLED与24C02共用) OLED_init(); // 初始化OLED显示屏 /* OLED静态信息显示 * 显示内容:学号、日期(固定位置显示) */ OLED_ShowString(0, 0, "23221093", 16); // 显示学号(16像素字体) OLED_ShowString(0, 40, "ATOM 2025/05/08", 12); // 显示日期(12像素字体) OLED_Refresh_Gram(); // 刷新显示缓冲区到OLED /* 字模数据指针初始化 */ point = &picture_tab[0]; // 指向字模数据首地址 /* 主循环逻辑 * 1. 动态显示汉字信息 * 2. 周期性刷新OLED屏幕 * 3. LED状态指示系统运行 */ while (1) { /* 汉字显示模块 * 自定义显示函数说明: * - OLED_SHOWliu:显示"刘"字 * - OLED_SHOWmu:显示"牧"字 * - OLED_SHOWwei:显示"伟"字 * 参数含义: * - x,y:显示起始坐标(像素) * - mode:0=反显,1=正显 * - size:字体大小(12/16/24) * - num:保留参数(未使用) */ OLED_SHOWliu(0, 24, 1, 16, 1); // 在(0,24)显示16像素"刘"字 OLED_SHOWmu(16, 24, 1, 16, 1); // 在(16,24)显示16像素"穆"字 OLED_SHOWwei(32, 24, 1, 16, 1); // 在(32,24)显示16像素"惟"字 OLED_Refresh_Gram(); // 更新OLED显示缓冲区 delay_ms(100); // 延时100ms(控制刷新频率) /* LED状态指示逻辑 * 实现原理:每300ms翻转一次LED状态 * 作用:直观提示系统程序正常运行 */ if (++i > 3) { LED1 = !LED1; // 翻转LED1电平状态 i = 0; // 计数器清零 } } } 修改以上代码,完成:使用定时器在OLED显示屏上实现秒倒计时器,倒计时器的初值设为1234,每秒数字减1,OLED屏在0列16行处显示,字号为16,oled屏只保留倒计时器,并为我提供timer.c的修改。timer.c如下: #include "timer.h" #include "led.h" ////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////// //通用定时器3中断初始化 //这里时钟选择为APB1的2倍,而APB1为36M //arr:自动重装值。 //psc:时钟预分频数 //这里使用的是定时器3! void TIM3_Int_Init(u16 arr,u16 psc) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能 //定时器TIM3初始化 TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值 TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值 TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式 TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据指定的参数初始化TIMx的时间基数单位 TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE ); //使能指定的TIM3中断,允许更新中断 //中断优先级NVIC设置 NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中断 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占优先级0级 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //从优先级3级 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能 NVIC_Init(&NVIC_InitStructure); //初始化NVIC寄存器 TIM_Cmd(TIM3, ENABLE); //使能TIMx } //定时器3中断服务程序 void TIM3_IRQHandler(void) //TIM3中断 { if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查TIM3更新中断发生与否 { TIM_ClearITPendingBit(TIM3, TIM_IT_Update ); //清除TIMx更新中断标志 LED1=!LED1; } } //TIM4 PWM部分初始化 //PWM输出初始化 //arr:自动重装值 //psc:时钟预分频数 void TIM4_PWM_Init(u32 arr,u32 psc) { //此部分需手动修改IO口设置 GPIO_InitTypeDef GPIO_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4,ENABLE); //TIM4时钟使能 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能PORTB时钟 // RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE); // //GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE);//GPIOB8复用位定时器4 // GPIO_PinAFConfig(GPIOB,GPIO_PinSource8,GPIO_AF_TIM4); //设置该引脚为复用输出功能,输出TIM3 CH2的PWM脉冲波形 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; //GPIOB8 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB,&GPIO_InitStructure); //初始化PB8 TIM_TimeBaseStructure.TIM_Prescaler=psc; //定时器分频 TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up; //向上计数模式 TIM_TimeBaseStructure.TIM_Period=arr; //自动重装载值 TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1; TIM_TimeBaseInit(TIM4,&TIM_TimeBaseStructure); //初始化TIM4 Channe3 PWM模式 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //选择定时器模式:TIM脉冲宽度调制模式2 TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能 TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low; //输出极性:TIM输出比较极性高 TIM_OCInitStructure.TIM_Pulse=9000; TIM_OC3Init(TIM4, &TIM_OCInitStructure); //根据T指定的参数初始化外设TIM4 OC3 TIM_OC3PreloadConfig(TIM4, TIM_OCPreload_Enable); //使能TIM4在CCR2上的预装载寄存器 TIM_ARRPreloadConfig(TIM4,ENABLE); TIM_Cmd(TIM4, ENABLE); //使能TIM4 }

#include "led.h" #include "delay.h" #include "key.h" #include "sys.h" //#include "oled.h" #include "24cxx.h" #include "OLED4PINiic.h" const unsigned char picture_tab[]={ /*------------------------------------------------------------------------------ ; 源文件 / 文字 : C:\Documents and Settings\Administrator\桌面\新建文件夹 (2)\logo.bmp字模 ; 宽×高(像素): 128×64 ------------------------------------------------------------------------------*/ //0x80,0x40,0x10,//宽的像素数,高的像素数,宽的字节数,参数设置可选 0xFF,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x20,0x44,0x08,0x00,0x18,0x48,0x28,0xC8,0x08,0x28,0x48,0x18,0x00, 0x40,0x40,0xFC,0x40,0x40,0xFC,0x00,0x00,0xF8,0x00,0x00,0xFC,0x00,0x40,0x40,0xA0, 0x90,0x88,0x84,0x88,0x90,0x20,0x40,0x40,0x00,0x00,0x40,0x44,0xD8,0x20,0xF0,0xAC, 0xA8,0xE8,0xB8,0xA8,0xE0,0x00,0x00,0x00,0xC0,0x7C,0x54,0x54,0x54,0x54,0x54,0x54, 0x7C,0x40,0x40,0x00,0x00,0xF0,0x90,0x90,0x90,0xFC,0x90,0x90,0x90,0xF0,0x00,0x00, 0x00,0x80,0x88,0x88,0x88,0x88,0x88,0xE8,0xA8,0x98,0x8C,0x88,0x80,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x04,0x3E,0x01,0x10,0x11,0x09,0x05,0x3F,0x05,0x09,0x11,0x11,0x00, 0x08,0x18,0x0F,0x24,0x14,0x0F,0x00,0x00,0x0F,0x00,0x00,0x3F,0x00,0x20,0x22,0x2A, 0x32,0x22,0x3F,0x22,0x32,0x2A,0x22,0x20,0x00,0x00,0x20,0x10,0x0F,0x10,0x28,0x24, 0x23,0x20,0x2F,0x28,0x2A,0x2C,0x00,0x30,0x0F,0x04,0x3D,0x25,0x15,0x15,0x0D,0x15, 0x2D,0x24,0x24,0x00,0x00,0x07,0x04,0x04,0x04,0x1F,0x24,0x24,0x24,0x27,0x20,0x38, 0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x3F,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0xFF, 0xFF,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x81,0x41,0x21,0x21,0x61,0x01,0x01,0x21,0xE1,0xE1,0x01,0xE1,0xE1, 0x21,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x21,0xE1,0x21,0x21,0x21,0x61,0x01,0x01, 0x21,0x21,0xE1,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0xC1,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x21,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01, 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x0F,0x10,0x20,0x24,0x1C,0x04,0x00,0x20,0x3F,0x01,0x3E,0x01,0x3F, 0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x3F,0x22,0x22,0x27,0x30,0x00,0x00, 0x20,0x20,0x3F,0x20,0x20,0x00,0x00,0x1E,0x25,0x25,0x25,0x16,0x00,0x00,0x1E,0x21, 0x21,0x21,0x13,0x00,0x01,0x01,0x1F,0x21,0x21,0x00,0x00,0x00,0x21,0x3F,0x22,0x21, 0x01,0x00,0x00,0x1E,0x21,0x21,0x21,0x1E,0x00,0x21,0x3F,0x22,0x01,0x01,0x3E,0x20, 0x00,0x21,0x21,0x3F,0x20,0x20,0x00,0x00,0x1E,0x21,0x21,0x21,0x13,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0xF0,0x08,0x04,0x04,0x04,0x0C,0x00,0xF0,0x08,0x04,0x04,0x08,0xF0, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0x80,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x04,0xFC,0x04,0x00, 0x00,0x00,0x00,0x00,0x0C,0x04,0xFC,0x04,0x0C,0x00,0x04,0xFC,0x04,0x04,0x08,0xF0, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF, 0xFF,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x81,0x82,0x84,0x84,0x84,0x82,0x80,0x81,0x82,0x84,0x84,0x82,0x81, 0x80,0x80,0x86,0x86,0x80,0x80,0x80,0x80,0x80,0x85,0x83,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x84,0x87,0x84,0x84, 0x84,0x86,0x80,0x80,0x80,0x84,0x87,0x84,0x80,0x80,0x84,0x87,0x84,0x84,0x82,0x81, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80, 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0xFF }; //要写入到24c02的字符串数组 const u8 TEXT_Buffer[]={"Explorer STM32F4 IIC TEST"}; #define SIZE sizeof(TEXT_Buffer) int main(void) { const unsigned char *point; u16 i=0; delay_init(); //延时函数初始化 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级 // uart_init(115200); //串口初始化为115200 LED_Init(); //初始化与LED连接的硬件接口 // KEY_Init(); //按键初始化 AT24CXX_Init(); //IIC初始化 OLED_init(); OLED_ShowString(0,0,"@BJTU",16); OLED_ShowString(0,24, "0.96' OLED TEST",16); OLED_ShowString(0,40,"ATOM 2020/11/24",12); OLED_ShowString(0,52,"ASCII:",12); OLED_ShowString(64,52,"CODE:",12); OLED_Refresh_Gram();//更新显示到OLED point= &picture_tab[0]; while(1) { OLED_DrawPoint(120,63,1); OLED_DrawPoint(126,63,1); OLED_DrawPoint(127,63,1); OLED_DrawLine(2,2,120,63); OLED_Refresh_Gram(); //更新显示到OLED delay_ms(100); if(++i > 3) { LED1=!LED1;//提示系统正在运行 i=0; } } } 怎么用这个程序在OLED显示屏上显示汉字”何丽思“

最新推荐

recommend-type

【电力系统优化调度】含可再生能源的机组组合优化模型设计:构建经济稳定运行系统(可实现,有问题可联系博主)

内容概要:本文详细探讨了机组组合优化模型的构建,旨在通过合理安排各类发电机组的启停计划和优化出力分配,实现电力系统在经济性和稳定性上的最佳平衡。文章首先介绍了电力系统的四大主要组件——传统火电机组、风电机组、光伏机组和储能系统的参数及运行特性。接着,围绕最小化系统总运行成本这一目标,设计了优化目标函数,并明确了包括功率平衡约束、机组出力上下限约束、风光发电功率约束、弃风弃光约束、爬坡速率约束、储能系统荷电状态约束、充放电功率约束和充放电互斥约束在内的多项约束条件。最后,文章列出了求解机组组合优化模型所需的关键变量,如传统机组的开停状态、机组出力、启停成本、风电光伏实际出力、弃风弃光比例及储能系统的充放电功率和荷电状态,以实现系统的经济调度和可再生能源的最大化利用。 适合人群:从事电力系统研究、规划和调度工作的工程师和技术人员,以及对电力系统优化感兴趣的科研人员。 使用场景及目标:①帮助电力系统工程师理解不同类型发电机组的特点及其对系统稳定性、经济性和环保性的影响;②为制定合理的电力系统调度策略提供理论依据和技术支持;③促进可再生能源的有效整合,提高电力系统的灵活性和可靠性。 其他说明:本文提供的模型和方法不仅适用于当前的电力系统,也可为未来含高比例可再生能源接入的电力系统提供参考。文中涉及的具体数学公式和参数设定为实际应用提供了详细的指导,有助于提升电力系统的运行效率和经济效益。
recommend-type

项目管理手册释义.ppt

项目管理手册释义.ppt
recommend-type

电子商务的网上支付.ppt

电子商务的网上支付.ppt
recommend-type

综合布线系统设计.ppt

综合布线系统设计.ppt
recommend-type

Evc Sql CE 程序开发实践与样例代码分享

在详细解释标题、描述和标签中提及的知识点之前,需要指出“压缩包子文件的文件名称列表”中的“8”可能是不完整的上下文信息。由于缺乏具体的文件列表内容,我们将主要集中在如何理解“Evc Sql CE 程序样例代码”这一主题。 标题“Evc Sql CE 程序样例代码”直接指向一个程序开发样例代码,其中“Evc”可能是某种环境或工具的缩写,但由于没有更多的上下文信息,很难精确地解释这个缩写指的是什么。不过,“Sql CE”则明确地指向了“SQL Server Compact Edition”,它是微软推出的一个轻量级数据库引擎,专为嵌入式设备和小型应用程序设计。 ### SQL Server Compact Edition (SQL CE) SQL Server Compact Edition(简称SQL CE)是微软公司提供的一个嵌入式数据库解决方案,它支持多种平台和编程语言。SQL CE适合用于资源受限的环境,如小型应用程序、移动设备以及不需要完整数据库服务器功能的场合。 SQL CE具备如下特点: - **轻量级**: 轻便易用,对系统资源占用较小。 - **易于部署**: 可以轻松地将数据库文件嵌入到应用程序中,无需单独安装。 - **支持多平台**: 能够在多种操作系统上运行,包括Windows、Windows CE和Windows Mobile等。 - **兼容性**: 支持标准的SQL语法,并且在一定程度上与SQL Server数据库系统兼容。 - **编程接口**: 提供了丰富的API供开发者进行数据库操作,支持.NET Framework和本机代码。 ### 样例代码的知识点 “Evc Sql CE 程序样例代码”这部分信息表明,存在一些示例代码,这些代码可以指导开发者如何使用SQL CE进行数据库操作。样例代码一般会涵盖以下几个方面: 1. **数据库连接**: 如何创建和管理到SQL CE数据库的连接。 2. **数据操作**: 包括数据的增删改查(CRUD)操作,这些是数据库操作中最基本的元素。 3. **事务处理**: 如何在SQL CE中使用事务,保证数据的一致性和完整性。 4. **数据表操作**: 如何创建、删除数据表,以及修改表结构。 5. **数据查询**: 利用SQL语句查询数据,包括使用 SELECT、JOIN等语句。 6. **数据同步**: 如果涉及到移动应用场景,可能需要了解如何与远程服务器进行数据同步。 7. **异常处理**: 在数据库操作中如何处理可能发生的错误和异常。 ### 标签中的知识点 标签“Evc Sql CE 程序样例代码”与标题内容基本一致,强调了这部分内容是关于使用SQL CE的示例代码。标签通常用于标记和分类信息,方便在搜索引擎或者数据库中检索和识别特定内容。在实际应用中,开发者可以根据这样的标签快速找到相关的样例代码,以便于学习和参考。 ### 总结 根据标题、描述和标签,我们可以确定这篇内容是关于SQL Server Compact Edition的程序样例代码。由于缺乏具体的代码文件名列表,无法详细分析每个文件的内容。不过,上述内容已经概述了SQL CE的关键特性,以及开发者在参考样例代码时可能关注的知识点。 对于希望利用SQL CE进行数据库开发的程序员来说,样例代码是一个宝贵的资源,可以帮助他们快速理解和掌握如何在实际应用中运用该数据库技术。同时,理解SQL CE的特性、优势以及编程接口,将有助于开发者设计出更加高效、稳定的嵌入式数据库解决方案。
recommend-type

【浪潮FS6700交换机配置实战】:生产环境快速部署策略与技巧

# 1. 浪潮FS6700交换机概述 浪潮FS6700交换机是一款高端、具备高密度端口的全千兆交换机。它采用模块化设计,支持万兆上行,可提供灵活的端口组合和高带宽解决方案,满足了企业网络对于高性能、高可靠性的需求。浪潮FS6700交换机通常部署在企业网络的核心层或汇聚层,不仅提供强劲的网络数据交换能力,而且支持丰富的路由协议和安全特性,为中大型网络构建提供了强有力的保障。 接下来,我们将深
recommend-type

YOLO11训练批次参考

<think>我们正在讨论YOLOv11的训练批次设置。首先,需要明确的是,截至2023年10月,官方并没有发布YOLOv11版本。YOLO系列的最新版本是YOLOv8(由Ultralytics发布)。因此,这里可能指的是YOLO的某个变体或非官方版本。不过,我们可以基于YOLO系列的一般训练实践来讨论训练批次(batch size)的设置。 训练批次(batch size)是指在训练神经网络时,每次迭代中用于计算梯度并更新权重的样本数量。设置合适的batch size对训练效果和速度有重要影响。 ### 影响batch size选择的因素: 1. **硬件限制**:显存大小是主要限制因素
recommend-type

数据库考试复习必备五套习题精讲

根据给定的文件信息,本文将详细解释数据库习题相关知识点。首先,从标题中我们可以得知,该文件为数据库习题集,包含五套习题卷,非常适合用来准备考试。由于文件描述中提到考完试后才打算分享,说明这些习题具有一定的质量和难度,可以作为考试前的必备材料。 首先,我们来解释“数据库”这一核心概念。数据库是存储、管理、处理和检索信息的系统,它能够帮助我们有效地存储大量的数据,并在需要的时候快速访问。数据库管理系统(DBMS)是负责数据库创建、维护和操作的软件,常见的数据库管理系统包括MySQL、Oracle、Microsoft SQL Server、PostgreSQL和SQLite等。 数据库习题通常包括以下知识点: 1. 数据库设计:设计数据库时需要考虑实体-关系模型(ER模型)、规范化理论以及如何设计表结构。重点包括识别实体、确定实体属性、建立实体之间的关系以及表之间的关联。规范化是指将数据库表结构进行合理化分解,以减少数据冗余和提高数据一致性。 2. SQL语言:结构化查询语言(SQL)是用于管理数据库的标准计算机语言,它包括数据查询、数据操纵、数据定义和数据控制四个方面的功能。对于数据库习题来说,重点会涉及到以下SQL语句: - SELECT:用于从数据库中查询数据。 - INSERT、UPDATE、DELETE:用于向数据库中插入、更新或删除数据。 - CREATE TABLE、ALTER TABLE、DROP TABLE:用于创建、修改或删除表结构。 - JOIN:用于连接两个或多个表来查询跨越表的数据。 - GROUP BY 和 HAVING:用于对数据进行分组统计和筛选。 -事务处理:包括事务的ACID属性(原子性、一致性、隔离性、持久性)等。 3. 数据库操作:涉及实际操作数据库的过程,包括数据导入导出、备份与恢复、索引创建与优化等。这些内容能够帮助理解如何高效地管理数据。 4. 数据库安全:保障数据库不受未授权访问和破坏的机制,例如用户权限管理、视图、存储过程等安全措施。 5. 数据库优化:如何提升数据库的性能,包括查询优化、数据库配置优化、索引策略、系统资源监控等。 6. 数据库应用开发:如何利用数据库在应用程序中实现数据的持久化存储,如数据库连接、事务管理、数据访问对象(DAO)设计模式等。 7. 高级主题:涉及到复杂查询、数据库触发器、存储过程的编写和优化,以及可能包含的特定数据库系统的特定特性(如Oracle的PL/SQL编程等)。 由于文件名称列表只提供“数据库习题”这一个信息点,我们无法得知具体的习题内容和难度,但是可以肯定的是,这份习题集应该覆盖了上述所提到的知识点。对于考生来说,这些习题将帮助他们巩固理论知识,并且提高解决实际问题的能力,是考试前准备的有力工具。 在准备数据库相关的考试时,建议先从基础概念开始复习,然后逐步过渡到SQL语法和数据库设计的实践操作。在习题练习中,注意不要仅限于死记硬背,更重要的是理解每一个操作背后的逻辑和原理。如果可能的话,实际操作一个数据库,将理论知识应用到实践中去,这会帮助你更加深刻地理解数据库的工作机制。最后,反复练习模拟题,可以帮助你熟悉考试的题型和难度,提高考试时的应试技巧。
recommend-type

【浪潮FS6700交换机故障诊断与排除】:掌握这些方法,让你的网络稳定如初

# 1. 浪潮FS6700交换机故障诊断基础知识 在本章中,我们将探讨浪潮FS6700交换机故障诊断的基础知识,为后续章节中更深层次的理论和实践内容打下坚实的基础。我们将从以下几个方面入手,逐步深化对故障诊断的认识: ## 1.1 交换机故障诊断的重要性 交换机作为网络的核心设备,其稳定性和性能直接影响着整个网络的运行效率。因此,准确且快速地诊断和解决交换机故障是网
recommend-type

JVM内存整体结构图

<think>我们正在处理一个关于JVM内存结构的问题,用户要求获取详细图解。由于我们无法直接发送图片,但可以通过文字描述和引用相关资源来满足需求。 根据引用内容,我们可以总结JVM内存结构的主要部分: 1. 线程栈(Thread Stacks):每个线程创建时分配,存储局部变量和方法调用栈。 2. 堆(Heap):存储所有对象、实例变量和数组,被所有线程共享。堆又分为年轻代(Young Generation)和老年代(Old Generation)。 3. 非堆内存(Non-Heap Memory):包括方法区(Method Area)和运行时常量池(Runtime Constant