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

智能推荐

已知num为无符号十进制整数,请写一非递归算法,该算法输出num对应的r进制的各位数字。要求算法中用到的栈采用线性链表存储结构(1<r<10)。-程序员宅基地

文章浏览阅读74次。思路:num%r得到末位r进制数,num/r得到num去掉末位r进制数后的数字。得到的末位r进制数采用头插法插入链表中,更新num的值,循环计算,直到num为0,最后输出链表。//重置,s指针与头指针指向同一处。//更新num的值,至num为0退出循环。//末位r进制数存入s数据域中。//头插法插入链表中(无头结点)//定义头指针为空,s指针。= NULL) //s不为空,输出链表,栈先入后出。

开始报名!CW32开发者扶持计划正式进行,将助力中国的大学教育及人才培养_cw32开发者扶持计划申请-程序员宅基地

文章浏览阅读176次。武汉芯源半导体积极参与推动中国的大学教育改革以及注重电子行业的人才培养,建立以企业为主体、市场为导向、产学研深度融合的技术创新体系。2023年3月,武汉芯源半导体开发者扶持计划正式开始进行,以打造更为丰富的CW32生态社区。_cw32开发者扶持计划申请

希捷硬盘开机不识别,进入系统后自动扫描硬件以识别显示_st2000dm001不认盘-程序员宅基地

文章浏览阅读5.7k次。2014年底买的一块2TB希捷机械硬盘ST2000DM001-1ER164,用了两年更换了主板、CPU等,后来出现开机不识别的情况,具体表现为:关机后开机,找不到硬盘,就进入BIOS了,只要在BIOS状态下待机半分钟左右再重启,硬盘就会出现。进入系统后,重启(这个过程中主板对硬盘始终处于供电状态),也不会出现不识别硬盘的现象。就好像是硬盘或主板上某个电容坏了一样,刚开始给硬盘通电的N秒钟内电容未能..._st2000dm001不认盘

ADO.NET包含主要对象以及其作用-程序员宅基地

文章浏览阅读1.5k次。ADO.NET的数据源不单单是DB,也可以是XML、ExcelADO.NET连接数据源有两种交互模式:连接模式和断开模式两个对应的组件:数据提供程序(数据提供者)&DataSetSqlConnectionStringBuilder——连接字符串Connection对象用于开启程序和数据库之间的连接public SqlConnection c..._列举ado.net在操作数据库时,常用的对象及作用

Android 自定义对话框不能铺满全屏_android dialog宽度不铺满-程序员宅基地

文章浏览阅读113次。【代码】Android 自定义对话框不能铺满全屏。_android dialog宽度不铺满

Redis的主从集群与哨兵模式_redis的主从和哨兵集群-程序员宅基地

文章浏览阅读331次。Redis的主从集群与哨兵模式Redis的主从模式全量同步增量同步Redis主从同步策略流程redis主从部署环境哨兵模式原理哨兵模式概述哨兵模式的作用哨兵模式项目部署Redis的主从模式1、Redis虽然读取写入的速度都特别快,但是也会产生读压力特别大的情况。2、为了分担读压力,Redis支持主从复制,保证主数据库的数据内容和从数据库的内容完全一致。3、Redis的主从结构可以采用一主多从或者级联结构,Redis主从复制可以根据是否是全量分为全量同步和增量同步。全量同步Redis全量复制一般发_redis的主从和哨兵集群

随便推点

mysql utf-8的作用_为什么不建议在MySQL中使用UTF-8-程序员宅基地

文章浏览阅读116次。作者:brightwang原文:https://www.jianshu.com/p/ab9aa8d4df7d最近我遇到了一个bug,我试着通过Rails在以“utf8”编码的MariaDB中保存一个UTF-8字符串,然后出现了一个离奇的错误:Incorrect string value: ‘😃 我用的是UTF-8编码的客户端,服务器也是UTF-8编码的,数据库也是,就连要保存的这个字符串“????..._mysql utf8的作用

MATLAB中对多张图片进行对比画图操作(包括RGB直方图、高斯+USM锐化后的图、HSV空间分量图及均衡化后的图)_matlab图像比较-程序员宅基地

文章浏览阅读278次。毕业这么久了,最近闲来准备把毕设过程中的代码整理公开一下,所有代码其实都是网上找的,但都是经过调试能跑通的,希望对需要的人有用。PS:里边很多注释不讲什么意思了,能看懂的自然能看懂。_matlab图像比较

16.libgdx根据配置文件生成布局(未完)-程序员宅基地

文章浏览阅读73次。思路:  screen分为普通和复杂两种,普通的功能大部分是页面跳转以及简单的crud数据,复杂的单独弄出来  跳转普通的screen,直接根据配置文件调整设置<layouts> <loyout screenId="0" bg="bg_start" name="start" defaultWinId="" bgm="" remark=""> ..._libgdx ui 布局

playwright-python 处理Text input、Checkboxs 和 radio buttons(三)_playwright checkbox-程序员宅基地

文章浏览阅读3k次,点赞2次,收藏13次。playwright-python 处理Text input和Checkboxs 和 radio buttonsText input输入框输入元素,直接用fill方法即可,支持 ,,[contenteditable] 和<label>这些标签,如下代码:page.fill('#name', 'Peter');# 日期输入page.fill('#date', '2020-02-02')# 时间输入page.fill('#time', '13-15')# 本地日期时间输入p_playwright checkbox

windows10使用Cygwin64安装PHP Swoole扩展_win10 php 安装swoole-程序员宅基地

文章浏览阅读596次,点赞5次,收藏6次。这是我看到最最详细的安装说明文章了,必须要给赞!学习了,也配置了,成功的一批!真不知道还有什么可补充的了,在此做个推广,喜欢的小伙伴,走起!_win10 php 安装swoole

angular2里引入flexible.js(rem的布局)_angular 使用rem-程序员宅基地

文章浏览阅读1k次。今天想实现页面的自适应,本来用的是栅格,但效果不理想,就想起了rem布局。以前使用rem布局,都是在原生html里,还没在框架里使用过,百度没百度出来,就自己琢磨,不知道方法规范不规范,反正成功了,操作如下:1、下载flexible.js2、引入到angular项目里3、根据自己的需要修改细节3.1、在flexible.js里修改每份的像素,3.2、引入cssrem插件,在设置里设..._angular 使用rem

推荐文章

热门文章

相关标签