0x00512031 处(位于 sandanfeiji.exe 中)引发的异常: 0xC0000005: 写入位置 0x0051C020 时发生访问冲突。

皮猪 2020-11-16 04:27:21
#pragma warning(disable:4996)
#include<stdio.h>
#include<stdlib.h>
#include<conio.h>
#include<windows.h>

#define High 15 //游戏画面尺寸
#define Width 25
#define EnemyNum 5 //敌机的个数

//全局变量
int position_x, position_y; //飞机的位置
int enemy_x[EnemyNum], enemy_y[EnemyNum]; //敌机的位置
int canvas[High][Width] = { 0 }; //二维数组存储游戏画布中对应的元素
//0为空格,1为飞机*,2为子弹|,3为敌机@
int score; //得分
int BulletWidth; //子弹的宽度
int EnemyMoveSpeed; //敌机的移动速度

void HideCursor() //隐藏光标
{
CONSOLE_CURSOR_INFO cursor_info = { 1,0 }; //第二个值为0表示隐藏光标
SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &cursor_info);
}

void gotoxy(int x, int y) //将光标移动到(x,y)位置
{
HANDLE handle = GetStdHandle(STD_OUTPUT_HANDLE);
COORD pos;
pos.X = x;
pos.Y = y;
SetConsoleCursorPosition(handle, pos);
}

void startup() //数据的初始化
{
position_x = High - 1;
position_y = Width / 2;
canvas[position_x][position_y] = 1;
int k;
for (k = 0; k < EnemyNum; k ++ )
{
enemy_x[k] = rand() % 2;
enemy_y[k] = rand() % Width;
canvas[enemy_x[k]][enemy_y[k]] = 3;
}
score = 0;
BulletWidth = 0;
EnemyMoveSpeed = 20;
}

void show() //显示画面
{
gotoxy(0, 0);
int i, j;
for (i = 0; i < High; i++)
{
for (j = 0; j < Width; j++)
{
if (canvas[i][j] == 0)
printf(" ");
else if (canvas[i][j] == 1)
printf("*");
else if (canvas[i][j] == 2)
printf("|");
else if (canvas[i][j] == 3)
printf("@");
}
printf("\n");
}
printf("得分:%d\n", score);
Sleep(20);
}

void updateWidthoutInput() //与用户输入无关的更新
{
int i, j, k;
for (i = 0; i < High; i++)
{
for (j = 0; j < Width; j++)
{
if (canvas[i][j] == 2)
{
for (k = 0; k < EnemyNum; k++)
{
if ((i == enemy_x[k]) && (j == enemy_y[k])) //子弹击中敌机
{
score++;
if (score % 5 == 0 && EnemyMoveSpeed > 3) //达到一定积分后敌机变快
EnemyMoveSpeed--;
if (score % 5 == 0) //达到一定积分后子弹变厉害
BulletWidth++;
canvas[enemy_x[k]][enemy_y[k]] = 0;
enemy_x[k] = rand() % 2; //产生新的飞机
enemy_y[k] = rand() % Width;
canvas[enemy_x[k]][enemy_y[k]] = 3;
canvas[i][j] = 0; //子弹消失
}
}
//子弹向上移动
canvas[i][j] = 0;
if (i > 0)
canvas[i - 1][j] = 2;
}
}
}

static int speed = 0;
if (speed < EnemyMoveSpeed)
speed++;

for (k = 0; k < EnemyNum; k++)
{
if ((position_x == enemy_x[k]) && (position_y == enemy_y[k])) //敌机撞到我机
{
printf("失败!\n");
Sleep(3000);
system("pause");
exit(0);
}

if (enemy_x[k] > High) //敌机跑出显示屏幕
{
canvas[enemy_x[k]][enemy_y[k]] = 0;
enemy_x[k] = rand() % 2; //产生新的飞机
enemy_y[k] = rand() % Width;
canvas[enemy_x[k]][enemy_y[k]] = 3;
score--; //减分
}

if (speed == EnemyMoveSpeed)
{
//敌机下落
for (k = 0; k < EnemyNum; k++)
{
canvas[enemy_x[k]][enemy_y[k]] = 0;
enemy_x[k]++;
speed = 0;
canvas[enemy_x[k]][enemy_y[k]] = 3;
}

}
}
}

