【C++】做一个飞机空战小游戏(五)——getch()控制两个飞机图标移动(控制光标位置)

news2024/9/22 7:06:39

  [导读]本系列博文内容链接如下:

【C++】做一个飞机空战小游戏(一)——使用getch()函数获得键盘码值

【C++】做一个飞机空战小游戏(二)——利用getch()函数实现键盘控制单个字符移动
【C++】做一个飞机空战小游戏(三)——getch()函数控制任意造型飞机图标移动

【C++】做一个飞机空战小游戏(四)——给游戏添加背景音乐(多线程技巧应用)

【C++】做一个飞机空战小游戏(五)——getch()控制两个飞机图标移动(控制光标位置)

​ 

目录

一、控制光标的位置

(二)为什么要控制光标位置

1、避免控制多个图标移动相互影响

2、消除光标的闪烁现象

(二)如何控制光标的输出位置

二、控制两个飞机图标移动程序

(一)头文件control_plane.h

1、更新键盘指令枚举内容

2、增加一个飞机的结构体类型坐标

3、更新飞机坐标计算函数声明

(二)控制函数库control_plane.cpp

1、增加光标控制函数

2、更新显示飞机图标函数

3、初始化函数

4、更新获取键盘指令函数

5、更新了飞机获取指令后位置计算函数的定义

(二)最终程序

1、主函数

2、头文件control_plane.h

3、库函数control_plane.cpp

三、运行效果


飞机空战游戏,一般都是两个飞机,今天介绍一下如何用getch()通过不同的方向键组控制两个飞机移动。

一、控制光标的位置

(二)为什么要控制光标位置

1、避免控制多个图标移动相互影响

本例中本节之前的代码对飞机位置的控制是通过cout函数输出空格和换行来确定飞机图标x和y坐标,这种方式对于游戏中只有一个被控制图标是可行的,但是要控制多个图标移动就行不通了,因为输出换行和空格时会对另外的图标产生影响。

2、消除光标的闪烁现象

输出换行和空格时,光标会不断移动,所以在移动飞机图标时会有光标闪烁的情况,游戏的体验感不好。

而通过控制光标把光标直接移动到要显示飞机图标的位置,就避免了频繁输出换行和空格的操作,不会对屏幕上其他的图标产生影响,也消除了光标闪烁的情况。

(二)如何控制光标的输出位置

对光标位置的控制,本文参考了以下博文

C++移动输出端的光标代码实现_c++光标_我行我素,向往自由的博客-CSDN博客

代码如下所示,需要引入头文件windows.h,并自定义一个gotoxy(x,y)函数,x,y就是光标输出的位置。

#include<windows.h>
void gotoxy(int x, int y) {
	COORD pos = { x,y };
	HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);//获取标准输出设备句柄
	SetConsoleCursorPosition(hOut, pos);//两个参数分别指定哪个窗口,具体位置
}

二、控制两个飞机图标移动程序

要想实现对两架飞机的控制,需要对代码进行部分更新,更新内容包括以下几方面。

(一)头文件control_plane.h

1、更新键盘指令枚举内容

增加一组控制飞机移动的指令,更新后的内容如下:

//定义移动方向命令枚举类型 
typedef  enum {up_cmd,down_cmd,left_cmd,right_cmd,up_cmd2,down_cmd2,left_cmd2,right_cmd2} direction_cmd;

"up_cmd2,down_cmd2,left_cmd2,right_cmd2"四项为新增加的内容。

2、增加一个飞机的结构体类型坐标

extern location plocation,plocation2;			//声明第二架飞机坐标 

3、更新飞机坐标计算函数声明

因为有两架飞机,需要计算两组坐标,所以在函数声明中增加了标识飞机序号的参量int n,具体代码如下:

//计算出接收指令后的飞机坐标 
location plane_locate(int n,location plct,direction_cmd dircmd);

(二)控制函数库control_plane.cpp

1、增加光标控制函数

void gotoxy(int x, int y) {
	COORD pos = { x,y };
	HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);//获取标准输出设备句柄
	SetConsoleCursorPosition(hOut, pos);//两个参数分别指定哪个窗口,具体位置
}

