贪吃蛇——Linux环境下,基于Ncurse图形库的c语言的小游戏

  

1.一些关于ncurse的知识

(1)使用ncurses原因:按键响应速度快
(2)一些ncurse函数:
initscr();//ncurse界面初始化函数
printw();//在ncurse模式下的printf
getch();//等待用户输入,如果没有这句话程序就退出了,看不到运行结果
endwin();//程序退出,调用函数来恢复shell终端的显示,如果没有这句话,shell终端会字乱码,坏掉
keypad();//设置了可以从stdscr接收键盘的功能键(快捷键)
//使用keypad(stdtcr,1);
move();//改变光标的位置
//使用move(1,1);
refresh();//刷新
noecho();//不要把功能键的信息打印出来

2.制作贪吃蛇的思路

(1)地图规划

大小20*20
地图竖直方向上的边界"|"
地图水平方向上的边界"--"
贪吃蛇的身子"[]"
食物"##"

(2)蛇的身子的初始化(链表)
(3)蛇向右移动(链表的添加,删除)
(4)蛇碰到边界,重新开始(初始化蛇)
(5)引入线程(方向改变和刷新界面)
(6)蛇四方向的移动(要解决不合理走位)
(7)蛇吃食物(食物的随机,链表的增加)
(8)贪吃蛇的头碰到身子(注意链表的头结点)

3.代码实现:

#include <curses.h>
#include <stdlib.h>
#include <pthread.h>

#define UP 1
#define DOWN -1
#define LEFT 2
#define RIGHT -2

struct Snake{
	int hang;
	int lie;
	struct Snake *next;
};

struct Snake *head = NULL;
struct Snake *tail = NULL;
int key;
int dir;
struct Snake food;

void initFood(){
	int x = rand()%20;
	int y = rand()%20;
	food.hang = x;
	food.lie = y;
}
void initNcurse(){
	initscr();
	keypad(stdscr,1);
	noecho();
}

int hasSnakeNode(int i,int j){
	struct Snake *p = head;
	while(p != NULL){
		if(p->hang == i && p->lie == j){
			return 1;
		}
		p = p->next;
	}
	return 0;
}

int hasFood(int i,int j){
	if(food.hang == i && food.lie == j){
		return 1;
	}
	return 0;
}

void initPic(){
	int hang;
	int lie;
	move(0,0);
	for(hang = 0;hang<20;hang++){
		if(hang == 0){
			for(lie = 0;lie <= 19;lie++){
				printw("--");
			}
			printw("\n");
		}
		if(hang > 0 && hang <= 19){
			for(lie = 0;lie <= 20;lie++){
				if(lie == 0 || lie == 20){
					printw("|");
				}else if(hasSnakeNode(hang,lie)){
					printw("[]");
				}else if(hasFood(hang,lie)){
					printw("##");
				}
				else{
					printw("  ");
				}
			}
			printw("\n");
		}
		if(hang == 19){
			for(lie = 0;lie <= 19;lie++){
				printw("--");
			}
		printw("\n");
		}
	}
	printw("By chenweiqiang,key = %d\n",key);
}

void addNode(){
	struct Snake *new;
	new = (struct Snake*)malloc(sizeof(struct Snake));
	new->next = NULL;
	switch(dir){
		case UP:
			new->hang = tail->hang-1;
			new->lie = tail->lie;
			break;

		case DOWN:
			new->hang = tail->hang+1;
			new->lie = tail->lie;
			break;

		case LEFT:
			new->hang = tail->hang;
			new->lie = tail->lie-1;
			break;

		case RIGHT:
			new->hang = tail->hang;
			new->lie = tail->lie+1;
			break;
	}
	tail->next = new;
	tail = new;
	

}

void initSnake(){
	struct Snake *p = NULL;
	initFood();
	dir = RIGHT;
	while(p != NULL){
		p = head;
		free(p);
		p = p->next;
	}
	head = (struct Snake*)malloc(sizeof(struct Snake));
	head->hang = 2;
	head->lie = 2;
	head->next = NULL;

	tail = head;

	addNode();
	addNode();
	addNode();

}

void deleteNode(){
	struct Snake *p;
	p = head;
	head = head->next;
	free(p);
}
int ifSnakeDie(){
	struct Snake *p = head;
// 	p = head;	
	if(tail->hang == 0 || tail->hang == 20 || tail->lie ==0 || tail->lie ==20){
		return 1;
	}
	while(p->next != NULL){
		if(tail->hang == p->hang && tail->lie == p->lie){
			return 1;
		}
		p = p->next;
	}
	return 0;
}
void moveSnake(){
	
	if(tail->hang == food.hang && tail->lie == food.lie){
		addNode();
		initFood();
	}
	addNode();
	deleteNode();
	
	if(ifSnakeDie()){
		initSnake();
	}
}

void turn(int drection){
	if(abs(dir) != abs(drection)){
		dir = drection;
	}
}
void* changeDir(){
	
	while(1){
		key = getch();
		switch(key){
			case KEY_UP:
				turn(UP);
				break;
			case KEY_DOWN:
				turn(DOWN);
				break;
			case KEY_LEFT:
				turn(LEFT);
				break;
			case KEY_RIGHT:
				turn(RIGHT);
				break;
		}
	}
}

void* refreshSnake(){
	while(1){
		moveSnake();
		initPic();
		refresh();
		usleep(100000);
	}

}

int main(){
	pthread_t t1;
	pthread_t t2;
	
	initNcurse();
	initPic();
	initSnake();	

	pthread_create(&t1,NULL,refreshSnake,NULL);
	pthread_create(&t2,NULL,changeDir,NULL);
	//pthread_create(&tid, NULL, fun, NULL);
	while(1);
	getch();
	endwin();
}
相关文章