當前位置:首頁 > 單片機 > 單片機
[導讀]基于51單片機的貪吃蛇實例

 一、元器件

1、AT89C51

關于51單片機就不在啰嗦了,相信大家都已經很熟悉了。

2、8x8點陣

點陣里面就是一些二極管啦,通過縱橫交叉連接,橫8豎8,每個交叉點都接一個二極管。這里給大家找到一個點陣的實物圖

 


 

我想大家看到這個圖就應該知道如何去點亮一個點陣了。假如要點亮最左上角那個,那么9號引腳拉高,13號引腳拉低,這樣既可。

二、原理圖

 


 

 


 

三、項目分析

1、首先定義一個結構體

struct snake{

unsigned char x[20];

unsigned char y[20];

unsigned char length;

unsigned char direction;

}snk;

數(shù)組x,y分別存放每一個點的橫縱坐標,length為蛇的長度,direction為蛇前進的方向

2、坐標系:點陣的左下角為點(0,0),橫縱坐標都是正向增長,P2控制橫坐標;P0控制縱坐標。通過坐標可以找到點陣中點的位置,然后將其點亮

假設現(xiàn)在有第2個點的坐標x[2] = 1, y[2] = 2,那么點亮這個點的方式為

P2 = 0x04; //0000 0100

P0 = 0xfb; //1111 1011

3、按鍵產生外部中斷,在中斷里判斷按下那個方向get_direction(),并且同時設置坐標set_location()

4、定時器每隔1s就應該更新位置,因為蛇要不停的前進。定時器不需要更新方向,因為方向只有按鍵才會改動,定時器用前一步的方向

5、關于點的位置更新方式

1)、向上移動

后面的點去覆蓋前面的點,第一個點用新坐標表示x[0]不變,y[0]+1

2)、向下移動

后面的點去覆蓋前面的點,第一個點用新坐標表示x[0]不變,y[0]-1

3)、向左移動

后面的點去覆蓋前面的點,第一個點用新坐標表示x[0]-1,y[0]不變

4)、向右移動

后面的點去覆蓋前面的點,第一個點用新坐標表示x[0]+1,y[0]不變

6、關于邊界問題:

1)、任何一個點的橫坐標 0 <= x[i] < 8

2)、任何一個點的縱坐標 0 <= y[i] < 8

3)、第一個點在移動的時候不能和其他點重復,否則就自己追尾了

7、關于原理圖按鍵的設計

貪吃蛇要求系統(tǒng)能迅速響應按鍵,因此輪詢的方式并不可取,只有靠外部中斷。然而51只有2個外部中斷,我們起碼需要4個方向鍵,這樣就不能一個按鍵配一個外部中斷,通過使用4輸入與門,將所有按鍵狀態(tài)集合在一起,然后送給外部中斷0。我們將4個按鍵都接在與門,只要有一個按下,那么與門的輸出就會產生一個下降沿,從而產生外部中斷。

四、源代碼

main.c

#include "snake.h"

int error = 0;

int time=0;

void interrupt_init()

{

EA = 0; //關閉總中斷

IT0 = 1; //外部中斷0方式 下降沿

EA = 1; //開啟總中斷

EX0 = 1; //開啟外部中斷

}

void timer_init()

{

EA = 0; //關總中斷

ET0 = 1; //開定時器0中斷

TMOD = 0x02; //定時器0工作方式2

TL0 = 6; //定時250us

TH0 = 6;

EA = 1; //開總中斷

TR0 = 1; //開始定時

}

int main()

{

// unsigned char tempx, tempy;

// unsigned char i,j;

interrupt_init();

timer_init();

snk_init();

while(1)

{

//如果位置錯了就重新初始化蛇

if(error)

snk_init();

//點亮點陣

matrix();

}

}

void inter0() interrupt 0

{

//按鍵產生外部中斷,獲取新的方向

get_direction();

//設置新的位置

error = set_location();

// matrix();

}

void timer0() interrupt 1

{

time++;

//定時器為250us 積累4000次就是1s

if(time == 4000)

{

//每隔1s都需要重新設置位置,讓蛇前進

error = set_location();

time = 0;

}

}

snake.c

點擊(此處)折疊或打開

#include "snake.h"

//蛇的結構體,x為橫坐標,y為縱坐標,length為蛇的長度,direction為蛇的前進方向

struct snake{

unsigned char x[20];

unsigned char y[20];

unsigned char length;

unsigned char direction;

}snk;

void matrix()

