当前位置: 首页 > article >正文

项目实战:基于Linux的Flappy bird游戏开发

一、项目介绍

项目总结

        1.按下空格键小鸟上升,不按小鸟下落

        2.搭建小鸟需要穿过的管道

        3.管道自动左移和创建

        4.小鸟撞到管道游戏结束

知识储备

        1.C语言

        2.数据结构-链表

        3.Ncurses库

        4.信号机制

二、Ncurses库介绍

         Ncurses是最早的System V Release 4.0 (SVr4)中 curses的一个克隆和升级。这是一个可自由配置的库,完全兼容旧版本curses。

        Ncurses构成了一个工作在底层终端代码之上的封装,并向用户提供了一个灵活高效的API(Application Programming Interface 应用程序接口)。它提供了创建窗口界面,移动光标,产生颜色,处理键盘按键等功能。使程序员编写应用程序不需要关心那些底层的终端操作。

        简而言之,它是一个管理应用程序在字符终端显示的函数库。

Ncurses库函数

注:安装命令:sudo apt-get install libncurses5-dev
       为了能够使用Ncurses库,必须在源程序中将#include<curses.h>包括进来,而且在编译的需要与它链接起来. 
       在gcc中可以使用参数-lncurses进行编译.

1.  initscr(void);

    是curses模式的入口。将终端屏幕初始化为curses模式,为当前屏幕和相关的数据结构分配内存。

2.  int  endwin(void); 

    是curses模式的出口,退出curses模式,释放curses子系统和相关数据结构占用的内存。

3.  int curs_set(int visibility); 

    设置光标是否可见,visibility:0(不可见),1(可见)  

4.  int move(int  new_y, int  new_x);

    将光标移动到new_y所指定的行和new_x所指定的列

5.  int addch(const  chtype  char); 

    在当前光标位置添加字符

6. int  refresh(void); 

    刷新物理屏幕。将获取的内容显示到显示器上。

7.  int  keypad(WINDOW  *window_ptr,  bool  key_on); 

    允许使用功能键。exp:keypad(stdscr,1);//允许使用功能按键

8.  int getch(void); 

    读取键盘输入的一个字符

9. chtype inch(void); 

    获取当前光标位置的字符。

    注:curses有自己的字符类型chtype,使用时强制类型转换为char

10. int start_color(void); 

    启动color机制,初始化当前终端支持的所有颜色

11. int init_pair(short  pair_number,  short  foreground,  short  background);

    配置颜色对        
    COLOR_BLACK         黑色        COLOR_MAGENTA      品红色
    COLOR_RED             红色        COLOR_CYAN              青色
    COLOR_GREEN        绿色        COLOR_WHITE            白色
    COLOR_YELLOW      黄色        COLOR_BLUE              蓝色

12. int  COLOR_PAIR(int  pair_number); 

    设置颜色属性,设置完颜色对,可以通过COLOR_PAIR实现

13. int  attron(chtype  attribute); 

    启用属性设置

14. int  attroff(chtype  attribute); 

    关闭属性设置

Ncurses库安装

        安装命令:sudo apt-get install libncurses5-dev

        注:为了能够使用Ncurses库,必须在源程序中将#include<curses.h>包括进来,而且在编译的需要与它链接起来. 在gcc中可以使用参数-lncurses进行编译。

编译gcc surses.c -o  curses -lncurses

三、信号

        在Linux中,软中断信号(signal,简称为信号)是在软件层次上对中断的一种模拟,用来通

知进程发生了异步事件。内核可以因为内部事件而给进程发送信号,通知进程发生了某个事件。

信号响应的方式:

        1.忽略信号,即对信号不做任何处理;

        2.捕捉信号,即信号发生时执行用户自定义的信号处理函数。

        3.执行缺省操作,Linux对每种信号都规定了默认操作。

设置信号相应方式-signal

#include  <unistd.h>
#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
        成功时返回原先的信号处理函数,失败时返回SIG_ERR
        signum:指明了所要处理的信号类型
        handler:描述了与信号关联的动作
        SIG_DFL代表缺省方式; SIG_IGN 代表忽略信号;
        指定的信号处理函数代表捕捉方式
