⑴ 單片機字母顯示
從DISP知道,這是一個顯示子程序
從SJMP DISP知道,這個是一個死循環,不停的在顯示
那麼顯示在哪裡
從P1可知,應該是P1口接了數碼管。
那麼顯示了什麼?
從#3CH可知,接了共陰極數碼管
顯示的是字母「b」且小數點不亮(可查閱段選碼)
然後為什麼這里是一個死循環?
應該是程序中有中斷,一直在等待中斷跳出。
SJMP DISP 跳到DISP:MOV P1,#3CH 這句,DISP是一個標號。也就是如果沒有中斷產生
那就一直在執行
DISP:MOV P1,#3C
-----SJMP DISP
可以看到你的程序是有中斷的
MOV IE,#85H中斷開放
-----SETB PX1
-----SETB IT0
-----SETB IT1
⑵ 顯示器如何給單片機命令顯示
靜心佛門
關注
單片機c語言顯示器編程,51單片機驅動LCD1602程序設計(C語言)很詳細的教程 轉載
2021-05-19 18:39:27
1點贊
靜心佛門
碼齡3年
關注
字元液晶絕大多數是基於HD44780液晶晶元的,控制原理是完全相同的,因此HD44780寫的控製程序可以很方便地應用於市面上大部分的字元型液晶。字元型LCD通常有14條引腳線或16條引腳線的LCD,多出來的2條線是背光電源線VCC(15腳)和地線GND(16腳),其控制原理與14腳的LCD完全一樣,定義如下表所示:
字元型LCD的引腳定義
.png
HD44780內置了DDRAM、CGROM和CGRAM。DDRAM就是顯示數據RAM,用來寄存待顯示的字元代碼。共80個位元組,其地址和屏幕的對應關系如下表:
.png
也就是說想要在LCD1602屏幕的第一行第一列顯示一個"A"字,就要向DDRAM的00H地址寫入「A」字的代碼就行了。但具體的寫入是要按LCD模塊的指令格式來進行的。在1602中我們用前16個就行了。第二行也一樣用前16個地址。對應如下:
DDRAM地址與顯示位置的對應關系
.png
.png
文本文件中每一個字元都是用一個位元組的代碼記錄的。一個漢字是用兩個位元組的代碼記錄。在PC上我們只要打開文本文件就能在屏幕上看到對應的字元是因為在操作系統里和BIOS里都固化有字元字模。什麼是字模?就代表了是在點陣屏幕上點亮和熄滅的信息數據。
例如「A」字的字模:
01110○■■■○
10001■○○○■
10001■○○○■
10001■○○○■
11111■■■■■
10001■○○○■
10001■○○○■
上圖左邊的數據就是字模數據,右邊就是將左邊數據用「○」代表0,用「■」代表1。看出是個「A」字了嗎?在文本文件中「A」字的代碼是41H,PC收到41H的代碼後就去字模文件中將代表A字的這一組數據送到顯卡去點亮屏幕上相應的點,你就看到「A」這個字了。
剛才說了想要在LCD1602屏幕的第一行第一列顯示一個"A"字,就要向DDRAM的00H地址寫入「A」字的代碼41H就行了,可41H這一個位元組的代碼如何才能讓LCD模塊在屏幕的陣點上顯示「A」字呢?同樣,在LCD模塊上也固化了字模存儲器,這就是CGROM和CGRAM。HD44780內置了192個常用字元的字模,存於字元產生器CGROM(Character Generator ROM)中,另外還有8個允許用戶自定義的字元產生RAM,稱為CGRAM(Character Generator RAM)。下圖說明了CGROM和CGRAM與字元的對應關系。
.png
從上圖可以看出,「A」字的對應上面高位代碼為0100,對應左邊低位代碼為0001,合起來就是01000001,也就是41H。可見它的代碼與我們PC中的字元代碼是基本一致的。因此我們在向DDRAM寫C51字元代碼程序時甚至可以直接用P1='A'這樣的方法。PC在編譯時就把「A」先轉為41H代碼了。
字元代碼0x00~0x0F為用戶自定義的字元圖形RAM(對於5X8點陣的字元,可以存放8組,5X10點陣的字元,存放4組),就是CGRAM了。後面我會詳細說的。
0x20~0x7F為標準的ASCII碼,0xA0~0xFF為日文字元和希臘文字元,其餘字元碼(0x10~0x1F及0x80~0x9F)沒有定義。
那麼如何對DDRAM的內容和地址進行具體操作呢,下面先說說HD44780的指令集及其設置說明,請瀏覽該指令集,並找出對DDRAM的內容和地址進行操作的指令。共11條指令:
1.清屏指令
.png
功能:<1> 清除液晶顯示器,即將DDRAM的內容全部填入"空白"的ASCII碼20H;
<2> 游標歸位,即將游標撤回液晶顯示屏的左上方;
<3> 將地址計數器(AC)的值設為0。
2.游標歸位指令
.png
功能:<1> 把游標撤回到顯示器的左上方;
<2> 把地址計數器(AC)的值設置為0;
<3> 保持DDRAM的內容不變。
3.進入模式設置指令
.png
200862219425666.jpg功能:設定每次定入1位數據後游標的移位方向,並且設定每次寫入的一個字元是否移動。參數設定的
情況如下所示:
位名 設置
I/D 0=寫入新數據後游標左移 1=寫入新數據後游標右移
S 0=寫入新數據後顯示屏不移動 1=寫入新數據後顯示屏整體右移1個字元
4.顯示開關控制指令
.png
功能:控制顯示器開/關、游標顯示/關閉以及游標是否閃爍。參數設定的情況如下:
位名 設置
D 0=顯示功能關 1=顯示功能開
C 0=無游標 1=有游標
B 0=游標閃爍 1=游標不閃爍
5.設定顯示屏或游標移動方向指令
.png
功能:使游標移位或使整個顯示屏幕移位。參數設定的情況如下:
S/C R/L 設定情況
0 0 游標左移1格,且AC值減1
0 1 游標右移1格,且AC值加1
1 0 顯示器上字元全部左移一格,但游標不動
1 1 顯示器上字元全部右移一格,但游標不動
6.功能設定指令
.png
功能:設定數據匯流排位數、顯示的行數及字型。參數設定的情況如下:
位名 設置
DL 0=數據匯流排為4位 1=數據匯流排為8位
N 0=顯示1行 1=顯示2行
F 0=5×7點陣/每字元 1=5×10點陣/每字元
7.設定CGRAM地址指令
.png
功能:設定下一個要存入數據的CGRAM的地址。
8.設定DDRAM地址指令
.png
功能:設定下一個要存入數據的CGRAM的地址。
9.讀取忙信號或AC地址指令
.png
功能:<1> 讀取忙碌信號BF的內容,BF=1表示液晶顯示器忙,暫時無法接收單片機送來的數據或指令;當BF=0時,液晶顯示器可以接收單片機送來的數據或指令;
<2> 讀取地址計數器(AC)的內容。
10.數據寫入DDRAM或CGRAM指令一覽
.png
功能:<1> 將字元碼寫入DDRAM,以使液晶顯示屏顯示出相對應的字元;
<2> 將使用者自己設計的圖形存入CGRAM。
11.從CGRAM或DDRAM讀出數據的指令一覽
.png
功能:讀取DDRAM或CGRAM中的內容。
基本操作時序:
讀狀態 輸入:RS=L,RW=H,E=H 輸出:DB0~DB7=狀態字
寫指令 輸入:RS=L,RW=L,E=下降沿脈沖,DB0~DB7=指令碼 輸出:無
讀數據 輸入:RS=H,RW=H,E=H 輸出:DB0~DB7=數據
寫數據 輸入:RS=H,RW=L,E=下降沿脈沖,DB0~DB7=數據 輸出:無
顯示操作的過程:首先確認顯示的位置,即在第幾行,第幾個字元開始顯示。也就是要顯示的地址,如下表所示的顯示地址。
第一行的顯示地址是0x80-0x8F,第二行的顯示地址是0xC0-0xCF。例如想要在第2行,第3個位置顯示一個字元,那麼地址碼就是0xC2。在編程過程中,通常編寫一個函數確定在某行某個位置顯示[url=]數據[/url]。函數需要 行[url=]參數[/url](y),和 列參數(x)來確定顯示位置。[url=]程序[/url]參考如下
/***************設置顯示位置**************************/
void LCD_set_xy( unsigned char x, unsigned char y )
{
unsigned char address;
if (0 == y) x |= 0x80; //當要顯示第一行時地址碼+0x80;
else x |= 0xC0; //在第二行顯示是地址碼+0xC0;
Write_com(x); //發送地址碼 0x80-0x8F 或者0xC0-0xCF
}
其次設置要顯示的內容,即上面提到的CGROM內的字元編碼。如顯示「A」,將編碼41H寫入到液晶屏顯示即可。通常設置地址和顯示內容用一個函數來完成。代碼參考如下:
//功能:按指定位置顯示一個字元
//輸入:列顯示地址x(取值范圍0-15) 行顯示地址y(取值范圍0-1), 指定字元
void DisplayOneChar(unsigned char x, unsigned char y, unsigned char Data)
{
if (0 == y) x |= 0x80; //當要顯示第一行時地址碼+0x80;
else x |= 0xC0; //在第二行顯示是地址碼+0xC0;
Write_com(x); //發送地址碼
Write_dat(Data); //發送要顯示的字元編碼
}
顯示字元「A」調用過程如下代碼:
DisplayOneChar(0,0,0x41);
//功能:在第1行 第1個字元 顯示一個大寫字母A
在C語言操作時,還可以顯示整個字元串。定義一個字元串顯示函數,可
以通過直接輸入字元方式進行顯示
//功能:按指定位置顯示一串字元
//輸入:列顯示地址x(取值范圍0-15) 行顯示地址y(取值范圍0-1), 指定字元串指針*p,要顯示的字元個數count (取值范圍1-16)
void DisplayListChar (unsigned char x,unsigned char y,unsigned char *p,unsigned char count)
{
unsigned char i;
for(i=0;i {
if (0 == y) x |= 0x80; //當要顯示第一行時地址碼+0x80;
else x |= 0xC0; //在第二行顯示是地址碼+0xC0;
Write_com(x); //發送地址碼
Write_dat(*p); //發送要顯示的字元編碼
x++;
p++;
}
}
調用方法如下:
DisplayListChar(0,0,"hello world",11); //液晶1602第一行顯示
DisplayListChar(0,1,"www*qm999*cn",12); //液晶1602第二行顯示
.png
舉個實例,就在LCD1602屏幕上第一行第一列顯示個「A」字。
//先定義介面
# include /*****************************************
P1------DB0~DB7
P2.0------RS
P2.1------RW
P2.2------E
*****************************************/
# define LCD_DB P1
sbit LCD_RS=P2^0;
sbit LCD_RW=P2^1;
sbit LCD_E=P2^2;
/******定義函數****************/
# define uchar unsigned char
# define uint unsigned int
void LCD_init(void);//初始化函數
void LCD_write_command(uchar command);//寫指令函數
void LCD_write_data(uchar dat);//寫數據函數
void LCD_disp_char(uchar x,uchar y,uchar dat);//在某個屏幕位置上顯示一個字元,X(0-16),y(1-2)
//void LCD_check_busy(void);//檢查忙函數。我沒用到此函數,因為通過率極低。
void delay_n40us(uint n);//延時函數
//********************************
//*******初始化函數***************
void LCD_init(void)
{
LCD_write_command(0x38);//設置8位格式,2行,5x7
LCD_write_command(0x0c);//整體顯示,關游標,不閃爍
LCD_write_command(0x06);//設定輸入方式,增量不移位
LCD_write_command(0x01);//清除屏幕顯示
delay_n40us(100);//實踐證明,用for循環200次就能可靠完成清屏指令。
⑶ 單片機中的AJMP LJMP SJMP JMP有什麼區別
1、位元組不同
SJMP 如果跳轉到的標號地址距離當前PC所指的地址小於256位元組,用SJMP。
AJMP 如果跳轉到的標號地址距離當前PC所指的地址小於2K位元組,用AJMP。
LJMP 如果跳轉到的標號地址距離當前PC所指的地址小於64K位元組,用LJMP。
ACALL是調用的子程序入口地址距離當前PC所指地址需介於0~2K,LCALL是0~64K。
2、跳轉的范圍不一樣。
AJMP 的范圍是11位地址,也就是2K的空間內,佔用存儲空間2個位元組,執行周期24個時鍾周。
LJMP 的范圍是16位地址,也就是64K的空間內,佔用存儲空間3個位元組,執行周期24個時鍾周期。
SJMP 的范圍是8位地址,也就是256BIT的空間內,佔用存儲空間2個位元組,執行周期24個時鍾周期。
JMP一般配合DPTR使用,存儲空間1個位元組,執行周期24個時鍾周期。一般用於多分枝選擇的時候使用,比如按鍵處理。
ACALL,LCALL和以上說明類似,是調用指令,ACALL佔用存儲空間2個位元組,執行周期24個時鍾周期。LCALL佔用存儲空間3個位元組,執行周期24個時鍾周期。
(3)單片機中字幕指令擴展閱讀:
①短程轉移(直接短轉移)
指令格式:JMP SHORT OPRD
語法格式: JMP 地址標號 ;(IP)←(IP)+8位位移量
指令功能:OPRD為轉移地址的標號,指令中的SHORT規定了OPRD為有符號的8位二進制數,OPRD為轉移地址的偏移量。該指令將程序執行的順序轉移到由(IP)+OPRD形成的新的程序執行的目標地址,從而實現程序的轉移。
轉移的目標地址OPRD在指令中可以直接使用標號地址,但要求轉移的目標地址的范圍只能在JMP指令所處地址的-128~+127位元組范圍之內,如超出該范圍,匯編時出錯。
② 近程轉移(段內直接轉移)
指令格式:JMP NEAR PTR OPRD
語法格式: JMP 地址標號 ;(IP)←(IP)+16位位移量
指令功能:與短程轉移的功能和要求相同,不同之處是近程轉移的OPRD為有符號的16位二進制數,指令將程序執行的順序轉移到由(IP)+OPRD形成的新的程序執行的目標地址。
轉移的目標地址的范圍只能在JMP指令所處地址的-32768~+32767位元組范圍之內,如超出該范圍,匯編時出錯。使用該指令時NEAR可省略
③段間直接轉移(遠程轉移)
指令格式:JMP FAR PTR OPRD
語法格式: JMP 地址標號 ;(IP)←新的偏移地址, ;(CS)←新的代碼段地址
指令功能:指令中用FAR PTR規定了該指令為段間的轉移,OPRD為目的地址的標號,目的地址與JMP指令所在地址不在同一段內。執行該指令時要修改CS和IP的內容,將OPRD所在段的段地址送CS中,OPRD的段內偏移地址送IP中。
④ 段內間接轉移
指令格式:JMP WORD PTR OPRD
語法格式:JMP reg16/mem ;(IP)←新的偏移地址
指令功能:與短程轉移的功能和要求相同,不同之處是段內間接轉移的OPRD 可以是除立即數外的任何寄存或存儲器定址方式,轉移的目標地址由OPRD的內容確定。
⑤ 段間間接轉移
指令格式:JMP DWORD PTR OPRD
語法格式:JMP mem32
指令功能:指令中用DWORD PTR規定了該指令為段間間接轉移,OPRD只能是存儲器定址方式。執行該指令時將定址到的內存單元的第一個字送入IP中,第二個字送入CS中。
⑷ 怎樣用51單片機和led8*8矩陣進行字元漢字顯示
8*8也就能顯示字元,顯示漢字比較吃力。
#include<reg51.h>
#include<intrins.h>
#define uchar unsigned char
#define uint unsigned int
uchar code Table_of_Digits[]=
{
0x00,0x3e,0x41,0x41,0x41,0x3e,0x00,0x00, //0
0x00,0x00,0x00,0x21,0x7f,0x01,0x00,0x00, //1
0x00,0x27,0x45,0x45,0x45,0x39,0x00,0x00, //2
0x00,0x22,0x49,0x49,0x49,0x36,0x00,0x00, //3
0x00,0x0c,0x14,0x24,0x7f,0x04,0x00,0x00, //4
0x00,0x72,0x51,0x51,0x51,0x4e,0x00,0x00, //5
0x00,0x3e,0x49,0x49,0x49,0x26,0x00,0x00, //6
0x00,0x40,0x40,0x40,0x4f,0x70,0x00,0x00, //7
0x00,0x36,0x49,0x49,0x49,0x36,0x00,0x00, //8
0x00,0x32,0x49,0x49,0x49,0x3e,0x00,0x00, //9
0xff,0x81,0x81,0x81,0x81,0x81,0x81,0xff
};
uchar code xdat[8]={0x80,0x40,0x20,0x10,0x08,0x04,0x02,0x01};
uchar code ydat[8]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
uchar i=0,j=0,t=0,Num_Index,key,xi,yi;
sbit we1=P1^1;
sbit we2=P1^3;
//主程序
void main()
{
//P1=0x80;
Num_Index=0; //從0 開始顯示
TMOD=0x01; //T0 方式0
TH0=(65536-2000)/256; //2ms 定時
TL0=(65536-2000)%256;
IE=0x82;
key=0;
xi=0;
yi=0;
EX0=1;
IT0=1;
TR0=1; //啟動T0
while(1);
}
//T0 中斷函數
void ext_int0() interrupt 0
{
key++;
key&=0x03;
}
void LED_Screen_Display() interrupt 1
{
TH0=(65536-2000)/256; //2ms 定時
TL0=(65536-2000)%256;
switch(key)
{
case 0:
P0=0xff;
we1=1;
P0=~Table_of_Digits[Num_Index*8+i];
we1=0;
P0=0xff; //輸出位碼和段碼
we2=1;
P0=xdat[i];
we2=0;
if(++i==8) i=0; //每屏一個數字由8 個位元組構成
if(++t==250) //每個數字刷新顯示一段時間
{
t=0;
if(++Num_Index==10) Num_Index=0; //顯示下一個數字
}
break;
case 1:
we1=1;
P0=~xdat[xi];
we1=0;
we2=1;
P0=ydat[yi];
we2=0;
if(++t==250) //每個數字刷新顯示一段時間
{
t=0;
yi++;
if(yi>7){yi=0;xi++;}
if(xi>7)xi=0;
}
break;
case 2:
we1=1;
P0=0x00;
we1=0;
P0=0xff; //輸出位碼和段碼
we2=1;
P0=xdat[i];
we2=0;
if(++t==250) //每個數字刷新顯示一段時間
{
if(++i==8) i=0; //每屏一個數字由8 個位元組構成
t=0;
}
break;
default:
key=0;
i=0;
j=0;
t=0;
xi=0;
yi=0;
Num_Index=0;
we1=1;
P0=0xff;
we1=0;
we2=1;
P1=0x80;
we2=0;
break;
}
}