經典的推箱子是一個很古老的游戲了,相信大家都不陌生。其目的是在訓練我們的邏輯思考能力。在一個狹小的倉庫中,要求把木箱放到指定的位置,稍不小心就會出現箱子無法移動或者通道被堵住的情況,所以需要巧妙的利用有限的空間和通道,合理安排移動的次序和位置,才能順利的完成任務。
我們將編寫推箱子游戲,玩家鍵盤控制游戲角色將所有黃色箱子推到白色方塊處,效果如圖所示:
操作方法:方向鍵↑↓←→控制移動推箱子,將箱子推到對應位置。
首先學習字符串與字符數組的概念,并應用字符數組初始化關卡數據;然后利用鍵盤控制游戲角色移動,實現地圖元素更新和游戲勝利的判斷;接著利用三維字符數組,實現多關卡的游戲;最后學習基于文件的關卡數據讀取,利用枚舉類型改進游戲代碼。
源碼:
#include
#include
#include
#define B_SIZE 60 // 方塊大小
#define B_NUM 8 // 方塊個數,一共8*8個方塊
struct Player // 結構體,用于記錄玩家位置
{
int i;
int j;
};
Player player; // 玩家全局變量
enum Element // 定義枚舉類型,小方塊所有的可能的種類
{
wall,target,box,empty,achieved,role
};
// 用于存儲地圖數據,用枚舉類型實現
Element level[B_NUM][B_NUM] =
{{wall,wall,wall,wall,wall,wall,wall,wall},
{wall,wall,wall,target,box,empty,empty,wall},
{wall,empty,empty,empty,empty,empty,empty,wall},
{wall,empty,empty,empty,empty,empty,empty,wall},
{wall,empty,empty,empty,empty,empty,empty,wall},
{wall,role,empty,box,target,wall,wall,wall},
{wall,empty,empty,empty,empty,wall,wall,wall},
{wall,wall,wall,wall,wall,wall,wall,wall}};
int targetNum,achievedNum; // 目標位置個數、完成目標個數
void startup() // 初始化函數
{
initgraph(B_NUM*B_SIZE,B_NUM*B_SIZE); // 新開一個畫面
setbkcolor(RGB(150,150,150)); // 灰色背景
BeginBatchDraw(); // 開始批量繪圖
int i,j;
targetNum = 0; // 目標個數,初始為0
// 對二維數組遍歷
for (i=0;i
for (j=0;j
{
if (level[i][j]==role) // 找到地圖中player位置
{
player.i = i; // 設定player位置
player.j = j; //
level[i][j]=empty; // 把地圖元素變成空白empty
}
else if (level[i][j]==target || level[i][j]==achieved ) // 如果元素是target或achieved
targetNum++; // 目標個數+1
}
}
void show() // 繪制函數
{
int i,j;
cleardevice(); // 以背景顏色清空屏幕
// 遍歷關卡二維數組數據
for (i=0;i
{
for (j=0;j
{
if (level[i][j]==empty) // empty 元素是空白區(qū)域
{
setfillcolor(RGB(150,150,150)); // 繪制灰色地面
setlinecolor(RGB(150,150,150));
fillrectangle(j*B_SIZE,i*B_SIZE,(j+1)*B_SIZE,(i+1)*B_SIZE);
}
else if (level[i][j]==wall) // wall 元素是墻
{
setfillcolor(RGB(155,0,0));
setlinecolor(RGB(150,150,150)); // 繪制淡紅色、灰色線的方框
fillrectangle(j*B_SIZE,i*B_SIZE,(j+1)*B_SIZE,(i+1)*B_SIZE);
}
else if (level[i][j]==box) // box 元素是可移動的箱子
{
setfillcolor(RGB(255,255,0)); // 繪制一個黃色的方塊
setlinecolor(RGB(150,150,150));
fillrectangle(j*B_SIZE,i*B_SIZE,(j+1)*B_SIZE,(i+1)*B_SIZE);
}
else if (level[i][j]==target) // target 元素是目標
{
setfillcolor(RGB(250,250,250)); // 繪制一個白色的小方塊
fillrectangle((j+0.3)*B_SIZE,(i+0.3)*B_SIZE,
(j+0.7)*B_SIZE,(i+0.7)*B_SIZE);
}
else if (level[i][j]==achieved) // achieved 元素是已完成目標
{
setlinecolor(RGB(150,150,150));
setfillcolor(RGB(255,255,0)); // 繪制一個黃色的方塊
fillrectangle(j*B_SIZE,i*B_SIZE,(j+1)*B_SIZE,(i+1)*B_SIZE);
setfillcolor(RGB(250,250,250)); // 繪制一個白色的小方塊
fillrectangle((j+0.3)*B_SIZE,(i+0.3)*B_SIZE,
(j+0.7)*B_SIZE,(i+0.7)*B_SIZE);
}
}
}
// 以下繪制玩家,繪制一個人臉圖案
i = player.i;
j = player.j;
setfillcolor(RGB(255,0,0));
fillcircle((j+0.5)*B_SIZE,(i+0.5)*B_SIZE,0.4*B_SIZE);//一個紅色圓臉
setfillcolor(RGB(80,80,80));
setlinecolor(RGB(80,80,80));
fillcircle((j+0.3)*B_SIZE,(i+0.45)*B_SIZE,0.08*B_SIZE);//兩個黑色眼睛
fillcircle((j+0.7)*B_SIZE,(i+0.45)*B_SIZE,0.08*B_SIZE);
setlinestyle(PS_SOLID,3);
line((j+0.35)*B_SIZE,(i+0.7)*B_SIZE,(j+0.65)*B_SIZE,(i+0.7)*B_SIZE);//深灰色嘴巴
setlinestyle(PS_SOLID,1);
if (achievedNum==targetNum) // 如完成目標個數==目標個數
{
setbkmode(TRANSPARENT); // 透明顯示文字
settextcolor(RGB(0,255,255)); // 設置字體顏色
settextstyle(80, 0, _T("宋體")); // 設置字體大小、樣式
outtextxy(80,200,_T("游戲勝利")); // 顯示游戲勝利文字
}
FlushBatchDraw(); // 開始批量繪制
}
void update() // 每幀更新運行
{
if(kbhit() && (achievedNum// 如果按鍵,并且游戲沒有勝利
{
char input = getch(); // 獲取按鍵
if (input=='a' || input=='s' || input=='d' || input=='w') // 如果是有效按鍵
{
int goal_i = player.i; // 移動的目標位置
int goal_j = player.j;
int goalNext_i = goal_i; // 目標位置再向前的一個位置
int goalNext_j = goal_j;
// 根據用戶的不同按鍵輸入,獲得目標位置、再向前的一個位置
if (input=='a') // 向左
{
goal_j = player.j -1 ; // 目標位置在玩家位置的左邊
goalNext_j = goal_j-1; // 目標的下一個位置,在其再左邊
}
else if (input=='d') // 向右
{
goal_j = player.j +1 ; // 目標位置在玩家位置的右邊
goalNext_j = goal_j+1; // 目標的下一個位置,在其再右邊
}
else if (input=='s') // 向下
{
goal_i = player.i+1; // 目標位置在玩家位置的下邊
goalNext_i = goal_i+1; // 目標的下一個位置,在其再下邊
}
else if (input=='w') // 向上
{
goal_i = player.i-1; // 目標位置在玩家位置的上邊
goalNext_i = goal_i-1; // 目標的下一個位置,在其再上邊
}
// 根據不同地圖元素的情況,判斷如何移動角色和更新地圖元素
if (level[goal_i][goal_j]==empty || level[goal_i][goal_j]==target )
{ // 如果目標位置是empty,或者target
player.i = goal_i; // 玩家移動到目標位置
player.j = goal_j;
}
else if (level[goal_i][goal_j]==box && level[goalNext_i][goalNext_j]==empty )
{ // 如果目標位置是box,再前面一個是empty
player.i = goal_i; // 玩家移動到目標位置
player.j = goal_j;
level[goal_i][goal_j]=empty; // 目標位置變成empty
level[goalNext_i][goalNext_j]=box; // 再前面變成box
}
else if (level[goal_i][goal_j]==box && level[goalNext_i][goalNext_j]==target)
{ // 如果目標位置是box,再前面一個是target
player.i = goal_i; // 玩家移動到目標位置
player.j = goal_j;
level[goal_i][goal_j] = empty; // 目標位置變成empty
level[goalNext_i][goalNext_j] = achieved; // 再前面變成achieved
}
else if (level[goal_i][goal_j]==achieved && level[goalNext_i][goalNext_j]== empty)
{ // 如果目標位置是achieved,再前面一個是empty
player.i = goal_i; // 玩家移動到目標位置
player.j = goal_j;
level[goal_i][goal_j] = target; // 目標位置變成target
level[goalNext_i][goalNext_j] = box; // 再前面變成box
}
else if (level[goal_i][goal_j]==achieved && level[goalNext_i][goalNext_j]== target)
{ // 如果目標位置是achieved,再前面一個是target
player.i = goal_i; // 玩家移動到目標位置
player.j = goal_j;
level[goal_i][goal_j] = target; // 目標位置變成target
level[goalNext_i][goalNext_j] = achieved; // 再前面變成achieved
}
else // 其他情況都推不動
return; // 不做任何處理,函數直接返回
}
achievedNum = 0; // 完成目標個數,初始為0
int i,j;
for (i=0;i// 對二維數組遍歷
for (j=0;j//
if (level[i][j]==achieved) // 如果元素是achieved
achievedNum++; // 完成目標個數+1
}
}
int main() // 主函數
{
startup(); // 初始化
while (1) // 游戲主循環(huán)
{
show(); // 繪制
update(); // 更新
}
return 0;
}
主要講解了字符串與字符數組、文件讀寫、枚舉類型等語法知識,實現了推箱子游戲。讀者可以嘗試在本章代碼基礎上繼續(xù)改進:
1、實現多關卡的選擇界面;
2、實現某一步移動的撤銷功能(類似于下棋游戲中的悔棋功能);
3、實現按'h'鍵后進行提示,播放正確步驟動畫的功能;
4、實現一個圖形編輯器,并將設計的關卡信息保存為txt文件。
審核編輯 :李倩
-
C語言
+關注
關注
180文章
7614瀏覽量
137382 -
編輯器
+關注
關注
1文章
806瀏覽量
31249
原文標題:C語言實現推箱子游戲!(超簡單詳細)代碼思路+源碼分享
文章出處:【微信號:cyuyanxuexi,微信公眾號:C語言編程學習基地】歡迎添加關注!文章轉載請注明出處。
發(fā)布評論請先 登錄
相關推薦
評論