首页 > 编程语言 > 用C语言实现五子棋游戏
2021
08-08

用C语言实现五子棋游戏

C语言写五子棋,使用多文件形式,使用代码看起来更好看;在这里我实现的功能是双人博弈,如果要实现人机对战,那么代码就会很复杂;

一.main.c

在主调函数中首先要提供一个给用户选择的界面,在这里我们假定选择1为开始游戏,2为退出游戏,代码如下:

#include "gobang.h"
void Mean(){
 printf("-----------------------\n");
 printf("  1.play      2.drop up\n");
 printf("-----------------------\n");
}
int main(){
 int seclet = 0;
 int c = 0;
 while (!c){
  Mean();
  printf("Please choose number:\n");
  scanf("%d", &seclet);
  switch (seclet){
  case 1:
   Game();
   break;
  case 2:
   c = 1;
   break;
  default:
   printf("Please Enter Once:\n");
   break;
  }
 }
 printf("Byebye~\n");
 system("pause");
 return 0;
}

函数执行开始,会在显示框中提示用户输入数字,1为进入游戏,此时会调用Game()函数;2为退出游戏。其中while循环的作用是当用户进入界面输入错误(非0或1)或者完成一次游戏后继续弹出选项,只有当输入0才将num置为0,退出循环。

二.gobang.h

函数的头文件,其中包含宏定义和函数的声明,代码如下:

#pragma once
 
#define _CRT_SECURE_NO_WARNINGS 1
 
#include <stdio.h>
#include <windows.h>
 
 
#define ROW 10//控制棋盘大小
#define COL 10//控制棋盘大小
 
#define PLAYER1 1//玩家1的棋为 1
#define PLAYER2 2//玩家2的棋为 2
#define NEXT 3//NEXT代表继续
#define DRAW 4//DRAW代表平局
 
#define U 10//上
#define RU 11//右上
#define R 12//右
#define RD 13//右下
#define D 14//下
#define LD 15//左下
#define L 16//左
#define LU 17//左上
 
extern void Game();//函数的声明

三.gobang.c

五子棋的主要逻辑就是:先打印出棋盘,然后玩家一走一步,判断是否连成五子(若成功则跳出),在打印出走之后的棋盘,玩家二走一步,再次判断是否连成五子,再打印出走之后的棋盘;

所以除了Game()函数外还需要实现以下几个接口:

Print()//打印棋盘
Player()//玩家下棋
Judge//判断是否连成五子

1.Game()

五子棋的主要代码都会写在这个文件里,test.c当中必须包含头文件test.h。Game()函数调用其他函数,实现整个下棋过程。因为两个玩家下棋是同样的操作,所以调用同一个函数,只是传入的玩家参数不同,定义变量who,使得每次进入while循环,who的值都会改变一次,详细过程见如下代码和注释。

void Game(){
 int checkerboard[ROW][COL] = { 0 };//定义一个二维数组
 int result = 0;//定义变量
 int who = PLAYER1;//定义变量who初始值为PLAYER1的值
 while (1){//一直做循环
  Print(checkerboard);//打印出初始面板
  Player(checkerboard, ROW, COL, who);//玩家开始下棋
  result = Judge(checkerboard);
  if (result != NEXT){//判断result的值是否等于NEXT,不等于则跳出循环
   break;
  }
  who = (who == PLAYER1 ? PLAYER2 : PLAYER1);//每进入一次循环who的值都会改变一次
 }
 Print(checkerboard);//打印出最终的面板
 switch (result){
 case PLAYER1://返回值为PLAYLER1,玩家一胜利
  printf("PLALYER1 win\n");
  break;
 case PLAYER2://返回值为PLAYER2,玩家二胜利
  printf("PLAYER2 win\n");
  break;
 case DRAW://返回值为DRAW,平局
  printf("IS DRAW");
  break;
 }
}

2.Print()

打印棋盘的函数并不难实现,代码如下:

void Print(int board[][COL]){//打印当前棋盘
 //system("cls");
 printf(" ");
 for (int i = 0; i < ROW; i++){//打印出横着1到10
  printf(" %d ", i);
 }
 printf("\n");
 for (int i = 0; i < ROW; i++){
  printf("%d", i);
  for (int j = 0; j < COL; j++){
   if (board[i][j] == 0){
    printf(" . ");//打印一个点
   }
   else{
    printf(" %d ", board[i][j]);//打印出当前位置的值
   }
  }
  printf("\n");
 }
}

