STM32F1通过IIC驱动0.96寸OLED说明

STM32F103ZET6 IIC驱动0.96寸OLED说明

1. 说明
之前购买了0.96寸OLED,当时使用淘宝客服提供的驱动程序,下载到单片机后OLED无反应,换个OLED正常工作,当时就很疑惑。最近时间比较空闲,因此把OLED拿过来重新设计驱动,设计了一个基于STM32F103ZET6的工程代码。网络上基于STM32F103ZET6的驱动代码免费的不是很容易找,因此本人提供一份代码出来供大家学习参考,希望大家不要像我一样买到手后不知道是程序问题还是OLED问题困扰,以至于无法使用,下面简单说明一下使用。

2. 原理介绍
关于原理的介绍网络上有很多,商家给的资料中关于原理讲解的也很详细,这里不在进行叙述,但是还是建议各位搞明白原理之后再去研究代码。
3. 注意事项
图片下载之后再计算机画图中打开,然后重新调整大小(调整后大小在128*64范围之内即可);然后点击另存为BMP格式,保存类型选择单色位图即可(若不是单色位图底色白色部分会显示)。 取模软件和使用说明请参考文章末尾网盘链接。

4.结束语
工程代码及取模软件链接如下,
链接:链接:https://pan.baidu.com/s/1lXwZvaR8pXJRs9uGvePv5g ,提取码:fenl;如果有问题加企鹅583671536 ;程序下载运行之后如下图所示。接线说明:
VCC —— 3.3V
GND —— GND
SCL —— PB6
SDA —— PB7

  • 21
    点赞
  • 72
    收藏
    觉得还不错? 一键收藏
  • 19
    评论