2、更新显示飞机图标函数

有了控制光标函数之后,就可以把显示飞机图标的函数进行调整了,具体代码如下所示。

//飞机图标刷新函数 
void show_plane(location plct)		//预先定义字符定位显示函数,x是列坐标,y是行坐标,原点(x=0,y=0)位于屏幕左上角 
{
	int x,y;
	int i,j;	
	int rows=sizeof(icon_plane)/sizeof(icon_plane[0]);
	x=plct.x;
	y=plct.y;
	
	for(i=0;i<rows;i++)				//图标每行前输出x个空格 
	{
	    gotoxy(x,y+i);              //将光标移动到(x,y+i)位置
		cout<<icon_plane[i]<<endl;  //按行输出飞机图标
	}
}

3、初始化函数

初始化函数中增加另外一架飞机的初始位置,具体代码如下所示。

//初始化函数 
void init(void)
{
	plocation.x=2*r_b/3;				//初始化飞机1图标的x坐标为屏幕横轴最大值的2/3 
	plocation.y=b_b;					//初始化飞机1图标的y坐标为屏幕纵轴最大值
	plocation2.x=r_b/3;					//初始化飞机2图标的x坐标为屏幕横轴最大值的1/3 
	plocation2.y=b_b;					//初始化飞机2图标的y坐标为屏幕纵轴最大值
}

4、更新获取键盘指令函数

因为两架飞机,需要有两组控制飞机移动方向的控制键组合,第一组为双码按键“↑、↓、←、→”四个键,第二组为单码键'w'、's'、'a'、'd'四个键。所以函数中增加了第二组键的返回值。

//获取键盘指令函数 
direction_cmd key(void)
{
	int key_value1,key_value2;			//声明两个变量,存放键值 
	key_value1=getch();					//先获取第一个码值	
	if(key_value1==224)					//如果第一个码值为224,则进行第二个码值的判断 
		{
			key_value2=getch();			//先获取第二个码值
			
			switch(key_value2)
			{
				case 72:				//向上方向键 
					return up_cmd;
				case 80:				//向下方向键
					return down_cmd;
				case 75:				//向左方向键
					return left_cmd;
				case 77:				//向右方向键
					return right_cmd;
			}
			
		}
		else
		{
			switch(key_value1)
			{
				case 119:				//向上方向键 
					return up_cmd2;
				case 115:				//向下方向键
					return down_cmd2;
				case 97:				//向左方向键
					return left_cmd2;
				case 100:				//向右方向键
					return right_cmd2;
			}
		}		
}

5、更新了飞机获取指令后位置计算函数的定义

参量n代表飞机的序号,不同的飞机响应不同的指令dircmd,然后再根据不同的指令分别计算两架飞机位置。

//计算获得移动指令后飞机的坐标 
location plane_locate(int n,location plct,direction_cmd dircmd)
{
 	int x,y;
 	x=plct.x;
 	y=plct.y;
 	
 	if(n==1)
 	{
		switch(dircmd)
		{
			case up_cmd:
				y--;				//字符上移一行,行值y减1
				if(y<t_b)			//限定y值最小值为0
				{
					y=t_b;
				}
				break;
			case down_cmd:
				y++;				//字符下移一行,行值y加1
				if(y>b_b)			//限定y高度 
				{
					y=b_b;
				}
				break;
			case left_cmd:
				x--;				//字符左移一列,列值x减1
				if(x<l_b)
				{
					x=l_b;			//限定x最小值为0; 
				}
				break;
			case right_cmd:
				x++;				//字符右移一列,列值x加1
				if(x>r_b)
				{
					x=r_b;			//限定x宽度
				}
				break;
				
			}
	}
	else if(n==2)
	{
		switch(dircmd)
		{
			case up_cmd2:
				y--;				//字符上移一行,行值y减1
				if(y<t_b)			//限定y值最小值为0
				{
					y=t_b;
				}
				break;
			case down_cmd2:
				y++;				//字符下移一行,行值y加1
				if(y>b_b)			//限定y高度 
				{
					y=b_b;
				}
				break;
			case left_cmd2:
				x--;				//字符左移一列,列值x减1
				if(x<l_b)
				{
					x=l_b;			//限定x最小值为0; 
				}
				break;
			case right_cmd2:
				x++;				//字符右移一列,列值x加1
				if(x>r_b)
				{
					x=r_b;			//限定x宽度
				}
				break;
				
		}
	}
		plct.x=x;
		plct.y=y;
		return plct;
 }

