一. 顯示圖片
顯示圖片的要點(diǎn)在于:1.取模 2.利用擴(kuò)展指令設(shè)置液晶3.清楚液晶地址的概念 1.1取模 取模軟件用的是“字模221”下圖是他的參數(shù)設(shè)置 
這里對參數(shù)設(shè)置先解釋一下,所謂橫向取模就是,一張圖片從圖片最左上角的8位開始取模,從左向右,每次取8位二進(jìn)制數(shù)據(jù)轉(zhuǎn)化為16進(jìn)制保存,第一排取完之后,接著到第二排最左邊8位開始取模,以此類推。為什么要這樣取模呢?因?yàn)?2864液晶的橫縱坐標(biāo)就是這個(gè)取模順序,這點(diǎn)在之后還會(huì)詳細(xì)提到。 1.2液晶的設(shè)置 液晶顯示圖片必須用擴(kuò)充指令集,初始化和顯示字符的初始化不一樣。我用的初始化函數(shù)如下:
void init_picture()//顯示圖片的初始化函數(shù) { lcd_wcmd(0x36);//寫指令函數(shù),擴(kuò)充指令集,繪圖G=1打開 lcd_wcmd(0x36); lcd_wcmd(0x3E); lcd_wcmd(0x01);//清屏 Light= 0;//打開背光 } 1.3將取出的字模寫進(jìn)相應(yīng)地址 首先應(yīng)該知道地址究竟是怎樣的,結(jié)合下圖講清楚。12864液晶分為半屏和下半屏。當(dāng)你想要點(diǎn)亮某個(gè)地方時(shí),必須先寫這個(gè)地方的垂直地址緊接著寫入它的水平地址,水平地址液晶可以自動(dòng)加1,而垂直地址不會(huì)。圖中水平坐標(biāo)從0x80+00到0x80+0F,一共16個(gè),其中0x80+00到0x80+07是上半屏的坐標(biāo),其中0x80+08到0x80+0F是下半屏的坐標(biāo)。水平坐標(biāo)每個(gè)兩字節(jié),先寫入的數(shù)據(jù)填充在高字節(jié)。垂直坐標(biāo)只有0x80+00到0x80+1F,圖中上半截0x80+00到0x80+1F是上半屏的垂直地址,另外的那部分一樣的是下半屏垂直地址。每個(gè)垂直地址只確定一排,所以水平和垂直地址不能確定某個(gè)點(diǎn)的位置,只能確定某個(gè)兩字節(jié)的位置,通過寫進(jìn)2字節(jié)數(shù)據(jù)確定點(diǎn)亮某個(gè)點(diǎn)或幾個(gè)點(diǎn)。比如我們寫入lcd_wcmd(0x81;(垂直地址)lcd_wcmd(0x80)(水平地址); 這就是說我們將在圖中水平坐標(biāo)00,垂直坐標(biāo)01的位置(紅圈處)輸入數(shù)據(jù)。 
知道地址的知識(shí)之后就明白為什么要橫向取模了,接下來只要將取模的數(shù)據(jù)一個(gè)個(gè)按取模生成的順序?qū)戇M(jìn)液晶就行了。下面是我用的代碼,其中uchar=unsignedchar,uint=unsigned int。 void show_Pic(uchar*address)//顯示圖片函數(shù) { //address是是指向數(shù)組的指針,用法:show_Pic(XY)當(dāng)中XY為數(shù)組名 uchari,j; for(i=0;i<32;i++) //上半屏的32排依次先寫滿 { lcd_wcmd(0x80+i);//先送垂直地址 lcd_wcmd(0x80); //再送水平地址,水平地址可自加1 for(j=0;j<16;j++)//每排128個(gè)點(diǎn),所以一共要16個(gè)兩位16進(jìn)制數(shù)(也就是8位二進(jìn)制數(shù))才能全部控制 { lcd_wdat(*address); address++; } } for(i=0;i<32;i++) //下半屏的32排操作原理和上半屏一樣 { lcd_wcmd(0x80+i); lcd_wcmd(0x88); for(j=0;j<16;j++) { lcd_wdat(*address); address++; } } } 1.4實(shí)際顯示結(jié)果 下圖是我顯示的一個(gè)坐標(biāo)系和一條龍 

