C語言實(shí)現(xiàn)俄羅斯方塊
本文實(shí)例為大家分享了C語言俄羅斯方塊的具體代碼,供大家參考,具體內(nèi)容如下
本代碼運(yùn)行環(huán)境是Windows下的VS2013
首先創(chuàng)建tetris.cpp
然后依次創(chuàng)建view.h以及view.cpp、model.h以及model.cpp。
代碼如下:
view.h
#pragma once #include <stdio.h> void ShowBackground(); void ShowBrick(); void ShowGame(); void OnLeft(); void OnRight(); void OnUp(); void OnDown();
view.cpp
#include <stdlib.h>
#include "view.h"
#include "model.h"
void OnLeft()
{//如果能夠左移,則左移
if (IsCanMove(g_nRow, g_nCol - 1))
{
g_nCol--;
ShowGame();
}
}
void OnRight()
{
if (IsCanMove(g_nRow, g_nCol + 1))
{
g_nCol++;
ShowGame();
}
}
void OnUp()
{
if (IsCanRotate())
{
Rotate();
ShowGame();
}
}
void OnDown()
{
if (IsCanMove(g_nRow+1, g_nCol))
{
g_nRow++;
ShowGame();
}
else
{
//固定方塊至背景,并且產(chǎn)生新方塊
CombineBgBrick();
GetNewBrick();
//判斷游戲是否結(jié)束,并給出對應(yīng)提示
}
}
void ShowGame()
{
system("cls");
CombineBgBrick();
ShowBackground();
DetachBgBrick();
}
void ShowBrick()
{
for (size_t i = 0; i < 4; i++)
{
for (size_t j = 0; j < 4; j++)
{
if (g_chBrick[i][j] == 1)
{
printf("■");
}
}
printf("\r\n");
}
}
void ShowBackground()
{
for (size_t nRow = 0; nRow < GAME_ROWS; nRow++)
{
for (size_t nCol = 0; nCol < GAME_COLS; nCol++)
{
if (g_chBackground[nRow][nCol] == 1)
{
printf("■");
}
else
{
printf("□");
}
}
printf("\r\n");
}
}
model.cpp
#include <stdio.h>
#include <time.h>
#include <stdlib.h>
#include "model.h"
char g_chBackground[GAME_ROWS][GAME_COLS];
char g_chBrick[4][4];
int g_nShape = 0; //是長條還是方塊,系數(shù)為16
int g_nRotate = 0; //朝向,系數(shù)為4
int g_nRow = 0;
int g_nCol = 0;
char g_chBrickPool[][4] = {
// 長條
1, 1, 1, 1,
0, 0, 0, 0,
0, 0, 0, 0,
0, 0, 0, 0,
1, 0, 0, 0,
1, 0, 0, 0,
1, 0, 0, 0,
1, 0, 0, 0,
1, 1, 1, 1,
0, 0, 0, 0,
0, 0, 0, 0,
0, 0, 0, 0,
1, 0, 0, 0,
1, 0, 0, 0,
1, 0, 0, 0,
1, 0, 0, 0,
// T形
1, 1, 1, 0,
0, 1, 0, 0,
0, 0, 0, 0,
0, 0, 0, 0,
0, 1, 0, 0,
1, 1, 0, 0,
0, 1, 0, 0,
0, 0, 0, 0,
0, 1, 0, 0,
1, 1, 1, 0,
0, 0, 0, 0,
0, 0, 0, 0,
1, 0, 0, 0,
1, 1, 0, 0,
1, 0, 0, 0,
0, 0, 0, 0,
//L形狀
1, 0, 0, 0,
1, 0, 0, 0,
1, 1, 0, 0,
0, 0, 0, 0,
1, 1, 1, 0,
1, 0, 0, 0,
0, 0, 0, 0,
0, 0, 0, 0,
1, 1, 0, 0,
0, 1, 0, 0,
0, 1, 0, 0,
0, 0, 0, 0,
0, 0, 1, 0,
1, 1, 1, 0,
0, 0, 0, 0,
0, 0, 0, 0,
};
int IsCanRotate()
{
char chNextShape[4][4] = { 0 };
int nNextRotate = (g_nRotate + 1) % 4;
int nPoolRows = g_nShape * 16 + nNextRotate * 4;
for (size_t nRow = 0; nRow < 4; nRow++)
{
for (size_t nCol = 0; nCol < 4; nCol++)
{
chNextShape[nRow][nCol] = g_chBrickPool[nRow + nPoolRows][nCol];
}
}
for (size_t nRow = 0; nRow < 4; nRow++)
{
for (size_t nCol = 0; nCol < 4; nCol++)
{
if (chNextShape[nRow][nCol] == 1)
{
if (g_chBackground[nRow + g_nRow][nCol + g_nCol] == 1)
{
return 0; //不能移動
}
}
}
}
return 1;
}
void Rotate()
{
g_nRotate = (g_nRotate + 1) % 4;
int nPoolRows = g_nShape * 16 + g_nRotate*4;
for (size_t nRow = 0; nRow < 4; nRow++)
{
for (size_t nCol = 0; nCol < 4; nCol++)
{
g_chBrick[nRow][nCol] = g_chBrickPool[nRow + nPoolRows][nCol];
}
}
}
int IsCanMove(int nToRow, int nToCol)
{
for (size_t nRow = 0; nRow < 4; nRow++)
{
for (size_t nCol = 0; nCol < 4; nCol++)
{
if (g_chBrick[nRow][nCol] == 1)
{
if (g_chBackground[nRow + nToRow][nCol + nToCol] == 1)
{
return 0; //不能移動
}
}
}
}
return 1;
}
void GetNewBrick()
{
srand((unsigned)time(NULL));
g_nRow = 0;
g_nCol = GAME_COLS / 2 - 1;
int nShapeCount = sizeof(g_chBrickPool) / sizeof(g_chBrickPool[0]) /16;
g_nShape = rand() % nShapeCount;
g_nRotate = rand() % 4;
int nPoolRows = g_nShape * 16 + g_nRotate * 4;
for (size_t nRow = 0; nRow < 4; nRow++)
{
for (size_t nCol = 0; nCol < 4; nCol++)
{
g_chBrick[nRow][nCol] = g_chBrickPool[nRow+nPoolRows][nCol];
}
}
}
void DetachBgBrick()
{
for (size_t nRow = 0; nRow < 4; nRow++)
{
for (size_t nCol = 0; nCol < 4; nCol++)
{
if (g_chBrick[nRow][nCol] == 1)
{
g_chBackground[nRow + g_nRow][nCol + g_nCol] = 0;
}
}
}
}
void CombineBgBrick()
{//組合塊
for (size_t nRow = 0; nRow < 4; nRow++)
{
for (size_t nCol = 0; nCol < 4; nCol++)
{
if (g_chBrick[nRow][nCol] == 1)
{
g_chBackground[nRow+g_nRow][nCol+g_nCol] = 1;
}
}
}
}
void InitBackground()
{//初始化背景
for (size_t nRow = 0; nRow < GAME_ROWS; nRow++)
{
for (size_t nCol = 0; nCol < GAME_COLS; nCol++)
{
if (nRow == GAME_ROWS - 1
|| nCol == 0
|| nCol == GAME_COLS - 1)
{
g_chBackground[nRow][nCol] = 1;
}
else
{
g_chBackground[nRow][nCol] = 0;
}
}
}
}
model.h
#pragma once #define GAME_ROWS 20 #define GAME_COLS 12 extern char g_chBackground[GAME_ROWS][GAME_COLS]; extern char g_chBrick[4][4]; extern int g_nRow; extern int g_nCol; void InitBackground(); void GetNewBrick(); void CombineBgBrick(); void DetachBgBrick(); int IsCanMove(int nToRow, int nToCol); void Rotate(); int IsCanRotate();
tetris.cpp
#include "stdafx.h"
#include <stdlib.h>
#include <conio.h>
#include <time.h>
#include "model.h"
#include "view.h"
int main(int argc, char* argv[])
{
InitBackground();
GetNewBrick();
CombineBgBrick();
ShowBackground();
DetachBgBrick();
char chInput = 0;
clock_t clkStart = clock();
clock_t clkEnd = clock();
while (1)
{
clkEnd = clock();
if (clkEnd - clkStart > 1000)
{
clkStart = clkEnd;
OnDown();
}
if (_kbhit() != 0)
{
chInput = _getch();
}
switch (chInput)
{
case 'a':
OnLeft();
break;
case 'w':
OnUp();
break;
case 's':
OnDown();
break;
case 'd':
OnRight();
break;
default:
break;
}
chInput = 0;
}
return 0;
}
更多關(guān)于俄羅斯方塊的文章,請點(diǎn)擊查看專題:《俄羅斯方塊》
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
深入探究C++中的容器適配器與仿函數(shù)技術(shù)
C++中的容器適配器和仿函數(shù)是實(shí)現(xiàn)數(shù)據(jù)結(jié)構(gòu)與算法的重要技術(shù),容器適配器可以將一個容器轉(zhuǎn)換為另一個形式,仿函數(shù)則可以自定義數(shù)據(jù)類型的比較、排序、計(jì)算等行為,提高程序的靈活性和可重用性2023-04-04
Qt設(shè)置窗體(QWidget)透明度的方法總結(jié)
在Qt開發(fā)中,有的時候需要為窗體設(shè)置透明度。這篇文章主要為大家介紹幾個Qt中窗體設(shè)置透明度的方法,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-11-11
探究C++中指針與數(shù)組運(yùn)算符優(yōu)先級
C++中與指針和數(shù)組相關(guān)的運(yùn)算符優(yōu)先級,通過實(shí)際代碼示例解釋了運(yùn)算符的左結(jié)合與右結(jié)合方式,以及如何使用圓括號()來改變默認(rèn)的結(jié)合順序,文章還提供了一個優(yōu)先級表,列出了運(yùn)算符的優(yōu)先級和結(jié)合性,幫助讀者更好地理解復(fù)雜表達(dá)式中運(yùn)算符的調(diào)用順序2024-10-10
MySQL的內(nèi)存表的基礎(chǔ)學(xué)習(xí)教程
這篇文章主要介紹了MySQL的內(nèi)存表的基礎(chǔ)學(xué)習(xí)教程,包括內(nèi)存表的創(chuàng)建以及使用限制等等,需要的朋友可以參考下2015-12-12