(二)最终程序

1、主函数

#include "control_plane.h"
using namespace std;

location plocation;
location plocation2;

int main(int argc, char** argv) {	
	init();	//初始化					 

	bgmusic();//播放背景音乐
	
	while(1)					//循环等待键盘指令 
	{
		system("cls");			//清屏
		show_plane(plocation);	//刷新飞机1图标
		show_plane(plocation2);	//刷新飞机2图标 		
		dir_cmd=key();			//获取按键指令 
		
		//计算收到键盘指令后的两架飞机坐标 
		plocation=plane_locate(1,plocation,dir_cmd);
		plocation2=plane_locate(2,plocation2,dir_cmd);	
	}
	return 0; 	
}

2、头文件control_plane.h

#ifndef CONTROL_PLANE_H
#define CONTROL_PLANE
#include <iostream>
#include <string>
#include<stdlib.h>
#include<windows.h>
#include <pthread.h>//导入线程头文件库
#include <mmsystem.h> //导入声音头文件库
#pragma comment(lib,"winmm.lib")//导入声音的链接库
#define _CRT_SECURE_NO_WARNINGS 
using namespace std;

#define t_b 0  	//图形显示区域上侧边界 
#define l_b 0	//图形显示区域左侧边界
#define r_b 100	//图形显示区域右侧边界
#define b_b 20	//图形显示区域下侧边界


//定义飞机造型 
const string icon_plane[]={"    ■   ","■  ■  ■","■■■■■","■  ■  ■","    ■   ","  ■■■ "};

//定义图标坐标结构体 
typedef struct{
	int x;
	int y;
} location;

//定义移动方向命令枚举类型 
typedef  enum {up_cmd,down_cmd,left_cmd,right_cmd,up_cmd2,down_cmd2,left_cmd2,right_cmd2} direction_cmd;


extern location plocation,plocation2;			//声明飞机坐标 
static direction_cmd dir_cmd; 		//声明存放按键码值的两个变量

//声明刷新飞机位置函数 
void show_plane(location plct);

//获取键盘指令 
direction_cmd key(void);

//计算出接收指令后的飞机坐标 
location plane_locate(int n,location plct,direction_cmd dircmd);

//计算出接收指令后的飞机坐标 
location plane_locate2(location plct,direction_cmd dircmd);

void init(void);
 
void* thread_bgmusic(void* arg);
void play_bgmusic();
void bgmusic();
 
#endif

3、库函数control_plane.cpp

#include <iostream>
#include "conio.h"
#include <string>
#include "control_plane.h"
#include<windows.h>
using namespace std;

//初始化函数 
void init(void)
{
	plocation.x=2*r_b/3;				//初始化飞机图标的x坐标为屏幕横轴最大值的一半 
	plocation.y=b_b;					//初始化飞机图标的y坐标为屏幕纵轴最大值
	plocation2.x=r_b/3;					//初始化飞机图标的x坐标为屏幕横轴最大值的一半 
	plocation2.y=b_b;					//初始化飞机图标的y坐标为屏幕纵轴最大值
}

//********************************************************************************
 //以下三个函数为播放背景音乐功能 
//********************************************************************************
 
 //播放一遍背景音乐 
 void play_bgmusic() {  

	mciSendString(TEXT("open hero.mp3 alias s1"),NULL,0,NULL);
	mciSendString(TEXT("play s1"),NULL,0,NULL);
 
	Sleep(153*1000);//153*1000意思是153秒,是整首音乐的时长 
	mciSendString(TEXT("close S1"),NULL,0,NULL); 
  
}

//循环播放音乐线程函数 
void* thread_bgmusic(void* arg) //
{ 
  while(1)
  {  	
  	play_bgmusic();
  }
} 