signal函数-示例
// 头文件省略
void handler (int signo) {
        printf(“HELLO!\n”);
}
int  main() {
        signal(SIGINT, handler);
        while ( 1 ) ;
        return 0;
}
设置定时器
struct itimerval {
    struct timeval it_interval; /* 计时器重新启动的间歇值 */
    struct timeval it_value;    /* 计时器安装后首次启动的初
 };                               始值,之后就没有用 */
struct timeval {
    long tv_sec;       /* 秒 */
    long tv_usec;      /* 微妙*/
};

计时器的实现

int setitimer(int which, const struct itimerval *value,struct itimerval *ovalue)

参数:

which:间歇计时器类型,

ITIMER_REAL      //数值为0,发送的信号是SIGALRM。

struct itimerval *value:将value指向的结构体设为计时器的当前值,
struct itimerval *ovalue:保存计时器原有值。一般设置为NULL。

返回值: 成功返回0。失败返回-1。

四、项目功能实现

项目安排

阶段1:初始化工作,小鸟功能实现

阶段2:管道功能实现

        1.创建链表

        2.显示管道

        3.清除管道

        4.移动管道

阶段3:完善代码,进行项目总结

        1.判断游戏结束:小鸟与管道碰到

        2.循环创建管道

        3.为管道和小鸟添加色彩

代码功能

#include <stdio.h>
#include <curses.h>
#include <signal.h>
#include <sys/time.h>
#include <stdlib.h>
#define BIRD '@'
#define BLANK ' '
#define PIPE '+'
/*定义关于管道的结构体*/
typedef struct Pipe{
	int x;//列坐标
	int y;//横坐标
	struct Pipe *next;
}Pipe_node, *Pipe_list;
 
Pipe_list head, tail;
 
void creat_list();//创建链表
void show_pipe();//显示管道
void clear_pipe();//清除管道
void move_pipe();//移动管道
 
int bird_y, bird_x;//小鸟坐标
 
void show_bird();//显示小鸟
void clear_bird();//清除小鸟
void move_bird();//移动小鸟
 
void init_curses();//curses库初始化
int set_timer(int ms_t);//设置定时器--ms
void handler(int sig);//信号处理函数
 
int main(int argc, const char *argv[])
{
	bird_y = 15;//行
	bird_x = 10;//列
	init_curses();
	signal(SIGALRM, handler);
	set_timer(500);//500ms
 
	srand(time(0));//随机种子
	creat_list();
	show_pipe();
 
	show_bird();
	move_bird();
 
 
	return 0;
}
void init_curses()//curses库初始化
{
	initscr();//进入curses模式
	curs_set(0);//禁止光标显示
	noecho();//禁止输入字符显示
	keypad(stdscr,1);//启动功能按键
	start_color();//启动颜色机制
	init_pair(1,COLOR_WHITE, COLOR_RED);//小鸟颜色设置
	init_pair(2,COLOR_WHITE, COLOR_GREEN);//管道颜色设置
}
int set_timer(int ms_t)//设置定时器--ms
{
	struct itimerval timer;
	long t_sec,t_usec;
	int ret;
 
	t_sec = ms_t / 1000; //s
	t_usec = (ms_t % 1000) * 1000;//us
 
	timer.it_value.tv_sec = t_sec;
	timer.it_value.tv_usec = t_usec;//首次启动定时值
 
	timer.it_interval.tv_sec = t_sec;
	timer.it_interval.tv_usec = t_usec;//定时时间间隔
 
	ret = setitimer(ITIMER_REAL, &timer, NULL);
	return ret;
 
}
void handler(int sig)
{
	Pipe_list p, new;
	int i,j;
	/*小鸟下落*/
	clear_bird();
	bird_y++;
	show_bird();
	/*游戏结束判断*/
	if((char)inch() == PIPE)
	{
		set_timer(0);
		endwin();
		exit(1);
	}
	p = head->next;
	if(p->x == 0)
	{
		head->next = p->next;
		for(i = p->x; i < p->x+10; i++)
		{
			/*上半部分管道*/
			for(j=0; j<p->y; j++)
			{
				move(j,i);
				addch(BLANK);
			}
			/*下半部分管道创建*/
			for(j = p->y+5; j < 25; j++)
			{
				move(j,i);
				addch(BLANK);
			}
		refresh();
		}
		free(p);
 
		new = (Pipe_list)malloc(sizeof(Pipe_node));
		new->x = tail->x + 20;
		new->y = rand() % 11 + 5;
		new->next = NULL;
		tail->next = new;
		tail = new;
 
 
	}
 
	/*管道移动*/
	clear_pipe();
	move_pipe();
	show_pipe();
}
void show_bird()//显示小鸟
{
	attron(COLOR_PAIR(1));
	move(bird_y,bird_x);
	addch(BIRD);
	refresh();
	attroff(COLOR_PAIR(1));
}
void clear_bird()//清除小鸟
{
	move(bird_y,bird_x);
	addch(BLANK);
	refresh();
}
void move_bird()//移动小鸟
{
	char key;
	while(1)
	{
		key = getch();
		if(key == ' ')
		{
			clear_bird();
			bird_y--;
			show_bird();
			/*游戏结束判断*/
			if((char)inch() == PIPE)
			{
				set_timer(0);
				endwin();
				exit(1);
			}
		}
	}
}
void creat_list()//创建链表
{
	int i;
	Pipe_list p, new;
	head = (Pipe_list)malloc(sizeof(Pipe_node));
	head->next = NULL;
	p = head;
 
	for(i = 0; i < 5; i++)
	{
		new = (Pipe_list)malloc(sizeof(Pipe_node));
		new->x = (i + 1) * 20;
		new->y = rand() % 11 + 5; // (5-15行)
		new->next = NULL;
		p->next = new;
		p = new;
	}
	tail = p;
 
}
void show_pipe()//显示管道
{
	Pipe_list p;
	int i,j;
	p = head->next;
	attron(COLOR_PAIR(2));
	while(p)
	{
		for(i = p->x; i < p->x+10; i++)
		{
			/*上半部分管道*/
			for(j=0; j<p->y; j++)
			{
				move(j,i);
				addch(PIPE);
			}
			/*下半部分管道创建*/
			for(j = p->y+5; j < 25; j++)
			{
				move(j,i);
				addch(PIPE);
			}
		}
		refresh();
		p = p->next;
	}
	attroff(COLOR_PAIR(2));
}
void clear_pipe()//清除管道
{
	Pipe_list p;
	int i,j;
	p = head->next;
	while(p)
	{
		for(i = p->x; i < p->x+10; i++)
		{
			/*上半部分管道*/
			for(j=0; j<p->y; j++)
			{
				move(j,i);
				addch(BLANK);
			}
			/*下半部分管道创建*/
			for(j = p->y+5; j < 25; j++)
			{
				move(j,i);
				addch(BLANK);
			}
		}
		refresh();
		p = p->next;
	}
 
}
void move_pipe()//移动管道
{
	Pipe_list p;
	p = head->next;
	while(p)
	{
		p->x--;
		p = p->next;
	}
 
}


http://www.kler.cn/a/376798.html

相关文章:

