我用的主控是STM32F103ZET6
程序里面有两种发法:一个是并行发送数据,另一个是串行发送数据
完整工程链接:12864完整驱动工程
12864学习资料和取模软件链接:12864学习资料和取模软件
程序运行效果图:
1.并行发送数据效果图
2.串行发送数据效果图
程序如下:
#include "sys.h"
#include "delay.h"
#include "led12864.h"
/*2020.11.20*/
unsigned char defchar[] = {
0x00,0x40,0x78,0x40,0x48,0x44,0x57,
0xFE,0x50,0x80,0x60,0xA0,0x51,0x20,
0x4A,0x28,0x4B,0xFC,0x48,0x20,0x69,
0x28,0x51,0x24,0x42,0x26,0x44,0x22,
0x40,0xA0,0x40,0x40}; //自定义字符数组,这里是陈
int main(void)
{
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //系统时钟设置PLL作为系统时钟
delay_init();
lcd12864_Pin_Init(); //引脚初始化共用
/*-----------并行发送数据------------------*/
// LED_Init(); //led12864并行初始化
// delay_ms(20);
//任意位置显示字符串(写DDRAM中的数据)
// display_DDRAM(0,0,"人之出,性本善。苟不教,性乃迁。性相近,习相远,教之道,贵以专");
// display_CGRAM(1,2,0x20,defchar); //任意位置显示自定义字符串(写CGRAM中的数据)
/*-----------串行发送数据------------------*/
Lcd_Init(); //led12864串行初始化
delay_ms(10);
LCD_Display_Words(0,0,"怕什么天道轮回");
/*-------------------------------------*/
while(1);
}
#include "led12864.h"
#include "delay.h"
/***************************************************************************************
* 2020.11.20
* LED12864模块
* 管脚号 管脚名称 LEVER 管脚功能描述
* 1 VSS 0 电源地
* 2 VDD +5.0V 电源电压
* 3 V0 - 液晶显示器驱动电压
* 4 D/I(RS) H/L D/I=“H” 表示DB7∽DB0为显示数据
* D/I=“L” 表示DB7∽DB0为显示指令数据
* 5 R/W(SID) H/L R/W=“H”,E=“H”数据被读到DB7∽DB0
* R/W=“L”,E=“H→L”数据被写到IR或DR
* 6 E(CLK) H/L R/W=“L”,E信号下降沿锁存DB7∽DB0
* R/W=“H”,E=“H”DDRAM数据读到DB7∽DB0
* 7 DB0 H/L 数据线
* 8 DB1 H/L 数据线
* 9 DB2 H/L 数据线
* 10 DB3 H/L 数据线
* 11 DB4 H/L 数据线
* 12 DB5 H/L 数据线
* 13 DB6 H/L 数据线
* 14 DB7 H/L 数据线
* 15 PSB(CS1) H/L 并/串行接口选择:H-并行 L-串行
* 16 NC(CS2) 空脚
* 17 RST H/L 复位信号,低电平复位
* 18 VOUT -10V LCD驱动负电压
* 19 LED+ + LED背光板电源
* 20 LED- - LED背光板电源
*********************************************************************************************/
/**************************************************
* 函数功能 : 引脚初始化
**************************************************/
void lcd12864_Pin_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOD, ENABLE); //使能GPIOD/GPIOA端口时钟
//数据/命令选择
//读/写选择rw
//使能信号en
//复位rst
GPIO_InitStructure.GPIO_Pin = RS|RW|E|RST;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_InitStructure.GPIO_Speed= GPIO_Speed_50MHz; //最高输出速率50MHz
GPIO_Init(GPIOA, &GPIO_InitStructure); //IO口初始化
//数据引脚
GPIO_InitStructure.GPIO_Pin = LCD_DATA;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_InitStructure.GPIO_Speed= GPIO_Speed_50MHz; //最高输出速率50MHz
GPIO_Init(GPIOD, &GPIO_InitStructure); //IO口初始化
}
/*************************************************
* 函数功能 : LED12864数据口引脚配置为输出模式
**************************************************/
void LED12864_DB_outputMode(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
//数据引脚
GPIO_InitStructure.GPIO_Pin = LCD_DATA;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_InitStructure.GPIO_Speed= GPIO_Speed_50MHz; //最高输出速率50MHz
GPIO_Init(GPIOD, &GPIO_InitStructure); //IO口初始化
}
/*************************************************
* 函数功能 : LED12864数据口引脚配置为上拉输入模式
**************************************************/
void LED12864_DB_inputMode(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
//数据引脚
GPIO_InitStructure.GPIO_Pin = LCD_DATA;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //推挽输出
GPIO_Init(GPIOD, &GPIO_InitStructure); //IO口初始化
}
/*************************************************
* 函数功能 : 写数据/命令
**************************************************/
void write_LCD12864_DB(uint8_t data)
{
GPIO_Write(GPIOD,data);
}
//
/*************************************************
* 函数功能 : 读DB0-7的数据
**************************************************/
uint8_t read_LCD12864_DB(void)
{
uint8_t value;
value=GPIO_ReadInputData(GPIOD);
return value;
}
/*************************************************
* 函数功能 : DB7引脚 忙信号检测
**************************************************/
void LCD_ReadyWait(void)
{
uint8_t state;
delay_ms(10);
GPIO_WriteBit(GPIOA,RS,Bit_RESET); //写指令 RS=0
GPIO_WriteBit(GPIOA,RW,Bit_SET); //写操作 R/W=1
do
{
GPIO_WriteBit(GPIOA,E,Bit_SET); //使能 E=1
LED12864_DB_inputMode(); //配置数据口为上拉输入模式
state =read_LCD12864_DB(); //查忙标志位,等待标志位为0,表示写入完毕;为1,表示忙状态
LED12864_DB_outputMode(); //配置数据口为输出模式
GPIO_WriteBit(GPIOA,E,Bit_RESET); //不使能 E=0
} while( state & 0x80);
}
/*************************************************
* 函数功能 : 向LCD12864写入命令
* 函数参数 :com写入的命令
**************************************************/
void LCD_Write_com(uchar com)
{
delay_ms(5);
LCD_ReadyWait();
delay_ms(10);
GPIO_WriteBit(GPIOA,RS,Bit_RESET); //写指令 RS=0
GPIO_WriteBit(GPIOA,RW,Bit_RESET); //写操作 R/W=0
GPIO_WriteBit(GPIOA,E,Bit_SET); //使能 E=高电平
write_LCD12864_DB(com); //写命令
GPIO_WriteBit(GPIOA,E,Bit_RESET); //不使能
}
/*************************************************
* 函数功能 : 向LCD12864写入数据
* 函数参数 :data写入的数据
**************************************************/
void LCD_Write_data(uchar data)
{
delay_ms(10);
GPIO_WriteBit(GPIOA,RS,Bit_SET); //写数据 RS=1
GPIO_WriteBit(GPIOA,RW,Bit_RESET); //写操作 R/W=0
GPIO_WriteBit(GPIOA,E,Bit_SET); //使能 E=高电平
write_LCD12864_DB(data); //写数据
GPIO_WriteBit(GPIOA,E,Bit_RESET); //不使能
}
/*************************************************
* 函数功能 : 清屏
*************************************************/
void LCD_clear(void)
{
LCD_Write_com(0x01); //写清屏指令0x01
delay_ms(10);
}
/*************************************************
* 函数功能 : 任意位置显示字符串(写DDRAM中的数据)
* 函数参数 :x第几列,y第几行,s输入的数据
*************************************************/
void display_DDRAM(uchar x,uchar y,uchar *s)
{
switch(y)
{
case 0:
LCD_Write_com(0x80+x); //第一行
break;
case 1:
LCD_Write_com(0x90+x); //第二行
break;
case 2:
LCD_Write_com(0x88+x); //第三行
break;
case 3:
LCD_Write_com(0x98+x); //第四行
break;
}
while(*s>0)
{
LCD_Write_data(*s++);
delay_ms(10);
}
}
//unsigned char defchar[] = {0x09,0x00,0xFD,0x08,0x09,0x08,0x09,
//0x10,0x09,0x20,0x79,0x40,0x41,0x04,
//0x47,0xFE,0x41,0x40,0x79,0x40,0x09,
//0x20,0x09,0x20,0x09,0x10,0x09,0x4E,
//0x51,0x84,0x21,0x00}; //自定义字符数组,这里是张
/*************************************************
* 函数功能 : 任意位置显示自定义字符串(写CGRAM中的数据)
* 函数参数 :x第几列,y第几行,addr 自编字符编码首地址,s输入的数据
* 声明 :12864具有4个自编字符,每个字符编码为0000H,0002H,0004H,0006H,4个自定义字符
* CGRAM地址分别为00H~0FH、10H~1FH、20H~2FH、30H~3FH
*************************************************/
void display_CGRAM(uchar y,uchar x,uchar addr,uchar *s)
{
unsigned char i,*address;
address = s;
LCD_Write_com(0x40+addr); //写CGRAM首行地址,创建第几个自编字符就写第几个自编字符的地址
for(i = 0;i<32;i++)
{
LCD_Write_data(*address++);
}
switch(y)
{
case 0:
LCD_Write_com(0x80+x); //第一行
break;
case 1:
LCD_Write_com(0x90+x); //第二行
break;
case 2:
LCD_Write_com(0x88+x); //第三行
break;
case 3:
LCD_Write_com(0x98+x); //第四行
break;
}
LCD_Write_data(0x00); //写自定义字符编码的高字节
switch(addr)
{
case 0x00:
LCD_Write_data(0x00); //写第一个自定义字符编码的低字节
break;
case 0x10:
LCD_Write_data(0x02); //写第二个自定义字符编码的低字节
break;
case 0x20:
LCD_Write_data(0x04); //写第三个自定义字符编码的低字节
break;
case 0x30:
LCD_Write_data(0x06); //写第四个自定义字符编码的低字节
break;
}
}
//led12864并行初始化
void LED_Init(void)
{
GPIO_WriteBit(GPIOA,RST,Bit_SET); //复位置高
delay_ms(10);
GPIO_WriteBit(GPIOA,RST,Bit_RESET); //复位
delay_ms(10);
GPIO_WriteBit(GPIOA,RST,Bit_SET); //复位结束
delay_ms(10);
LCD_Write_com(0x30); //设置为基本指令集动作
delay_ms(10);
LCD_Write_com(0x08); //设置显示、光标、闪烁全关。
LCD_Write_com(0x01); //清屏,并且DDRAM数据指针清零
LCD_Write_com(0x06); //进入模式设置
LCD_Write_com(0x0c); //开显示
delay_ms(10);
}
/*----------------------------------12864串行发送数据----------------------------------------------------*/
/*
* PSB引脚接低,串行模式
* RS引脚,拉高数据才能发送
*/
/*---------串行----------------*/
/* 字符显示RAM地址4行8列 */
uint8_t LCD_addr[4][8]={
{
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87}, //第一行
{
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97}, //第二行
{
0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F}, //第三行
{
0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F} //第四行
};
//led串行发送一个字节
void SendByte(u8 byte)
{
u8 i;
for(i = 0; i< 8; i++)
{
if((byte <<i)&0x80)
{
GPIO_WriteBit(GPIOA,RW,Bit_SET); //读操作 R/W=1
}
else
{
GPIO_WriteBit(GPIOA,RW,Bit_RESET); //写操作 R/W=0
}
GPIO_WriteBit(GPIOA,E,Bit_RESET); //不使能
delay_us(5); //延时使数据写入
GPIO_WriteBit(GPIOA,E,Bit_SET); //使能
}
}
//写命令
void Lcd_WriteCmd(u8 Cmd )
{
delay_ms(1); //由于我们没有写LCD正忙的检测,所以直接延时1ms,使每次写入数据或指令间隔大于1ms 便可不用写忙状态检测
SendByte(WRITE_CMD); //11111,RW(0),RS(0),0
SendByte(0xf0&Cmd); //高四位
SendByte(Cmd<<4); //低四位(先执行<<)
}
//写数据
void Lcd_WriteData(u8 Dat )
{
delay_ms(1); //由于我们没有写LCD正忙的检测,所以直接延时1ms,使每次写入数据或指令间隔大于1ms 便可不用写忙状态检测
SendByte(WRITE_DAT); //11111,RW(0),RS(1),0
SendByte(0xf0&Dat); //高四位
SendByte(Dat<<4); //低四位(先执行<<)
}
//清屏
void LCD_Clear(void)
{
Lcd_WriteCmd(0x01); //清屏指令
delay_ms(2); //延时以待液晶稳定【至少1.6ms】
}
/***********************************************************
* 函数功能 : 显示字符或汉字
* 参数 : x: row(0~3)
* y: line(0~7)
* str: 要显示的字符或汉字
***********************************************************/
void LCD_Display_Words(uint8_t x,uint8_t y,uint8_t*str)
{
Lcd_WriteCmd(LCD_addr[x][y]); //写初始光标位置
while(*str>0)
{
Lcd_WriteData(*str); //写数据
str++;
}
}
/*
* led12864串行初始化
*/
void Lcd_Init(void)
{
GPIO_WriteBit(GPIOA,RST,Bit_SET); //复位置高
delay_ms(10);
GPIO_WriteBit(GPIOA,RST,Bit_RESET); //复位
delay_ms(10);
GPIO_WriteBit(GPIOA,RST,Bit_SET); //复位结束
delay_ms(10);
delay_ms(50); //等待液晶自检(延时>40ms)
Lcd_WriteCmd(0x30); //功能设定:选择基本指令集 ,选择8bit数据流
delay_ms(1);//延时>137us
Lcd_WriteCmd(0x0c); //开显示
delay_ms(1); //延时>100us
Lcd_WriteCmd(0x01); //清除显示,并且设定地址指针为00H
delay_ms(30); //延时>10ms
Lcd_WriteCmd(0x06); //每次地址自动+1,初始化完成
LCD_Clear();
delay_ms(30); //延时>10ms
}
#include "sys.h"
/*----------------------------------12864并行发送----------------------------------------------------*/
#define RS GPIO_Pin_0 //数据/命令选择
#define RW GPIO_Pin_1 //读/写选择
#define E GPIO_Pin_2 //使能信号
#define RST GPIO_Pin_3 //复位引脚
#define LCD_DATA GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7 //数据线
#define uchar unsigned char
//引脚初始化
void lcd12864_Pin_Init(void);
//DB7引脚 忙信号检测
void LCD_ReadyWait(void);
//向LCD12864写入命令
void LCD_Write_com(uchar com);
//向LCD12864写入数据
void LCD_Write_data(uchar data);
//清屏
void LCD_clear(void);
//led12864并初始化
void LED_Init(void);
//任意位置显示字符串(写DDRAM中的数据)
void display_DDRAM(uchar x,uchar y,uchar *s);
//任意位置显示自定义字符串(写CGRAM中的数据)
void display_CGRAM(uchar y,uchar x,uchar addr,uchar *s);
/*----------------------------------12864串行发送----------------------------------------------------*/
#define WRITE_CMD 0xF8//写命令
#define WRITE_DAT 0xFA//写数据
/*
* PSB引脚接低,串行模式
* RS引脚,拉高数据才能发送
*/
//RW引脚为PA1,E引脚为PA2
//清屏
void LCD_Clear(void);
//led串行发送一个字节
void SendByte(u8 byte);
//写命令
void Lcd_WriteCmd(u8 Cmd );
//写数据
void Lcd_WriteData(u8 Dat );
//显示字符或汉字
void LCD_Display_Words(uint8_t x,uint8_t y,uint8_t*str);
//led12864串行初始化
void Lcd_Init(void);
文章浏览阅读290次,点赞8次,收藏10次。1.背景介绍稀疏编码是一种用于处理稀疏数据的编码技术,其主要应用于信息传输、存储和处理等领域。稀疏数据是指数据中大部分元素为零或近似于零的数据,例如文本、图像、音频、视频等。稀疏编码的核心思想是将稀疏数据表示为非零元素和它们对应的位置信息,从而减少存储空间和计算复杂度。稀疏编码的研究起源于1990年代,随着大数据时代的到来,稀疏编码技术的应用范围和影响力不断扩大。目前,稀疏编码已经成为计算...
文章浏览阅读217次。EasyGBS - GB28181 国标方案安装使用文档下载安装包下载,正式使用需商业授权, 功能一致在线演示在线API架构图EasySIPCMSSIP 中心信令服务, 单节点, 自带一个 Redis Server, 随 EasySIPCMS 自启动, 不需要手动运行EasySIPSMSSIP 流媒体服务, 根..._easygbs-windows-2.6.0-23042316使用文档
文章浏览阅读1.2k次,点赞27次,收藏7次。2023巅峰极客 BabyURL之前AliyunCTF Bypassit I这题考查了这样一条链子:其实就是Jackson的原生反序列化利用今天复现的这题也是大同小异,一起来整一下。_原生jackson 反序列化链子
文章浏览阅读734次,点赞9次,收藏7次。微服务架构简单的说就是将单体应用进一步拆分,拆分成更小的服务,每个服务都是一个可以独立运行的项目。这么多小服务,如何管理他们?(服务治理 注册中心[服务注册 发现 剔除])这么多小服务,他们之间如何通讯?这么多小服务,客户端怎么访问他们?(网关)这么多小服务,一旦出现问题了,应该如何自处理?(容错)这么多小服务,一旦出现问题了,应该如何排错?(链路追踪)对于上面的问题,是任何一个微服务设计者都不能绕过去的,因此大部分的微服务产品都针对每一个问题提供了相应的组件来解决它们。_spring cloud
文章浏览阅读5.9k次,点赞6次,收藏20次。Js实现图片点击切换与轮播图片点击切换<!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title></title> <script type="text/ja..._点击图片进行轮播图切换
文章浏览阅读10w+次,点赞245次,收藏1.5k次。在开始安装前,如果你的电脑装过tensorflow,请先把他们卸载干净,包括依赖的包(tensorflow-estimator、tensorboard、tensorflow、keras-applications、keras-preprocessing),不然后续安装了tensorflow-gpu可能会出现找不到cuda的问题。cuda、cudnn。..._tensorflow gpu版本安装
文章浏览阅读243次。0x00 简介权限滥用漏洞一般归类于逻辑问题,是指服务端功能开放过多或权限限制不严格,导致攻击者可以通过直接或间接调用的方式达到攻击效果。随着物联网时代的到来,这种漏洞已经屡见不鲜,各种漏洞组合利用也是千奇百怪、五花八门,这里总结漏洞是为了更好地应对和预防,如有不妥之处还请业内人士多多指教。0x01 背景2014年4月,在比特币飞涨的时代某网站曾经..._使用物联网漏洞的使用者
文章浏览阅读786次。A. Epipolar geometry and triangulationThe epipolar geometry mainly adopts the feature point method, such as SIFT, SURF and ORB, etc. to obtain the feature points corresponding to two frames of images. As shown in Figure 1, let the first image be and th_normalized plane coordinates
文章浏览阅读708次,点赞2次,收藏3次。开放信息抽取(OIE)系统(三)-- 第二代开放信息抽取系统(人工规则, rule-based, 先关系再实体)一.第二代开放信息抽取系统背景 第一代开放信息抽取系统(Open Information Extraction, OIE, learning-based, 自学习, 先抽取实体)通常抽取大量冗余信息,为了消除这些冗余信息,诞生了第二代开放信息抽取系统。二.第二代开放信息抽取系统历史第二代开放信息抽取系统着眼于解决第一代系统的三大问题: 大量非信息性提取(即省略关键信息的提取)、_语义角色增强的关系抽取
文章浏览阅读1.1w次,点赞6次,收藏51次。快速完成网页设计,10个顶尖响应式HTML5网页模板助你一臂之力为了寻找一个优质的网页模板,网页设计师和开发者往往可能会花上大半天的时间。不过幸运的是,现在的网页设计师和开发人员已经开始共享HTML5,Bootstrap和CSS3中的免费网页模板资源。鉴于网站模板的灵活性和强大的功能,现在广大设计师和开发者对html5网站的实际需求日益增长。为了造福大众,Mockplus的小伙伴整理了2018年最..._html欢迎页面
文章浏览阅读282次。原标题:2018全国计算机等级考试调整,一、二级都增加了考试科目全国计算机等级考试将于9月15-17日举行。在备考的最后冲刺阶段,小编为大家整理了今年新公布的全国计算机等级考试调整方案,希望对备考的小伙伴有所帮助,快随小编往下看吧!从2018年3月开始,全国计算机等级考试实施2018版考试大纲,并按新体系开考各个考试级别。具体调整内容如下:一、考试级别及科目1.一级新增“网络安全素质教育”科目(代..._计算机二级增报科目什么意思
文章浏览阅读240次。conan简单使用。_apt install conan