//创建音乐播放线程,开始循环播放音乐 
void bgmusic()
{
	pthread_t tid; 
  	pthread_create(&tid, NULL, thread_bgmusic, NULL);
}


void gotoxy(int x, int y) {
	COORD pos = { x,y };
	HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);//获取标准输出设备句柄
	SetConsoleCursorPosition(hOut, pos);//两个参数分别指定哪个窗口,具体位置
}


//飞机图标刷新函数 
void show_plane(location plct)		//预先定义字符定位显示函数,x是列坐标,y是行坐标,原点(x=0,y=0)位于屏幕左上角 
{
	int x,y;
	int i,j;	
	int rows=sizeof(icon_plane)/sizeof(icon_plane[0]);
	x=plct.x;
	y=plct.y;
	
	for(i=0;i<rows;i++)				//图标每行前输出x个空格 
	{
		gotoxy(x,y+i);
		cout<<icon_plane[i]<<endl;
	}
}


//获取键盘指令函数 
direction_cmd key(void)
{
	int key_value1,key_value2;			//声明两个变量,存放键值 
	key_value1=getch();					//先获取第一个码值 
	
	
	if(key_value1==224)					//如果第一个码值为224,则进行第二个码值的判断 
		{
			key_value2=getch();			//先获取第二个码值
			
			switch(key_value2)
			{
				case 72:				//向上方向键 
					return up_cmd;
				case 80:				//向下方向键
					return down_cmd;
				case 75:				//向左方向键
					return left_cmd;
				case 77:				//向右方向键
					return right_cmd;
			}
			
		}
		else
		{
			switch(key_value1)
			{
				case 119:				//向上方向键 
					return up_cmd2;
				case 115:				//向下方向键
					return down_cmd2;
				case 97:				//向左方向键
					return left_cmd2;
				case 100:				//向右方向键
					return right_cmd2;
			}
		}		
}


//计算获得移动指令后飞机的坐标 
location plane_locate(int n,location plct,direction_cmd dircmd)
{
 	int x,y;
 	x=plct.x;
 	y=plct.y;
 	
 	if(n==1)
 	{
		switch(dircmd)
		{
			case up_cmd:
				y--;				//字符上移一行,行值y减1
				if(y<t_b)			//限定y值最小值为0
				{
					y=t_b;
				}
				break;
			case down_cmd:
				y++;				//字符下移一行,行值y加1
				if(y>b_b)			//限定y高度 
				{
					y=b_b;
				}
				break;
			case left_cmd:
				x--;				//字符左移一列,列值x减1
				if(x<l_b)
				{
					x=l_b;			//限定x最小值为0; 
				}
				break;
			case right_cmd:
				x++;				//字符右移一列,列值x加1
				if(x>r_b)
				{
					x=r_b;			//限定x宽度
				}
				break;
				
			}
	}
	else if(n==2)
	{
		switch(dircmd)
		{
			case up_cmd2:
				y--;				//字符上移一行,行值y减1
				if(y<t_b)			//限定y值最小值为0
				{
					y=t_b;
				}
				break;
			case down_cmd2:
				y++;				//字符下移一行,行值y加1
				if(y>b_b)			//限定y高度 
				{
					y=b_b;
				}
				break;
			case left_cmd2:
				x--;				//字符左移一列,列值x减1
				if(x<l_b)
				{
					x=l_b;			//限定x最小值为0; 
				}
				break;
			case right_cmd2:
				x++;				//字符右移一列,列值x加1
				if(x>r_b)
				{
					x=r_b;			//限定x宽度
				}
				break;
				
		}
	}
		plct.x=x;
		plct.y=y;
		return plct;
 }
 
 //计算获得移动指令后飞机的坐标 
location plane_locate2(location plct,direction_cmd dircmd)
{
 	int x,y;
 	x=plct.x;
 	y=plct.y;
	 switch(dircmd)
		{
			case up_cmd2:
				y--;				//字符上移一行,行值y减1
				if(y<t_b)			//限定y值最小值为0
				{
					y=t_b;
				}
				break;
			case down_cmd2:
				y++;				//字符下移一行,行值y加1
				if(y>b_b)			//限定y高度 
				{
					y=b_b;
				}
				break;
			case left_cmd2:
				x--;				//字符左移一列,列值x减1
				if(x<l_b)
				{
					x=l_b;			//限定x最小值为0; 
				}
				break;
			case right_cmd2:
				x++;				//字符右移一列,列值x加1
				if(x>r_b)
				{
					x=r_b;			//限定x宽度
				}
				break;
				
		}
		plct.x=x;
		plct.y=y;
		return plct;
 }
 
 
 

