全局变量

简而言之,定义在函数外的变量,就是全局变量。
所有的函数都可以引用和修改全局变量(共享)。

全局变量被分配在内存的全局区域,可以调试看地址验证。

基础问题:
可否有两个同名的全局变量(不可以)
全局变量可否在另一个文件定义,这个文件使用(可以,但是要讲究技巧)
全局变量的定义,与普通变量没有区别。
此外,全局变量还可以声明,而不定义,方法是使用extern关键字。
声明,告诉编译器“有这个全局变量,但是不在我这个c文件中”

工程中的使用惯例:
将全局变量定义在.c文件中
将全局变量的声明写在.h文件中

程序的模块化之MVC

程序的模块化有利于代码维护。
MVC是一种模块划分的依据,其实是:Model、View、Control。
Model:模型,即数据
View:UI,即数据如何显示
Control:Control是用于连接Model和View的手段,比如用户操作
我们的俄罗斯方块游戏,将遵循MVC模式开发。也就是我们会创建多个cpp文件,分
工Model和View。
因为我们的俄罗斯方块游戏较为简单,所以就不再单独设立Control。

关于俄罗斯方块的代码实现要点

创建model.cpp、view.cpp及其头文件
记得在头文件中加入:

#pragma once

使用数组表示背景和方块

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;
            }
        }
    }
}

并在view中显示出来。

方块表示及其初始化

我们使用一个4*4的数组,存储方块。
对于生成一个方块的算法:
先使用一个大数组,保存所有的方块形状
需要新方块时,从存储了所有方块形状的大叔组中,随机取一个
因为决定形状和朝向的东西其实就是g_chBrcikPool中的起始行数,所以,定义两个变
量来存储:

int g_nShape = 0; //是长条还是方块,系数为16
int g_nRotate = 0; //朝向,系数为4

可以用sizeof优化代码

int nShapeCount = sizeof(g_chBrickPool) / sizeof(g_chBrickPool[0]) /16;

因为printf输出后,光标会自动后移。所以,如果先输出背景,再输出砖,已经来不及了。
采取的算法逻辑是:
先在内存中合并背景和砖
再输出合并后的结果
再分离背景和砖

要让游戏动起来

因为游戏要长期运行,程序内部一定需要一个长期运行的循环结构。
我们将实现一个循环结构,在循环结构中,接受用户的输入,并改变游戏的行为。

while (1)
    {
        clkEnd = clock();
        if (clkEnd - clkStart > 1000)
        {
            clkStart = clkEnd;
            OnDown();
        }

        if (_kbhit() != 0)
        {
            chTnput = _getch();
        }
        switch (chTnput)
        {
        case 'a':
            OnLeft();
            break;
        case 'w':
            OnUp();
            break;
        case'd':
            OnRight();
            break;
        case 's':
            OnDown();
            break;
        default:
            break;
        }
        chTnput = 0;
    }

    return 0;
}

封装上下左右的响应函数因为需要做碰撞检查等,所以我们封装对应的函数。使得代码结构更清晰。
怎么进行碰撞检测
我们可以先假设方块移动,然后判断方块与背景是否有重合。
如果有重合,则不能移动
如果没有重合,则可以移动
旋转的实现
依据我们取方块的逻辑,所谓的旋转,就是在形状不变的基础上,朝向加1.
是否能够旋转的逻辑,类似于移动,就是先假设旋转,然后看是否与背景重叠。
如何省略掉回车
C标准库中没有方法解决。不过windows中提供了_kbhit()函数,检测键盘是否按下。

int _kbhit( void );

当键盘有按下时,这个函数会返回非0值,并且,按下的内容,可以通过_getch获取。
这样配合,省掉了回车。

方块自动下落

铺垫知识:标准库中,提供了clock函数,可以返回当前程序运行的时间(单位是毫
秒)。
我们准备两个时刻,一个代表当前时间段的开始,一个代表当前时刻。当两个时刻的
差值大于一定时间,则自动下落。

12-15 09:02