以下是使用STM32F103C8T6驱动0.96寸4针IIC OLED屏幕的完整代码: ```c #include "stm32f10x.h" #include "stm32f10x_i2c.h" #include "stm32f10x_gpio.h" #include "stm32f10x_rcc.h" #include "delay.h" #define OLED_ADDRESS 0x78 // OLED屏幕地址 // 定义IIC相关引脚 #define OLED_SCL_PIN GPIO_Pin_6 #define OLED_SDA_PIN GPIO_Pin_7 #define OLED_SCL_PORT GPIOB #define OLED_SDA_PORT GPIOB // 定义OLED指令宏 #define OLED_CMD_MODE 0x00 // 指令模式 #define OLED_DATA_MODE 0x40 // 数据模式 #define OLED_SET_CONTRAST_CONTROL 0x81 // 设置对比度控制 #define OLED_DISPLAY_ALL_ON_RESUME 0xA4 // 全局显示开启; 常规显示模式 #define OLED_DISPLAY_ALL_ON 0xA5 // 全局显示开启; 全部显示模式 #define OLED_NORMAL_DISPLAY 0xA6 // 常规显示模式; 反相显示 #define OLED_INVERSE_DISPLAY 0xA7 // 反相显示模式 #define OLED_DISPLAY_OFF 0xAE // OLED关闭显示 #define OLED_DISPLAY_ON 0xAF // OLED开启显示 #define OLED_SET_DISPLAY_OFFSET 0xD3 // 设置显示偏移 #define OLED_SET_COM_PINS 0xDA // 设置COM硬件引脚配置 #define OLED_SET_VCOM_DETECT 0xDB // 设置VCOMH电压倍增比 #define OLED_SET_DISPLAY_CLOCK_DIV_RATIO 0xD5 // 设置显示时钟分频比; 频率 #define OLED_SET_PRECHARGE_PERIOD 0xD9 // 设置预充电周期 #define OLED_SET_MULTIPLEX_RATIO 0xA8 // 设置驱动路数 #define OLED_SET_LOW_COLUMN 0x00 // 设置列低地址 #define OLED_SET_HIGH_COLUMN 0x10 // 设置列高地址 #define OLED_SET_START_LINE 0x40 // 设置起始行 #define OLED_MEMORY_MODE 0x20 // 内存地址模式 #define OLED_COLUMN_ADDR 0x21 // 列地址 #define OLED_PAGE_ADDR 0x22 // 页面地址 #define OLED_COM_SCAN_INC 0xC0 // 从COM0到COM[N-1];输出扫描 #define OLED_COM_SCAN_DEC 0xC8 // 从COM[N-1]到COM0;输出扫描 #define OLED_SEG_REMAP 0xA0 // 列地址0映射到SEG0 #define OLED_CHARGE_PUMP 0x8D // 电荷泵设置 // IIC初始化函数 void IIC_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; I2C_InitTypeDef I2C_InitStructure; // 开启GPIOB时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); // 配置GPIOB6和GPIOB7为开漏输出, IIC总线需要 GPIO_InitStructure.GPIO_Pin = OLED_SCL_PIN | OLED_SDA_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD; // 开漏输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(OLED_SCL_PORT, &GPIO_InitStructure); // 开启I2C1时钟 RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE); // 配置I2C1参数 I2C_InitStructure.I2C_Mode = I2C_Mode_I2C; // I2C模式 I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2; // 占空比 I2C_InitStructure.I2C_OwnAddress1 = 0x00; // 本机地址 I2C_InitStructure.I2C_Ack = I2C_Ack_Enable; // 允许应答 I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; // 7位地址 I2C_InitStructure.I2C_ClockSpeed = 400000; // 时钟速度 // 初始化I2C1 I2C_Cmd(I2C1, ENABLE); // 使能I2C1 I2C_Init(I2C1, &I2C_InitStructure); // 初始化I2C1 } // IIC发送数据函数 void IIC_SendByte(uint8_t byte) { uint8_t i; // 开始发送 I2C_GenerateSTART(I2C1, ENABLE); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); // 等待事件发生 // 发送设备地址 I2C_Send7bitAddress(I2C1, OLED_ADDRESS, I2C_Direction_Transmitter); // 读写方向为写 while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); // 等待事件发生 // 发送数据 I2C_SendData(I2C1, byte); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); // 等待事件发生 // 停止发送 I2C_GenerateSTOP(I2C1, ENABLE); for(i=0; i<50; i++); // 延时 } // OLED写入命令函数 void OLED_WriteCmd(uint8_t cmd) { IIC_SendByte(OLED_CMD_MODE); // 指令模式 IIC_SendByte(cmd); // 发送指令 } // OLED写入数据函数 void OLED_WriteData(uint8_t data) { IIC_SendByte(OLED_DATA_MODE); // 数据模式 IIC_SendByte(data); // 发送数据 } // OLED初始化函数 void OLED_Init(void) { OLED_WriteCmd(OLED_DISPLAY_OFF); // 关闭OLED显示 OLED_WriteCmd(OLED_SET_DISPLAY_CLOCK_DIV_RATIO); // 设置显示时钟分频比; 频率 OLED_WriteCmd(0x80); // 0x80是分频比值 OLED_WriteCmd(OLED_SET_MULTIPLEX_RATIO); // 设置驱动路数 OLED_WriteCmd(0x1F); // 驱动路数(1-64) OLED_WriteCmd(OLED_SET_DISPLAY_OFFSET); // 设置显示偏移 OLED_WriteCmd(0x00); // 偏移量为0 OLED_WriteCmd(OLED_SET_START_LINE | 0x00); // 设置起始行 OLED_WriteCmd(OLED_CHARGE_PUMP); // 电荷泵设置 OLED_WriteCmd(0x14); // 0x14是启用内部VCC OLED_WriteCmd(OLED_MEMORY_MODE); // 内存地址模式 OLED_WriteCmd(0x00); // 水平地址模式 OLED_WriteCmd(OLED_SEG_REMAP | 0x01); // 列地址0映射到SEG0 OLED_WriteCmd(OLED_COM_SCAN_DEC); // 从COM[N-1]到COM0;输出扫描 OLED_WriteCmd(OLED_SET_COM_PINS); // 设置COM硬件引脚配置 OLED_WriteCmd(0x12); // 0x12是设置COM硬件引脚配置 OLED_WriteCmd(OLED_SET_CONTRAST_CONTROL); // 设置对比度控制 OLED_WriteCmd(0xCF); // 0xCF是对比度值 OLED_WriteCmd(OLED_SET_PRECHARGE_PERIOD); // 设置预充电周期 OLED_WriteCmd(0xF1); // 预充电周期(1-15) OLED_WriteCmd(OLED_SET_VCOM_DETECT); // 设置VCOMH电压倍增比 OLED_WriteCmd(0x40); // 0x40是VCOMH电压倍增比 OLED_WriteCmd(OLED_DISPLAY_ALL_ON_RESUME); // 全局显示开启; 常规显示模式 OLED_WriteCmd(OLED_NORMAL_DISPLAY); // 常规显示模式; 反相显示 OLED_WriteCmd(OLED_DISPLAY_ON); // OLED开启显示 } // OLED清屏函数 void OLED_Clear(void) { uint8_t i, j; for(i=0; i<8; i++) { OLED_WriteCmd(OLED_SET_LOW_COLUMN); // 设置列低地址 OLED_WriteCmd(OLED_SET_HIGH_COLUMN); // 设置列高地址 OLED_WriteCmd(OLED_PAGE_ADDR + i); // 设置页面地址 for(j=0; j<128; j++) { OLED_WriteData(0x00); // 清屏,发送0x00 } } } // OLED显示字符串函数 void OLED_ShowString(uint8_t x, uint8_t y, const uint8_t *str) { uint8_t c = 0, i = 0, j = 0; while (str[c] != '\0') { i = 0; j = 0; c = str[c] - 32; if(x > 120) { x = 0; y++; } OLED_WriteCmd(OLED_SET_LOW_COLUMN | (x & 0x0f)); // 设置列低地址 OLED_WriteCmd(OLED_SET_HIGH_COLUMN | ((x >> 4) & 0x0f)); // 设置列高地址 OLED_WriteCmd(OLED_PAGE_ADDR | y); // 设置页面地址 for(i=0; i<6; i++) { OLED_WriteData(Font_6x8[c][i]); } x += 6; } } int main(void) { SystemInit(); Delay_Init(); IIC_Init(); // IIC初始化 OLED_Init(); // OLED初始化 OLED_Clear(); // OLED清屏 while(1) { OLED_ShowString(0, 0, "Hello, World!"); // OLED显示字符串 Delay_Ms(1000); // 延时1秒 OLED_Clear(); // OLED清屏 Delay_Ms(1000); // 延时1秒 } } ``` 其中,`Font_6x8`是一个6x8的字符点阵数组,用于在OLED上显示字符串。你可以在自己的工程中自行定义和修改。
评论 19
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值