{

unsigned char i;

int count=500;

//關閉所有的點

P2 = 0x00;

P0 = 0xff;

//根據(jù)蛇每一個點的坐標,將對應的點陣點亮

for(i=0; i

{

P2 = 1<

P0 = ~(1<

}

}

void snk_init()

{

//初始化坐標,總共4個點(3,0) (2,0) (1,0) (1,0)

snk.x[0] = 3;

snk.y[0] = 0;

snk.x[1] = 2;

snk.y[1] = 0;

snk.x[2] = 1;

snk.y[2] = 0;

snk.x[3] = 0;

snk.y[3] = 0;

//初始長度4

snk.length = 4;

//初始移動方向 向右

snk.direction = RIGHT;

//點亮點陣

matrix();

}

void get_direction()

{

//通過按鍵的狀態(tài)獲取方向

if(!up)

snk.direction = UP;

if(!down)

snk.direction = DOWN;

if(!left)

snk.direction = LEFT;

if(!right)

snk.direction = RIGHT;

}

int set_location()

{

unsigned char i;

int err = 0;

if(snk.direction == UP)

{

for(i=snk.length-1; i>0; i--)

{

snk.x[i] = snk.x[i-1];

snk.y[i] = snk.y[i-1];

}

//如果向上運動,第0個點的橫坐標不變,縱坐標加1

snk.x[0] = snk.x[0];

snk.y[0] = snk.y[0] + 1;

}

else if(snk.direction == DOWN)

{

for(i=snk.length-1; i>0; i--)

{

snk.x[i] = snk.x[i-1];

snk.y[i] = snk.y[i-1];

}

//如果向下運動,第0個點的橫坐標不變,縱坐標減1

snk.x[0] = snk.x[0];

snk.y[0] = snk.y[0] - 1;

}

else if(snk.direction == LEFT)

{

for(i=snk.length-1; i>0; i--)

{

snk.x[i] = snk.x[i-1];

snk.y[i] = snk.y[i-1];

}

//如果向左運動,第0個點的橫坐標減1,縱坐標不變

snk.x[0] = snk.x[0] - 1;

snk.y[0] = snk.y[0];

}

else

{

for(i=snk.length-1; i>0; i--)

{

snk.x[i] = snk.x[i-1];

snk.y[i] = snk.y[i-1];

}

//如果向右運動,第0個點的橫坐標加1,縱坐標不變

snk.x[0] = snk.x[0] + 1;

snk.y[0] = snk.y[0];

}

err = is_location_error();

return err;

}

int is_location_error()

{

unsigned char i;

//如果第0個點的坐標和其他任意一個點重復,那么蛇就自己撞自己,出錯

for(i=1; i

{

if((snk.x[0]==snk.x[i]) && (snk.y[0]==snk.y[i]))

return 1;

}

//如果蛇的坐標超出范圍,也出錯

if(snk.x[0]>7 || snk.y[0]>7)

return 1;

return 0;

}

snake.h

#include

//定義四個方向按鍵

sbit up = P3^4;

sbit down = P3^5;

sbit left = P3^6;

sbit right = P3^7;

//定義1個游戲級別按鍵

sbit level = P3^0;

//定義一個復位按鍵

sbit reset = P3^1;

//定義4個方向的值

#define RIGHT 0

#define UP 1

#define LEFT 2

#define DOWN 3

void delay_us();

void delay_10us();

void delay_ms();

void delay_10ms();

void delay_100ms();

void delay_s();

int is_location_error();

void matrix();

void snk_init();

void set_direction();

int get_location();

int is_location_error();

『本文轉載自網絡,版權歸原作者所有,如有侵權請聯(lián)系刪除』

本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯(lián)系該專欄作者,如若文章內容侵犯您的權益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或將催生出更大的獨角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉型技術解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術公司SODA.Auto推出其旗艦產品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關鍵字: 汽車 人工智能 智能驅動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務中斷的風險,如企業(yè)系統(tǒng)復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務連續(xù)性,提升韌性,成...

關鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網易近期正在縮減他們對日本游戲市場的投資。

關鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產業(yè)博覽會開幕式在貴陽舉行,華為董事、質量流程IT總裁陶景文發(fā)表了演講。

關鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數(shù)據(jù)產業(yè)博覽會上,華為常務董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權最終是由生態(tài)的繁榮決定的。

關鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經營業(yè)績穩(wěn)中有升 落實提質增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務引領增長 以科技創(chuàng)新為引領,提升企業(yè)核心競爭力 堅持高質量發(fā)展策略,塑強核心競爭優(yōu)勢...

關鍵字: 通信 BSP 電信運營商 數(shù)字經濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術學會聯(lián)合牽頭組建的NVI技術創(chuàng)新聯(lián)盟在BIRTV2024超高清全產業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術創(chuàng)新聯(lián)...

關鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關鍵字: BSP 信息技術
關閉
關閉