3.Player()

此函数无非就是给board[x][y]按照x,y坐标赋值,赋值为PLAYER1或者PLAYER2。要注意将x,y定义为全局变量,延长其生命周期,作用是记录每次落子位置,便于计算是否连成五子。Player()函数代码如下:

int x = 0;//全局变量x
int y = 0;//全局变量y
void Player(int board[][COL],int row,int col,int  c){
 while (1){
  printf("Please Enter x y:\n");
  scanf("%d%d",&x,&y);
  if (x<0 || x>row - 1 || y<0 || y>col - 1){//x,y坐标不满足条件则返回到while
   printf("Eorr\n");
   continue;
  }
  if (board[x][y] == 0){//此处为初始值,可以在此处下棋
   board[x][y] = c;//给board[][]赋值为PLAYER1或者PLAYER2
   break;//跳出循环
  }
  else{
   printf("此处不为空,重新输入\n:");
   continue;
  }
 }
}

4.Judge()

判断是否连成五子,这是最难得一步,在这里之前定义得八个方向就用的上了。连成五子无非就四种情况,横着,竖着,斜着(两种情况),则只需要统计则四个方向棋子的数量。在这里说明为什么if()判断中的条件是>=4。在Calculation()函数中统计某一个方向的棋子数量(那八个方向)时,当前棋子的位置已知,假如它的上方有四颗棋子,则五子已经连成,但因为计数器的初始值为0,所以此时count的值为4,函数的返回值也为4,所以在Judge()函数中,if()的条件为>=4(此时>4的情况一般不会发生)。其余详细代码实现如下:

int Judge(int board[][COL]){
 if (Calculation(board, U) + Calculation(board, D)>=4 || \
  //统计上和下棋子数量,此时结果为竖直方向上的相同棋子数量
  Calculation(board, RU) + Calculation(board, LD) >= 4 || \
  //统计右上和左下棋子数量,此时结果为斜着向上的相同棋子数量
  Calculation(board, R) + Calculation(board, L) >= 4 || \
  //统计右和左棋子数量,此时结果为横向上的相同棋子数量
  Calculation(board, RD) + Calculation(board, LU) >= 4){
  //统计右下和左上棋子数量,此时结果为斜着方向上的相同棋子数量
  return board[x][y];
 }
 for (int i = 0; i < ROW; i++){//如果还有一个坐标为初始值,游戏继续
  for (int j = 0; j < COL; j++){
   if (board[i][j] == 0){
    return NEXT;
   }
  }
 }
 return DRAW;//每个坐标都不为初始值且没人胜利,平局
}
int Calculation(int board[][COL], int direction){//传入了方向参数
 int _x = x;//局部变量使其等于当前坐标
 int _y = y;//局部变量使其等于当前坐标
 int count = 0;//计数器
 while (1){//一直做循环直到统计完某个方向
  switch (direction){
  case U://往上则y坐标不变,x坐标减一,以下情况类似
   _x--; break;
  case D:
   _x++; break;
  case L:
   _y--; break;
  case R:
   _y++; break;
  case RU:
   _x--; _y++; break;
  case RD:
   _x++; _y++; break;
  case LD:
   _x++; _y--; break;
  case LU:
   _x--; _y--; break;
  default:
   break;
  }
  if (_x<0 || _x>ROW - 1 || _y<0 || _y>COL - 1){//统计的某个方向已经到了边界,无需统计跳出循环
   break;
  }
  else{
   if (board[x][y] == board[_x][_y]){//棋子和当前下的棋子相同
    count++;//计数器加一
   }
   else{
    break;//棋子和当前下的棋子不同,跳出循环
   }
  }
 }
 return count;
}

我们还可以在Print()函数中加上system("cls"),此函数为清屏操作,加上后就是在一张棋盘下棋了,还可以改进输出棋子的内容,如将

这样就可以用不同的符号代表棋子了,最终的运行结果如下图:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持自学编程网。

编程技巧