void updateWithInput() //与用户输入有关的更新
{
char input;
if (kbhit()) //判断是否有输入
{
input = getch(); //根据用户的不同输入来移动,不必输入回车
if (input == 'a' && position_y > 0)
{
canvas[position_x][position_y] = 0;
position_y--; //位置左移
canvas[position_x][position_y] = 1;
}
else if (input == 'd' && position_y <Width-1)
{
canvas[position_x][position_y] = 0;
position_y++; //位置右移
canvas[position_x][position_y] = 1;
}
else if (input == 'w' )
{
canvas[position_x][position_y] = 0;
position_x--; //位置上移
canvas[position_x][position_y] = 1;
}
else if (input == 'a' && position_y > 0)
{
canvas[position_x][position_y] = 0;
position_x++; //位置下移
canvas[position_x][position_y] = 1;
}
else if (input == ' ') //发射子弹
{
int left = position_y - BulletWidth;
int right = position_y + BulletWidth;
if (left < 0)
left = 0;
if (right > Width - 1)
right = Width - 1;
int k;
for (k = left; k <= right; k ++ ) //发射子弹
canvas[position_x - 1][k] = 2; //发射子弹的初始位置在飞机的正上方
}
}
}

int main()
{
HideCursor();
startup(); //数据的初始化
while (1) //游戏循环执行
{
show(); //显示画面
updateWidthoutInput(); //与用户输入无关的更新
updateWithInput(); //与用户输入有关的更新
}

return 0;
}
...全文
1131 5 打赏 收藏 举报
写回复
5 条回复
切换为时间正序
请发表友善的回复…
发表回复
皮猪 2020-12-10
引用 6 楼 赵4老师的回复:
学会找错即debug的方法比找到本次bug更有用。
嗯嗯,感谢俩位大佬
  • 打赏
  • 举报
回复
皮猪 2020-12-09
哎,还是找到原因了,是for循环里都用了k来判断循环字数,当数组越界时,就没有重新随机了,把其中的一个k换成另外的字母就没问题了
  • 打赏
  • 举报
回复 1
赵4老师 2020-12-09
学会找错即debug的方法比找到本次bug更有用。
  • 打赏
  • 举报
回复
forever74 2020-12-08
数组canvas的第一下标的上限是14,可是你引用的时候出现了26,38,24,36 这就是原因。
  • 打赏
  • 举报
回复 2
赵4老师 2020-12-07
将139行对应的表达式拆成若干条功能等价的简单表达式,必要时还得声明几个临时变量保存中间结果。 然后单步调试这几行简单表达式,观察每一行简单表达式计算时涉及的值,是否如你所愿。
  • 打赏
  • 举报
回复
皮猪 2020-11-16
引用 1 楼 赵4老师 的回复:
崩溃的时候在弹出的对话框按相应按钮进入调试,按Alt+7键查看Call Stack即“调用堆栈”里面从上到下列出的对应从里层到外层的函数调用历史。双击某一行可将光标定位到此次调用的源代码或汇编指令处,看不懂时双击下一行,直到能看懂为止
请问这咋看啊?少的可怜。
  • 打赏
  • 举报
回复
赵4老师 2020-11-16
崩溃的时候在弹出的对话框按相应按钮进入调试,按Alt+7键查看Call Stack即“调用堆栈”里面从上到下列出的对应从里层到外层的函数调用历史。双击某一行可将光标定位到此次调用的源代码或汇编指令处,看不懂时双击下一行,直到能看懂为止
  • 打赏
  • 举报
回复
发帖
新手乐园
加入

3.3w+

社区成员

C/C++ 新手乐园
社区管理员
  • 新手乐园社区
申请成为版主
帖子事件
创建了帖子
2020-11-16 04:27
社区公告
暂无公告