C语言实现五子棋_c语言 五子棋 算法-程序员宅基地

技术标签: c/c++编程艺术  

C语言实现五子棋

首先项目的目录结构是这样的

p1QdC6.png

函数声明在头文件里,函数实现在game.c中,测试和主函数写在test.c中。

整个代码是以TDD模式写下来的,先写测试函数,再反过头去实现具体的函数。

代码流程

  1. 用户输入,选择游戏难度,此处使用了枚举常量PLAY1和PLAY2与SWitch语句进行搭配使用,可以让代码清晰明了。
  2. 进入游戏的流程是根据用户输入的难度选择,创建不同大小的数组空间(棋盘),这里本应该用malloc动态申请内存来做,但是偷懒,我用符号常量定义了一个大空间ROW和COL,在选择处做一个判断,分别传入不同的row和col。
  3. 完成初始化棋盘,展示棋盘,随机种子,玩家走,电脑走(随机走),判赢的函数。

其中需要重点说一下的是判赢函数

我的判断思路是判断上一子的落点(电脑或玩家),从落点开始分别向横竖主对角次对角线进行向前计数,遇到边界或是对手的子就停下,若任一个方向子数count加起来大于5,则说明五子连珠。其中上一子的落点,我用全局变量 lx,ly 表示,在这里就要说一下全局变量的用法

extern int lx ; //在头文件中声明一下这个变量,以免多次引用头文件造成重复定义的错误

int lx = 0; //在game.c 中真的定义,分配空间。

game.h

#ifndef __GAME_H__
#define __GAME_H__

#include <stdio.h>
#include <Windows.h>
#include <time.h>
#include <stdlib.h>

#define ROW 11    //从1,1坐标开始
#define COL 11
extern int Piece ;        //n子棋
extern int lx,ly;         //记录上一子的位置

enum OPTION 
{
    EXIT,           //从0开始
    PLAY,
    PLAY2
};

void InitBoard(char board[ROW][COL], int row, int col);
void DisplayBoard(char board[ROW][COL], int row, int col);
void PlayerMove(char board[ROW][COL], int row, int col);
void ComputerMove(char board[ROW][COL], int row, int col);
char CheckWin(char board[ROW][COL], int row, int col);         
int  IsFull(char board[ROW][COL], int row, int col);
int  seek(char board[ROW][COL],int row,int col,int x,int y,char ch);
#endif 

game.c

#include "game.h"
#include <string.h>
#define _CRT_SECURE_NO_WARNINGS 1
int lx,ly;
void InitBoard(char board[ROW][COL],int row,int col){
    //memset(board,'0',sizeof(board));  //是不可以的,因为只传了board首元素地址
    memset(board,' ',row*col*sizeof(char));

}
char CheckWin(char board[ROW][COL],int row,int col){          //满了返回3,玩家赢返回1,电脑赢返回2
    int i = 0,j = 0;



            if(seek(board,row,col,lx,ly,'*')){
                    return 1;
            }
            if(seek(board,row,col,lx,ly,'+')){
                    return 2;
            }

    if(!IsFull(board,row,col)){                 //判断满要放在后面,若最后一子下满也有可能胜利
        return 3;
    }
    return 0;
}
int seek(char board[ROW][COL],int row,int col,int x,int y,char ch){
    int dir[4][2][2] ={
   {
   {
   0,-1},{
   0,1}},{
   {-1,0},{
   1,0}},{
   {-1,-1},{
   1,1}},{
   {-1,1},{
   1,-1}}};//分别是横竖主对角线
    int i,j;
    int flag = 1;
    int tmpx = x;
    int tmpy = y;
    int count = 1;
    for(i=0; i<4; i++){

        if(board[x][y]== ch){         //从遍历点开始,若是对应的子,则初始count=1
            count = 1;
        }
        else{
            count = 0;
        }                                           //为左右两个方向遍历计数
        for(j=0; j<2; j++){                          //j=0是向棋子左侧方向遍历
            flag = 1;
            while(flag){
                tmpx = tmpx +dir[i][j][0];
                tmpy = tmpy +dir[i][j][1];
                if(tmpx<1||tmpx>row||tmpy<1||tmpy>col){

                    break;
                }
                if(ch == board[tmpx][tmpy]){

                    count++;
                }else{
                    flag = 0;
                }
            }
            tmpx = x;
            tmpy = y;

        }
        if(count >= Piece){    // 几子棋
            return 1;         //获胜
        }
    }
    return 0;
}
int IsFull(char board[ROW][COL], int row, int col){   //1是非满
    int i = 0;
    int j = 0;
    for(i=1; i<row; i++){
        for(j=1; j<col; j++){
            if(board[i][j]== ' '){
                return 1;
            }
        }
    }
    return 0;
}
void DisplayBoard(char board[ROW][COL],int row,int col){
    int i = 0;
    int j = 0;
    //for(i=0; i<ROW; i++){
    
    //  for(j=0; j<COL; j++){
    
    //      printf("%c ",board[i][j]);
    //  }
    //  printf("\n");
    //}
    for(i=1; i<row; i++){
        if(i == 1){
            for(j=1; j<col;j++){
                if(1 == j){
                    printf("   ");
                }
                printf("|%2d|",j);
            }
            printf("\n");
            for(j=1; j<col;j++){
                if(1 == j){
                    printf("   ");
                }
                printf(" ---");
            }
            printf("\n");
        }
        for(j=1; j<col;j++){
            if(1 == j){
                printf("%2d|",i);
            }
            printf("| %c ",board[i][j]);
        }
        printf("|\n");
        for(j=1; j<col;j++){
            if(1 == j){
                printf("   ");
            }
            printf(" ---");
        }
        printf("\n");
    }
}
void PlayerMove(char board[ROW][COL], int row, int col){
    int x = 0;
    int y = 0;
    while(1){
        printf("请输入你要下的位置(x,y)->");
        scanf("%d%d",&x,&y);
        lx = x;
        ly = y;
        if(x>=row|| y>=col|| x<=0|| y<=0){
            printf("输入非法的位置!\n");
            continue;
        }
        else if(board[x][y]=='*'||board[x][y]=='+'){
            printf("此位置已经有棋子了!\n");
            continue;
        }
        else{
            board[x][y] = '*';
            break;
        }
    }
    DisplayBoard(board, row, col);
}
void ComputerMove(char board[ROW][COL], int row, int col){
    int x = 0;
    int y = 0;
    printf("我江流儿走一步~\n");
    while(1){
        x = rand()%(row-1)+1;  //因为传的row是4 如果3X3生成1~3的随机数
        y = rand()%(col-1)+1;
        lx = x;
        ly = y;
        if((x<row&&y<col&&x>0&&y>0) && board[x][y]==' '){
            board[x][y] = '+';
            break;
        }
    }
    DisplayBoard(board, row,col);
}