三、运行效果

(未完待续) 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/840184.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ffmpeg-ffplay代码架构简述

全局变量 /* Minimum SDL audio buffer size, in samples. */ // 最小音频缓冲 #define SDL_AUDIO_MIN_BUFFER_SIZE 512 /* Calculate actual buffer size keeping in mind not cause too frequent audio callbacks */ // 计算实际音频缓冲大小&#xff0c;并不需要太频繁…

c语言基础知识帮助理解(详解数组)

前面梳理完函数和递归的知识后&#xff0c;来进行数组知识的梳理 对函数有疑惑的同学&#xff0c;可以看我之前的文章&#xff1a;c语言基础知识帮助理解&#xff08;详解函数&#xff09;_总之就是非常唔姆的博客-CSDN博客 c语言基础知识帮助理解&#xff08;函数递归详解&am…

类的6个默认成员函数 构造函数

类的6个默认成员函数 如果一个类中什么成员都没有&#xff0c;简称为空类。 空类中真的什么都没有吗&#xff1f;并不是&#xff0c;任何类在什么都不写时&#xff0c;编译器会自动生成以下6个默认成员函数。 默认成员函数&#xff1a;用户没有显式实现&#xff0c;编译器会生…

ruoyi-cloud-notes01

1、Maven中的dependencyManagement Maven中的dependencyManagement元素提供了一种管理依赖版本号的方式。在dependencyManagement元素中声明所依赖的jar包的版本号等信息&#xff0c;那么所有子项目再次引入此依赖jar包时则无需显式的列出版本号。Maven会沿着父子层级向上寻找…

HCIP MPLS总结

一、MPLS--多协议标签交换 多协议&#xff1a;可以基于多种不同的3层协议来生成2.5层的标签信息&#xff1b; 包交换&#xff1a;包为网络层的PDU&#xff0c;故包交换是基于IP地址进行数据转发&#xff1b;就是路由器的路由行为&#xff1b; 原始的包交换&#xff1a;数据包…

STM32 CubeMX USB_CDC(USB_转串口)

STM32 CubeMX STM32 CubeMX 定时器&#xff08;普通模式和PWM模式&#xff09; STM32 CubeMX一、STM32 CubeMX 设置USB时钟设置USB使能UBS功能选择 二、代码部分添加代码实验效果 ![请添加图片描述](https://img-blog.csdnimg.cn/a7333bba478441ab950a66fc63f204fb.png)printf发…

分库分表概念、原理、拆分策略和实现技术讲解

文章目录 1.什么是分库分表2.分库分表拆分策略2.1 垂直拆分2.2 水平拆分 3.分库分表实现技术简介 1.什么是分库分表 分库分表的中心思想就是将数据分散存储&#xff0c;使得单一数据库/表的数据量变小来缓解单一数据库的性能问题&#xff0c;从而达到提升数据库性能的目的。 …

python中使用yt-dlp模块实现带进程条下载音视频

当代的互联网时代&#xff0c;视频内容的流行无疑是其中的重要组成部分。作为全球最大的视频分享平台&#xff0c;每天吸引着数以亿计的用户观看各种各样的视频内容。有时候&#xff0c;我们可能希望将某些喜欢的视频保存到本地进行观看&#xff0c;或者将它们用于其他用途。在…

406 · 和大于S的最小子数组

链接&#xff1a;LintCode 炼码 - ChatGPT&#xff01;更高效的学习体验&#xff01; 题解&#xff1a;同向双指针 九章算法 - 帮助更多程序员找到好工作&#xff0c;硅谷顶尖IT企业工程师实时在线授课为你传授面试技巧 class Solution { public:/*** param nums: an array …