下面的代碼是我的主函數(shù),這部分加上上面我上面講的函數(shù)和一些基本設(shè)置就是整個(gè)代碼,XY是這幅坐標(biāo)圖片取模得到的數(shù)組
void main() { init_picture(); show_Pic(XY); while(1) //進(jìn)入程序主循環(huán) { } } 二. 用打點(diǎn)方式顯示任意圖像 有的同學(xué)認(rèn)為打點(diǎn)只需按照上面的顯示圖片的方法點(diǎn)亮需要的點(diǎn)就是了,其實(shí)這不行。因?yàn)槟銓戇M(jìn)去的是八個(gè)點(diǎn)的控制,會(huì)影響周圍的點(diǎn),很容易出現(xiàn)亂碼。比如,你寫進(jìn)去的是0x80,x想的是只點(diǎn)亮左邊一個(gè)點(diǎn),其他的都不要影響,但是右邊的7個(gè)0也是會(huì)顯示的,如果在要顯示0的地方原來顯示的是1的話,你現(xiàn)在寫進(jìn)去的0就把1覆蓋了,這樣就容易產(chǎn)生亂碼。所以我建議,你先把液晶的數(shù)據(jù)讀出來,再用data&=(~(0x01<<(7-bit)))(bit是你點(diǎn)亮哪一位,data是讀出來的數(shù)據(jù))置0,置1也是類似的,最后再把data重新寫進(jìn)去就行了。 我的單片機(jī)接口是亂的,所以每一個(gè)液晶接口都用了位定義,并且用到了寄存器B,寄存器B的每一個(gè)當(dāng)做一個(gè)變量的位來操作。我的具體代碼如下:
bit lcd_busy() // 讀寫判斷數(shù)據(jù)的D7讀寫位,用于判斷1602是否忙 { bitresult; D7 =1; //數(shù)據(jù)口D7置1,為讀狀態(tài)做準(zhǔn)備 LCD_RS = 0; // 選擇指令寄存 LCD_RW = 1; // 選擇讀控制線 LCD_EN = 1; // 開使能控制線 delayNOP(); // 時(shí)序延時(shí) result = D7; // 讀D7的電平 LCD_EN = 0; // 關(guān)使能 return(result); // 返回值1:忙,0可以執(zhí)行操作 } unsigned charreadData(void) //讀取數(shù)據(jù)函數(shù) { uchar i; D0 = 1; D1 = 1; D2 = 1; D3 = 1; D4 = 1; D5 = 1; D6 = 1; D7 = 1; lcd_busy(); LCD_RS=1; LCD_RW=1; LCD_EN=0; LCD_EN=1; B_0=D0;B_1=D1; B_2=D2; B_3=D3;//B_1=B^1,在程序前段有位定義 B_4=D4;B_5=D5; B_6=D6;B_7=D7; for(i=1;i<=7;i++) delayNOP();//這個(gè)函數(shù)是{_nop_();_nop_();_nop_();_nop_();}; LCD_EN=0; returnB; } uchar DrawDots(uchar x,uchar y,ucharcolor)//打點(diǎn)函數(shù) { ucharrow,xlabel,xlabel_bit; uchar Read_H=0,Read_L=0; lcd_wcmd(0x34); //擴(kuò)充指令 lcd_wcmd(0x36); //繪圖指令 xlabel=x>>4; //去16*16首地址 xlabel_bit=x & 0x0F; if(y<32) row=y; else { row=y-32; xlabel+=8; } lcd_wcmd(row+0x80); lcd_wcmd(xlabel+0x80); readData(); Read_H=readData(); Read_L=readData(); lcd_wcmd(row+0x80); lcd_wcmd(xlabel+0x80); if(xlabel_bit<8) { switch(color) { case 0:Read_H&=(~(0x01<<(7-xlabel_bit))); break; //變暗,看不見 case 1:Read_H |=(0x01<<(7-xlabel_bit));break; //變亮 ,看得見 case 2:Read_H ^=(0x01<<(7-xlabel_bit));break; //反轉(zhuǎn) default:break; } lcd_wdat(Read_H); lcd_wdat(Read_L); } else { switch(color) { case 0:Read_L&=(~(0x01<<(15-xlabel_bit))); break; //變暗 ,看不見 case 1:Read_L |=(0x01<<(15-xlabel_bit));break; //變亮 ,看得見 case 2:Read_L ^=(0x01<<(15-xlabel_bit));break; //反轉(zhuǎn) default:break; } lcd_wdat(Read_H); lcd_wdat(Read_L); } lcd_wcmd(0x30);//恢復(fù)正常模式 } 如果你的單片機(jī)和12864接口不是亂的,那可以這樣寫讀取函數(shù) unsigned charreadData(void) { uchar i ,data; P0=0xFF; lcd_busy(); LCD_RS=1; LCD_RW=1;// LCD_EN=0; LCD_EN=1; data=P0; for(i=1;i<=7;i++) delayNOP(); LCD_EN=0; returndata; } 這樣,整個(gè)代碼就全部講完了,下面是顯示正弦波的圖片 
若有疏漏之處,希望大家多指正
|