test.c

#include <stdio.h>
#include "game.h"
#define _CRT_SECURE_NO_WARNINGS 1
int Piece;
void menu(){
    printf("******************************\n");
    printf("********和江流儿下棋**********\n");
    printf("********1.和他来一局三子棋****\n");
    printf("********2.和他来一局五子棋****\n");
    printf("********0.退出游戏************\n");
    printf("******************************\n");
}
void game(int choice){
    char board[ROW][COL];
    int ret = 0;//CheckWin(board,ROW,COL);
    int row = 0;
    int col = 0;
    if(choice == 1){
        row = 4;
        col = 4;
        Piece = 3;
    }
    else {
        row = 11;
        col = 11;
        Piece = 5;
    }
    InitBoard(board, ROW, COL);     //初始化棋盘
    DisplayBoard(board, row, col);      //展示该棋盘
    srand((unsigned int)time(NULL));    
    while(!CheckWin(board,row,col)){                         //产生结果或者棋盘满了

        PlayerMove(board,row,col);

        if(ret = CheckWin(board,row,col)){
            break;
        }
        ComputerMove(board, row,col);

        if(ret = CheckWin(board,row,col)){
            break;
        }
    }
    if(1 == ret){                                     
        printf("-----------大吉大利,今晚吃鸡!\n");
    }
    else if(2 == ret){
        printf("-----------败北!!\n");
    }
    else{
        printf("-----------势均力敌!!\n");
    }
}
int main(){
    int choice = 0;

    do{
        menu();
        printf("请输入->");
        scanf("%d",&choice);
        switch(choice){
            case PLAY:game(choice);
                continue;
            case PLAY2:game(choice);
                continue;
            case EXIT:printf("退出游戏");
                break;
            default:printf("没有这个选项,请重新输入\n");
                continue;
        }

    }while(choice);
}

运行结果

p1QBvD.png

p1QWPP.png

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/hanzheng6602/article/details/79034692

智能推荐

Python实现对Mysql数据库的增删改查_编写python程序,实现对mysql数据的增删改查-程序员宅基地

文章浏览阅读750次。客户需求使用Python脚本实现对对Mysql数据库的增删改查、_编写python程序,实现对mysql数据的增删改查

纯源码编译安装nginx、MySQL、redis、java_nginx java mysql-程序员宅基地

文章浏览阅读919次,点赞24次,收藏20次。纯源码编译安装nginx、MySQL、redis、java。_nginx java mysql

基于单片机的智能交通灯控制系统设计_基于单片机智能交通灯系统框图-程序员宅基地

文章浏览阅读1.2k次,点赞27次,收藏11次。本文采用AT89C52 单片机设计了一款智能交通灯控制系统,该系统通过检测和比较南北方向、东西方向车流量的大小,灵活地调节交通灯的通行时间,达到对交通灯自动实时控制的目的。系统不但在传统模式下能够正常工作,还可以实现根据车流量自动调节交通信号灯的智能模式,以及特殊车辆到来时的紧急模式,有效地缓解了车辆拥堵情况,提高了车行效率和道路利用率。_基于单片机智能交通灯系统框图