  • HTTP 405 Method Not Allowed:解析与解决
  • 设备搜索相关协议使用
  • GitHub | 发布到GitHub仓库并联文件夹的方式
  • 来了解一下!!!——React
  • Java实现动态切换ubuntu壁纸功能
  • Python复习1:
  • 第二届开放原子大赛-开源工业软件算法集成大赛即将启动!
  • 网络基础知识概览
  • 贪心算法习题其三【力扣】【算法学习day.20】
  • angular登录按钮输入框监听
  • Python 定时调度任务
  • nignx代理获取真实地址request.getRequestURL()
  • el-select 的默认选中 以及后端返回的数据进行默认选中
  • Java多态特性的向上转型
  • SD-WAN分布式组网:构建高效、灵活的企业网络架构
  • 协议(OSI-tcp-udp)
  • linux 运行 activemq,Linux 安装 ActiveMQ 服务器详解
  • windows C#-泛型类型
  • Rust 力扣 - 1461. 检查一个字符串是否包含所有长度为 K 的二进制子串
  • 动态SQL在梧桐数据库的使用介绍
  • MySQL锁表快速解决办法
  • ELK的ElasticStack概念
  • 11408 计网===物理层
  • 使用 Redis 作为异步队列:原理、实现及最佳实践
  • 小新学习k8s第六天之pod详解
  • Linux 常用安装软件