任务12、Quality指令加持,Midjourney生成电影级数码作品

12.1 任务概述 本次实验任务旨在帮助你掌握Midjourney AI绘画中的Quality指令。通过深入介绍Quality指令的概念和作用,我们将解释为什么它在绘画中至关重要。通过测试不同的Quality参数对绘画效果的影响,并提供实战演示,你将学会如何在Midjourney中设置Quality参数以达到更…

Spring 事务详解(注解方式)

目 录 序言 1、编程式事务 2、配置声明式事务 2.1 基于TransactionProxyFactoryBean的方式&#xff08;不常用&#xff0c;因为要为每一个类配置TransactionProxyFactoryBean&#xff09; 2.2 基于AspectJ的XML方式&#xff08;常用&#xff0c;可配置在某些类下的所有子…

⛳ StringBuffer and StringBuilder 处理字符串

目录 ⛳ StringBuffer and StringBuilder 处理字符串&#x1f3a8; 一&#xff0c;简介&#x1f3ed; 二&#xff0c;常用方法&#x1f69c; 三 &#xff0c;StringBugger&#x1f43e; 四&#xff0c;StringBuilder⭐ 五&#xff0c;StringBuffer和StringBuilder面试 ⛳ Strin…

【李宏毅机器学习·学习笔记】Tips for Training: Adaptive Learning Rate

本节课主要介绍了Adaptive Learning Rate的基本思想和方法。通过使用Adaptive Learning Rate的策略&#xff0c;在训练深度神经网络时程序能实现在不同参数、不同iteration中&#xff0c;学习率不同。 本节课涉及到的算法或策略有&#xff1a;Adgrad、RMSProp、Adam、Learning …

Qt应用开发(基础篇)——时间微调输入框QDateTimeEdit、QDateEdit、QTimeEdit

一、前言 QAbstractSpinBox是全部微调输入框的父类&#xff0c;这是一种允许用户通过点击上下箭头按钮或输入数字来调整数值的图形用户界面控件&#xff0c;父类提供了当前值text、对齐方式align、只读readOnly等通用属性和方法。在上一篇数值微调输入框中有详细介绍。 QDateTi…

【雕爷学编程】Arduino动手做(186)---WeMos ESP32开发板4

37款传感器与模块的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&#x…

PyTorch深度学习实战(9)——学习率优化

PyTorch深度学习实战&#xff08;9&#xff09;——学习率优化 0. 前言1. 学习率简介2. 梯度值、学习率和权重之间的相互作用3. 学习率优化实战3.1 学习率对缩放后的数据集的影响3.2 学习率对未缩放数据集的影响 小结系列链接 0. 前言 学习率( learning rate )是神经网络训练中…

(十二)大数据实战——hadoop集群之HDFS高可用自动故障转移

前言 本节内容主要介绍一下hadoop集群下实现HDFS高可用的自动故障转移&#xff0c;HDFS高可用的自动故障转移主要通过zookeeper实现故障的监控和主节点的切换。自动故障转移为 HDFS 部署增加了两个新组件&#xff1a;ZooKeeper 和 ZKFailoverController &#xff08;ZKFC&…

【笔记】湖仓一体架构演进与发展

https://www.bilibili.com/video/BV1oF411F7rQ/?spm_id_from333.788.recommend_more_video.0&vd_sourcefa36a95b3c3fa4f32dd400f8cabddeaf

Linux中的firewall-cmd

2023年8月4日&#xff0c;周五上午 目录 打开端口关闭端口查看某个端口是否打开查看当前防火墙设置firewall-cmd中的服务在防火墙中什么是服务&#xff1f;为什么会有服务&#xff1f;打开或关闭服务查看某个服务是否打开firewall-cmd中的 zones查看所有可用的zones&#xff0…

elementui Cascader 级联选择使用心得

相信大家对于elementui并不陌生&#xff0c;作为适配Vue的优秀UI框架之一&#xff0c;一直被所有的开发者痛并快乐着。今天要记录的就是里边的主角之一Cascader。 首先先介绍一下Cascader ---> 当一个数据集合有清晰的层级结构时&#xff0c;可通过级联选择器逐级查看并选择…