学习数据库概论第六天_在数据库demo02中,在删除student表中的元组时把该元组的前两个属性值插入bian_s中-程序员宅基地

文章浏览阅读740次。第五章数据库的完整性是指数据的正确性和相容性。数据的正确性是指数据是符合现实世界语义、反映当前实际状况的,数据的相容性是指数据库同一对象在不同关系表中的数据是符合逻辑的。数据的完整性是为了防止数据库中存在不符合语义的数据,也就是防止数据库中存在不正确的护数据。数据的安全性是保护数据库防止恶意破坏和非法存取。因此,完整性检查和控制的防范对象是不合语义的、不正确的数据,防止它们进入数据库。安全性控..._在数据库demo02中,在删除student表中的元组时把该元组的前两个属性值插入bian_s中

(新手友好!)Ubuntu20安装xv6-riscv_ubuntu20 xv6-程序员宅基地

文章浏览阅读5.5k次,点赞17次,收藏41次。1.安装qemusudo apt-get install qemu这里已经显示是最新版本了不知道为什么没下载成功,就根据提示来输入命令sudo apt install qemu-system-x86 结果下载过程中又显示有些包装不了~根据提示更新Linux的apt的包列表,输入:sudo apt-get update然后更新完成,再试一次输入:sudo apt install qemu-system-x86 这样应该是可以了验证一下,输入:qemu-system-i_ubuntu20 xv6

vue如何兼容IE8以上浏览器_vue 2.7 ie吗-程序员宅基地

文章浏览阅读2.6k次,点赞2次,收藏6次。参考:https://blog.csdn.net/around_primary/article/details/79087466https://blog.csdn.net/u012733501/article/details/826657361、npm安装babel-polyfillnpm install babel-polyfill --save-dev2、在入口文件main.js中引入..._vue 2.7 ie吗

随便推点

SHL逻辑测试-日程安排-程序员宅基地

文章浏览阅读262次。日程安排目录日程安排1.1 开会的第一个工作日1.2 两个同事一起开会1.3 销售日历1.4 团队日程安排1.5 轮班安排1.6 主管安排1.7 客户参观1.8 安装安排1.9 会议安排1.10 办公室日历1.11 计划安排1.12 工作安排1.13 审计安排1.14 欢迎安排1.15 员工表彰1.16 加班安排1.17 安装安排1.18 人员配置请求1.19 工时记录1.20 见习安排1.1..._shl测评题 团队日历

时间序列预测----(基于多变量深度模型)_多变量时间序列模型-程序员宅基地

文章浏览阅读1.4w次,点赞39次,收藏263次。1. 什么是多变量时序预测:多变量时间序列预测问题可以被理解为,利用历史时刻的各项数据来预测下一个时刻的目标数据。2. 实验数据集:在本文中,我使用了北京市空气污染历史监测数据集来进行时序预测实验,那么时序预测任务则是利用过去一段时间所记录的温度、气压、风速以及空气污染程度等数据来预测下一时刻的空气污染程度。数据集下载地址:http://archive.ics.uci.edu/ml/datasets/Beijing+PM2.5+Data数据来源自位于北京的美国大使馆在2010年至2014年共5_多变量时间序列模型

通过python实现linux切换用户_Python操作远程服务器切换到root用户-程序员宅基地

文章浏览阅读1.5k次。在自动化运维过程中,需要远程服务器切换到root用户下执行命令,尝试了一些方法,得到如下好用的方法,供大家使用:import timeimport paramikodef verification_ssh(host,username,password,port,root_pwd,cmd):s=paramiko.SSHClient()s.load_system_host_keys()s.set_mi..._python 远程root登录

vue如何使用原生js写动画效果_Vue.js - Transition过渡动画的使用1(使用CSS过渡或动画)...-程序员宅基地

文章浏览阅读197次。一、基本介绍1, 组件(1)如果某个元素或者组件需要使用过渡动画效果,只需使用 vue提供的 组件将其包裹起来封装成过渡组件。(2)Vue只有在插入,更新或者移除 DOM元素时才会应用过渡效果,例如:v-if(条件渲染)v-show(条件展示)动态组件在组件的根节点上,并且被 vue实例 DOM方法触发。比如使用appendTo方法把组件添加到某个根节点上2,过渡效果实现方式过渡效果具体的实现方..._原生动画 js transition

ENVI-程序员宅基地

文章浏览阅读200次。1、完整的遥感图像处理平台2、采用交互式数据语言IDL(Interactive Data Language)_envi汉化官网博客

setVariable和setVariableLocal区别_execution.setvariable和execution.setlocalvariable-程序员宅基地

文章浏览阅读517次。delegateTask.setVariableLocal是set进当前execution里面_execution.setvariable和execution.setlocalvariable

推荐文章

热门文章

相关标签