C++畅玩五子棋项目

news/2024/10/17 18:20:43/

畅玩五子棋项目说明

​ 项目创建环境:Windows 10 + Visual Studio 2019

​ 棋子类型:实心●和空心◎。其中,棋子彩色棋子为当前落子

​ 实现功能:对战模式(人人对战)、人机模式(人机模式)、加载历史、帮助反馈

功能介绍:

1.对战模式

​ 选择该模式,直接进入人人对战,默认实心棋子先手。

2.人机模式

​ 选择该模式,需要再选择一次先手和后手,再进入人机对战。

​ 进入游戏后,在你落子之后,机器落子需要1s的“停顿”时间,以方便你查看当前局势。

3.加载历史

​ 当下完一局棋后(已分出胜负),系统为你自动保存对局状态。若查看,选择此项。

4.帮助反馈

​ 按R可悔棋:在对战模式下,按一下R则退回一步;在人机模式下,按一下R则棋手和机器人各退回一步。对战中可N次按R且悔N步棋。

源代码文件列表

main.cpp
WelcomeMenu.h
WelcomeMenu.cpp
SetBoardValue.h
AbstractPlayer.h
AbstractPlayer.cpp
HumanPlayer.h
HumanPlayer.cpp
HumanAnalysic.h
HumanAnalysic.cpp
RebotPlayer.h
RebotPlayer.cpp
RebotAnalysis.h
RebotAnalysis.cpp
HistoryFun.h
HistoryFun.cpp

源代码文件

main.cpp

#include<iostream>
#include"SetBoardValue.h"
#include"WelcomeMenu.h"
#include"HumanPlayer.h"
#include"RebotPlayer.h"
#include"HumanAnalysic.h"
#include"HistoryFun.h"using namespace std;//这是一个打印字符集,并使用关键字extern作了全局声明。在原版中通过形状来标识当前落子,改版后通过颜色标识
string symbol[14] = { "?" , "┏" , "┓" , "┗" , "┛" , "┠" , "┨" , "┯" , "┷" , "┼" , "●" , "◎" , "●" , "◎" };
int mode;	//对战模式
int step = 0;	//下棋步数int main()
{enum chessSign { black = 12, white = 13 };mode = WelcomeMenu();AbstractPlayer::ClearBoardArray();AbstractPlayer::DisplayBoard();switch (mode){case 1:{HumanPlayer Student1(black);HumanPlayer Student2(white);while (true){if (HumanAnalysic(Student1)){break;}if (HumanAnalysic(Student2)){break;}}break;//RebotPlayer Student1(black);//RebotPlayer Student2(white);//while (true)//{//	if (Student1.ComputerPlacePieces())//	{//		break;//	}//	if (Student2.ComputerPlacePieces())//	{//		break;//	}//}//break;}case 2:{HumanPlayer Student1(black);RebotPlayer Student2(white);while (true){if (HumanAnalysic(Student1)){break;}if (Student2.ComputerPlacePieces()){break;}}break;}case 3:{HumanPlayer Student1(white);RebotPlayer Student2(black);while (true){if (Student2.ComputerPlacePieces()){break;}if (HumanAnalysic(Student1)){break;}}break;}}if (SaveHistory()){cout << endl;cout << "    ";cout << "当前对战已保存!" << endl;}cout << endl;system("pause");return 0;
}

WelcomeMenu.h

#pragma onceint WelcomeMenu();bool WelcomeView();

WelcomeMenu.cpp

/*进入模式选择界面
* 模式选择:
*   1.人人对战
*   2.人机对战[先手/后手]
*   3.加载历史
*   4.帮助反馈
*/#include<iostream>
#include<string>
#include"WelcomeMenu.h"
#include"HistoryFun.h"using namespace std;int WelcomeMenu()
{int mode1, mode2;string command;do{WelcomeView();cout << "         ";cout << "您好!请选择游戏模式: _" << '\b';getline(cin, command);//cin >> command;cout << endl;if (command.length() != 1 || command.at(0) < '1' || command.at(0) > '4'){cout << "           ";cout << "输入错误, 请重试!" << endl << endl;system("pause");continue;}mode1 = command.at(0) - '0';if (mode1 == 1){return 1;}else if (mode1 == 2){do{cout << "       ";cout << "请选择模式[先手1/后手2]: _" << '\b';getline(cin, command);//cin >> command;if (command.length() != 1 || command.at(0) < '1' || command.at(0) > '2'){cout << "           ";cout << "输入错误, 请重试!" << endl << endl;system("pause");continue;}mode2 = command.at(0) - '0';if (mode2 == 1){return 2;}else if (mode2 == 2){return 3;}} while (true);}else if (mode1 == 3){if (!loadHistory()){cout << "             ";cout << "找不到历史对战!" << endl << endl;system("pause");continue;}else{cout << "             ";cout << "已加载完毕!" << endl << endl;system("pause");continue;}}else if (mode1 == 4){cout << "           ";cout << " 按R: 悔一手棋!" << endl << endl;system("pause");continue;}} while (true);}bool WelcomeView()
{system("cls");const int WIDTH_N = 32;int i;cout << "    ";for (i = 0; i < WIDTH_N; i++){cout << "=";}cout << endl;cout << "    ";cout << "*   |> 欢迎进入五子棋游戏 <|   *" << endl;cout << "    ";cout << "***                          ***" << endl;cout << "    ";cout << "*     [ AUTHOR:CAO ZHEN ]     *" << endl;cout << "    ";for (i = 0; i < WIDTH_N; i++){cout << "=";}cout << endl;cout << endl;cout << "    ";cout << "     ┌-------------------┐       " << endl;cout << "    ";cout << "     │  → 1-对战模式 ← │         " << endl;cout << "    ";cout << "     └-------------------┘       " << endl;cout << "    ";cout << "     ┌-------------------┐       " << endl;cout << "    ";cout << "     │  → 2-人机模式 ← │         " << endl;cout << "    ";cout << "     └-------------------┘       " << endl;cout << "    ";cout << "     ┌-------------------┐       " << endl;cout << "    ";cout << "     │  → 3-加载历史 ← │         " << endl;cout << "    ";cout << "     └-------------------┘       " << endl;cout << "    ";cout << "     ┌-------------------┐       " << endl;cout << "    ";cout << "     │  → 4-帮助反馈 ← │         " << endl;cout << "    ";cout << "     └-------------------┘       " << endl;cout << endl;cout << "    ";for (i = 0; i < WIDTH_N; i++){cout << "-";}cout << endl << endl;return true;
}

SetBoardValue.h

#pragma once#include<iostream>
#define M 15
#define N 15//预定义棋子颜色
#define RESET "\033[0m"
#define RED "\033[31m" /* Red */
#define GREEN "\033[32m" /* Green */
#define BLUE "\033[34m" /* Blue */extern std::string symbol[14];
extern int mode;
extern int step;

AbstractPlayer.h

#pragma once#include"SetBoardValue.h"class AbstractPlayer
{
public:int GetChessSign();static bool ClearBoardArray();static bool DisplayBoard();static int JudgeOutcome();static int BoardArray[M * N];	//棋局矩阵static int StorageBoardArray[M * N * 225];	//保存历史int m_ChessSign;	//棋子类型};

AbstractPlayer.cpp

#include "AbstractPlayer.h"
#include<iostream>
#include<string>
#include<iomanip>
#include"HistoryFun.h"using namespace std;int AbstractPlayer::BoardArray[M * N] = { 0 };
int AbstractPlayer::StorageBoardArray[M * N * 225] = { 0 };/**************************** 获取当前类的符号 *******************************/
int AbstractPlayer::GetChessSign()
{return m_ChessSign - 2;
}/********************* 静态成员函数:清空棋局,开局 ***************************/
bool AbstractPlayer::ClearBoardArray()
{int i, j;BoardArray[0] = 1;BoardArray[N - 1] = 2;BoardArray[(M - 1) * N] = 3;BoardArray[(M - 1) * N + N - 1] = 4;for (i = 1; i < M - 1; i++){BoardArray[i * N + 0] = 5;BoardArray[i * N + N - 1] = 6;}for (j = 1; j < N - 1; j++){BoardArray[0 * N + j] = 7;BoardArray[(M - 1) * N + j] = 8;}for (i = 1; i < M - 1; i++){for (j = 1; j < N - 1; j++){BoardArray[i * N + j] = 9;}}StorageHistoryArray(BoardArray);return true;
}/************************静态成员函数: 显示棋局状态 *************************/
bool AbstractPlayer::DisplayBoard()
{system("cls");cout << endl;;cout << setw(N - 5) << " ";if (mode == 1){cout << " 畅玩五子棋(对战版)" << endl;}else if (mode == 2 || mode == 3){cout << " 畅玩五子棋(人机版)" << endl;}else{cout << "畅玩五子棋(历史对战)" << endl;}int i, j;char ch_y = 'A';for (i = 0; i < M; i++){cout << setw(4) << M - i;for (j = 0; j < N; j++){if (BoardArray[i * N + j] < 10){if (j != 0 && BoardArray[i * N + j - 1] >= 10){cout << setw(2) << symbol[BoardArray[i * N + j]];}else{cout << setw(3) << symbol[BoardArray[i * N + j]];}}else if (BoardArray[i * N + j] == 10 || BoardArray[i * N + j] == 11){if (j != 0 && BoardArray[i * N + j - 1] >= 10){cout << setw(2) << symbol[BoardArray[i * N + j]];}else{cout << setw(3) << symbol[BoardArray[i * N + j]];}}else if (BoardArray[i * N + j] == 12){if (j != 0 && BoardArray[i * N + j - 1] >= 10){cout << setw(5) << RED << symbol[BoardArray[i * N + j]] << RESET;}else{cout << setw(6) << RED << symbol[BoardArray[i * N + j]] << RESET;}}else if (BoardArray[i * N + j] == 13){if (j != 0 && BoardArray[i * N + j - 1] >= 10){cout << setw(5) << RED << symbol[BoardArray[i * N + j]] << RESET;}else{cout << setw(6) << GREEN << symbol[BoardArray[i * N + j]] << RESET;}}}cout << endl;}cout << setw(4) << " ";for (j = 0; j < N; j++){cout << setw(2) << ch_y++;}cout << endl << endl;return true;
}/*************************** 构造判断矩阵,判断是否有人赢棋*************************/
int AbstractPlayer::JudgeOutcome()
{int i, j, JudgeBoardArray[M][N];//构造一个判断矩阵,这里设未落子位置值为0,棋手1落子位置值为-1,棋手2落子位置值为1for (i = 0; i < M; i++){for (j = 0; j < N; j++){if (BoardArray[i * N + j] == 10){JudgeBoardArray[i][j] = -1;}else if (BoardArray[i * N + j] == 11){JudgeBoardArray[i][j] = 1;}else{JudgeBoardArray[i][j] = 0;}}}//判断 ' - ' ,是否有棋手赢棋for (i = 0; i < M; i++){for (j = 0; j <= N - 5; j++){if (JudgeBoardArray[i][j] == -1 && JudgeBoardArray[i][j + 1] == -1 && JudgeBoardArray[i][j + 2] == -1 && JudgeBoardArray[i][j + 3] == -1 && JudgeBoardArray[i][j + 4] == -1){return -1;}if (JudgeBoardArray[i][j] == 1 && JudgeBoardArray[i][j + 1] == 1 && JudgeBoardArray[i][j + 2] == 1 && JudgeBoardArray[i][j + 3] == 1 && JudgeBoardArray[i][j + 4] == 1){return 1;}}}//判断 ' | ' ,是否有棋手赢棋for (i = 0; i <= M - 5; i++){for (j = 0; j < N; j++){if (JudgeBoardArray[i][j] == -1 && JudgeBoardArray[i + 1][j] == -1 && JudgeBoardArray[i + 2][j] == -1 && JudgeBoardArray[i + 3][j] == -1 && JudgeBoardArray[i + 4][j] == -1){return -1;}if (JudgeBoardArray[i][j] == 1 && JudgeBoardArray[i + 1][j] == 1 && JudgeBoardArray[i + 2][j] == 1 && JudgeBoardArray[i + 3][j] == 1 && JudgeBoardArray[i + 4][j] == 1){return 1;}}}//判断 ' \ ' ,是否有棋手赢棋for (i = 0; i <= M - 5; i++){for (j = 0; j <= N - 5; j++){if (JudgeBoardArray[i][j] == -1 && JudgeBoardArray[i + 1][j + 1] == -1 && JudgeBoardArray[i + 2][j + 2] == -1 && JudgeBoardArray[i + 3][j + 3] == -1 && JudgeBoardArray[i + 4][j + 4] == -1){return -1;}if (JudgeBoardArray[i][j] == 1 && JudgeBoardArray[i + 1][j + 1] == 1 && JudgeBoardArray[i + 2][j + 2] == 1 && JudgeBoardArray[i + 3][j + 3] == 1 && JudgeBoardArray[i + 4][j + 4] == 1){return 1;}}}//判断 ' / ' ,是否有棋手赢棋for (i = 0; i <= M - 5; i++){for (j = 4; j <= N; j++){if (JudgeBoardArray[i][j] == -1 && JudgeBoardArray[i + 1][j - 1] == -1 && JudgeBoardArray[i + 2][j - 2] == -1 && JudgeBoardArray[i + 3][j - 3] == -1 && JudgeBoardArray[i + 4][j - 4] == -1){return -1;}if (JudgeBoardArray[i][j] == 1 && JudgeBoardArray[i + 1][j - 1] == 1 && JudgeBoardArray[i + 2][j - 2] == 1 && JudgeBoardArray[i + 3][j - 3] == 1 && JudgeBoardArray[i + 4][j - 4] == 1){return 1;}}}return 0;
}

HumanPlayer.h

#pragma once#include"AbstractPlayer.h"class HumanPlayer :public AbstractPlayer
{
public:HumanPlayer(int ChessSign);bool PlacePieces();private:int m_X;	//当前落子的x坐标int m_Y;	//当前落子的y坐标
};

HumanPlayer.cpp

/*人工棋手类[继承抽象棋手类]的实现(cao zhen 2020/11/15)
* 继承方法:
*   GetChessSign 获取当前类的棋子符号(黑或白)
*   ClearBoardArray [静态]清空棋局,开局
*   DisplayBoard [静态]显示棋局状态
*   JudgeOutcome 判断是否有人赢棋
* 创建方法:
*  HumanPlayer 有参构造函数
*  PlacePieces 棋手落子
*/#include<iostream>
#include<string>
#include<iomanip>
#include "HumanPlayer.h"
#include"HistoryFun.h"using namespace std;HumanPlayer::HumanPlayer(int ChessSign)
{m_ChessSign = ChessSign;
}/******************************** 棋手落子 ********************************/
bool HumanPlayer::PlacePieces()
{string command;do{getline(cin, command);//cin >> command;if (command.length() == 1){if (command.at(0) == 'R'){if (mode == 1){RegretMove();}else{RegretMove();RegretMove();}cout << symbol[GetChessSign()] << "方请落子:" << flush;}else{break;}}else{break;}} while (true);if (!(command.length() == 2 || command.length() == 3)){return false;}if (!(command.at(0) >= 'A' && command.at(0) <= 'O')){return false;}if (command.length() == 2){if (!(command.at(1) >= '1' && command.at(1) <= '9')){return false;}m_X = 14 - (command.at(1) - '1');m_Y = command.at(0) - 'A';}if (command.length() == 3){if (!(command.at(1) == '1' && command.at(2) >= '0' && command.at(2) <= '5')){return false;}m_X = 14 - (9 + command.at(2) - '0');m_Y = command.at(0) - 'A';}if (BoardArray[m_X * N + m_Y] >= 10){return false;}BoardArray[m_X * N + m_Y] = m_ChessSign;DisplayBoard();StorageHistoryArray(BoardArray);BoardArray[m_X * N + m_Y] = m_ChessSign - 2;return true;
}

HumanAnalysic.h

#pragma once#include"HumanPlayer.h"int HumanAnalysic(HumanPlayer& Student);

HumanAnalysic.cpp

#include<iostream>
#include<string>
#include"SetBoardValue.h"
#include"HumanPlayer.h"
#include"HumanAnalysic.h"using namespace std;int HumanAnalysic(HumanPlayer& Student)
{int outcome;cout << symbol[Student.GetChessSign()] << "方请落子:" << flush;while (Student.PlacePieces() == false){cout << "    输入格式错误,请重新输入!" << endl;cout << symbol[Student.GetChessSign()] << "方请落子:" << flush;};if (outcome = AbstractPlayer::JudgeOutcome()){cout << "    ";cout << symbol[Student.GetChessSign()] << "胜出!" << endl;}return outcome;
}

RebotPlayer.h

#pragma once#include"AbstractPlayer.h"class RebotPlayer :public AbstractPlayer
{
public:RebotPlayer(int ChessSign);int ComputerPlacePieces();
};

RebotPlayer.cpp

/*电脑棋手类[继承抽象棋手类]的实现(cao zhen 2020/11/15)
* 继承方法:
*   GetChessSign 获取当前类的棋子符号(黑或白)
*   ClearBoardArray [静态]清空棋局,开局
*   DisplayBoard [静态]显示棋局状态
*   JudgeOutcome [静态]判断是否有人赢棋
* 创建方法:
*   RebotPlayer 有参构造函数
*   ComputerPlacePieces 电脑落子
*/#include<iostream>
#include<string>
#include<windows.h>
#include "RebotPlayer.h"
#include"RebotAnalysis.h"
#include"HistoryFun.h"using namespace std;RebotPlayer::RebotPlayer(int ChessSign)
{m_ChessSign = ChessSign;
}/******************************* 电脑落子 **************************************/
int RebotPlayer::ComputerPlacePieces()
{int i, j, JudgeBoardArray[M][N] = { 0 };int labX, labY;int chessSign = 0, outcome = 0;//构造一个判断矩阵,这里设未落子位置值为0,棋手1落子位置值为-1,棋手2落子位置值为1for (i = 0; i < M; i++){for (j = 0; j < N; j++){if (BoardArray[i * N + j] == 10){JudgeBoardArray[i][j] = -1;}else if (BoardArray[i * N + j] == 11){JudgeBoardArray[i][j] = 1;}else{JudgeBoardArray[i][j] = 0;}}}if (this->m_ChessSign == 12){chessSign = -1;}else if (this->m_ChessSign == 13){chessSign = 1;}//以经典开局方式开局。其中若机器先手,则落子天元if (JudgeClassicStart(*JudgeBoardArray, &labX, &labY, chessSign)){BoardArray[labX * N + labY] = this->m_ChessSign;DisplayBoard();StorageHistoryArray(BoardArray);BoardArray[labX * N + labY] = this->m_ChessSign - 2;return 0;}//经典开局后,布局评分if (BoardAnalysis(*JudgeBoardArray, &labX, &labY, chessSign)){BoardArray[labX * N + labY] = this->m_ChessSign;Sleep(1000);DisplayBoard();StorageHistoryArray(BoardArray);BoardArray[labX * N + labY] = this->m_ChessSign - 2;}if (outcome = JudgeOutcome()){cout << "    ";cout << symbol[GetChessSign()] << "胜出!" << endl;}return outcome;
}

RebotAnalysis.h

#pragma oncebool JudgeClassicStart(int* judgeArray, int* labX, int* labY, int chessSign);bool BoardAnalysis(int* judgeArray, int* x, int* y, int chessSign);int scoreFunction(const int* judgeArray, int i, int j, int chessSign);int GetLocalChessSign(const int* judgeArray, int labX, int labY, int direction, int distance);

RebotAnalysis.cpp

#include<time.h>
#include"SetBoardValue.h"
#include"RebotAnalysis.h"/**************** 对矩阵进行判断,若为经典开局,则返回1;否,则返回0 ***************/
bool JudgeClassicStart(int* judgeArray, int* labX, int* labY, int chessSign)
{int i, j, delta1, delta2;int x1 = 0, y1 = 0, x2 = 0, y2 = 0;int num1 = 0, num2 = 0;for (i = 0; i < M; i++){for (j = 0; j < N; j++){if (judgeArray[i * N + j] == -1){x1 = i;y1 = j;if (++num1 > 1){return false;}}if (judgeArray[i * N + j] == 1){x2 = i;y2 = j;if (++num2 > 1){return false;}}}}if (num1 == 0 && num2 == 0){*labX = 7;*labY = 7;return true;}if ((num1 == 0 && num2 == 1) || (num1 == 1 && num2 == 0)){if (judgeArray[7 * N + 7] == 0){*labX = 7;*labY = 7;}else{int dx = 0, dy = 0;do{srand(time(NULL));dx = rand() % 3 - 1;srand(time(NULL));dy = rand() % 3 - 1;} while (dx == 0 && dy == 0);*labX = 7 + dx;*labY = 7 + dy;}return true;}if (num1 == 1 && num2 == 1){delta1 = abs(x2 - x1);delta2 = abs(y2 - y1);if (delta1 <= 1 && delta2 <= 1){if (chessSign == -1){int dx = 0, dy = 0;do{srand(time(NULL));dx = rand() % 3 - 1;srand(time(NULL));dy = rand() % 3 - 1;} while (dx == 0 && dy == 0 || ((x1 + dx)==x2 &&(y1+dy)==y2));*labX = x1 + dx;*labY = y1 + dy;}else if (chessSign == 1){int dx = 0, dy = 0;do{srand(time(NULL));dx = rand() % 3 - 1;srand(time(NULL));dy = rand() % 3 - 1;} while (dx == 0 && dy == 0 || ((x2 + dx) == x1 && (y2 + dy) == y1));*labX = x2 + dx;*labY = y2 + dy;}}else{int dx = 0, dy = 0;do{srand(time(NULL));dx = rand() % 3 - 1;srand(time(NULL));dy = rand() % 3 - 1;} while (dx == 0 && dy == 0);*labX = 7 + dx;*labY = 7 + dy;}return true;}
}/*************** 权值法给当前布局打分,并返回最优反馈坐标 ********************/
bool BoardAnalysis(int* judgeArray, int* x, int* y, int chessSign)
{int i, j, scoreArray[M * N] = { 0 };int maxScore=-250000;for (i = 0; i < M; i++){for (j = 0; j < N; j++){if (judgeArray[i * N + j] == 0){scoreArray[i * N + j] = scoreFunction(judgeArray, i, j, chessSign);if (scoreArray[i * N + j] > maxScore){maxScore = scoreArray[i * N + j];*x = i;*y = j;}}}}return true;
}/**************************** 统计某个点的得分 *********************************/
int scoreFunction(const int* judgeArray, int i, int j, int chessSign)
{int score = 0;for (int maySign = -1; maySign <= 1; maySign += 2){int twoNum = 0;for (int direction = 1; direction <= 8; direction++){//活四: 0 1 1 1 1 *if (GetLocalChessSign(judgeArray, i, j, direction, -1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -3) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -4) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -5) == 0){score += 300000;if (maySign != chessSign){score -= 500;}continue;}//死四(一): -1 1 1 1 1 *if (GetLocalChessSign(judgeArray, i, j, direction, -1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -3) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -4) == maySign &&(GetLocalChessSign(judgeArray, i, j, direction, -5) == -maySign ||GetLocalChessSign(judgeArray, i, j, direction, -5) == -2)){score += 250000;if (maySign != chessSign){score -= 500;}continue;}//死四(二): 1 1 1 * 1if (GetLocalChessSign(judgeArray, i, j, direction, -1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -3) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, 1) == maySign){score += 240000;if (maySign != chessSign){score -= 500;}continue;}//死四(三): 1 1 * 1 1if (GetLocalChessSign(judgeArray, i, j, direction, -1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, 1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, 2) == maySign){score += 230000;if (maySign != chessSign){score -= 500;}continue;}//活三(一): 1 1 1 * 0if (GetLocalChessSign(judgeArray, i, j, direction, -1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -3) == maySign){if (GetLocalChessSign(judgeArray, i, j, direction, 1) == 0){score += 750;if (GetLocalChessSign(judgeArray, i, j, direction, -4) == 0){score += 3150;if (maySign != chessSign){score -= 300;}}}if ((GetLocalChessSign(judgeArray, i, j, direction, 1) == -maySign ||GetLocalChessSign(judgeArray, i, j, direction, 1) == -2)&& GetLocalChessSign(judgeArray, i, j, direction, -4) == 0){score += 500;}continue;}//活三(二): 1 1 1 0 *if (GetLocalChessSign(judgeArray, i, j, direction, -1) == 0 &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -3) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -4) == maySign){score += 350;continue;}// 死三: 1 1 * 1if (GetLocalChessSign(judgeArray, i, j, direction, -1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, 1) == maySign){score += 600;if (GetLocalChessSign(judgeArray, i, j, direction, -3) == 0 &&GetLocalChessSign(judgeArray, i, j, direction, 2) == 0){score += 3150;continue;}else if ((GetLocalChessSign(judgeArray, i, j, direction, -3) == -maySign ||GetLocalChessSign(judgeArray, i, j, direction, -3) == -2) &&(GetLocalChessSign(judgeArray, i, j, direction, 2) == -maySign ||GetLocalChessSign(judgeArray, i, j, direction, 2) == -2)){continue;}else{score += 700;continue;}}//活二数量(twoNum)if (GetLocalChessSign(judgeArray, i, j, direction, -1) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -2) == maySign &&GetLocalChessSign(judgeArray, i, j, direction, -3) != -maySign &&GetLocalChessSign(judgeArray, i, j, direction, 1) != -maySign){++twoNum;}//散棋数量(plyerNum)int plyerNum = 0;for (int si = -4; si <= 0; si++){int tempPlayerNum = 0;for (int sj = 0; sj <= 4; sj++){if (GetLocalChessSign(judgeArray, i, j, direction, si + sj) == maySign){tempPlayerNum++;}else if (GetLocalChessSign(judgeArray, i, j, direction, si + sj) == -maySign ||GetLocalChessSign(judgeArray, i, j, direction, si + sj) == -2){tempPlayerNum = 0;break;}}plyerNum += tempPlayerNum;}score += plyerNum * 15;}if (twoNum >= 2){score += 3000;if (chessSign != maySign){score -= 100;}}}return score;
}/*************************** 搜索将要落子位置的附近值 *******************************/
int GetLocalChessSign(const int* judgeArray, int labX, int labY, int direction, int distance)
{int x = labX, y = labY;switch (direction) {case 1:x = labX + distance;break;case 2:x = labX + distance;y = labY + distance;break;case 3:y = labY + distance;break;case 4:x = labX - distance;y = labY + distance;break;case 5:x = labX - distance;break;case 6:x = labX - distance;y = labY - distance;break;case 7:y = labY - distance;break;case 8:x = labX + distance;y = labY - distance;}if (x < 0 || y < 0 || x > 14 || y > 14){return -2;}return judgeArray[x * N + y];
}

HistoryFun.h

#pragma oncebool RegretMove();bool StorageHistoryArray(const int* currentArray);bool SaveHistory();bool loadHistory();

HistoryFun.cpp

#include<fstream>
#include"HistoryFun.h"
#include"HumanPlayer.h"
#include"RebotPlayer.h"using namespace std;const bool TurnOn = true;
const bool TurnOff = false;
bool turn;/***************************** 悔一步棋 *********************************/
bool RegretMove()
{int i, j;if (turn == TurnOn){if ((step = step - 2) < 0){return false;}}else{if ((step = step - 1) < 0){return false;}}for (i = 0; i < M; i++){for (j = 0; j < N; j++){AbstractPlayer::BoardArray[i * N + j] = AbstractPlayer::StorageBoardArray[(step * M + i) * N + j];AbstractPlayer::StorageBoardArray[(step * M + i) * N + j] = 0;}}AbstractPlayer::DisplayBoard();for (i = 0; i < M; i++){for (j = 0; j < N; j++){if (AbstractPlayer::BoardArray[i * N + j] == 12 || AbstractPlayer::BoardArray[i * N + j] == 13){AbstractPlayer::BoardArray[i * N + j] -= 2;}}}turn = TurnOff;return 1;
}/***************************** 内存存储当前布局 *********************************/
bool StorageHistoryArray(const int* currentArray)
{int i, j;for (i = 0; i < M; i++){for (j = 0; j < N; j++){AbstractPlayer::StorageBoardArray[(step * M + i) * N + j] = currentArray[i * N + j];}}++step;turn = TurnOn;return true;
}/***************************** 保存历史对战 *********************************/
bool SaveHistory()
{int i, j, k;ofstream outBuff;outBuff.open("history.txt", ios::out);outBuff << step << '\t';for (k = 0; k < step; k++){for (i = 0; i < M; i++){for (j = 0; j < N; j++){outBuff << AbstractPlayer::StorageBoardArray[(k * M + i) * N + j] << '\t';}}}outBuff.close();return true;
}/***************************** 加载历史对战 *********************************/
bool loadHistory()
{int i, j, k;ifstream inBuff;inBuff.open("history.txt", ios::in);if (!inBuff.is_open()){return false;}inBuff >> step;for (k = 0; k < step; k++){for (i = 0; i < M; i++){for (j = 0; j < N; j++){inBuff >> AbstractPlayer::StorageBoardArray[(k * M + i) * N + j];}}}inBuff.close();for (k = 0; k < step; k++){for (i = 0; i < M; i++){for (j = 0; j < N; j++){AbstractPlayer::BoardArray[i * N + j] = AbstractPlayer::StorageBoardArray[(k * M + i) * N + j];}}AbstractPlayer::DisplayBoard();cout << "     ";cout << "查看历史对战[按任意键继续]" << endl;cout << endl;system("pause");}return true;
}

http://www.ppmy.cn/news/563053.html

相关文章

基于C++的五子棋项目

一、五子棋项目介绍 1.游戏规则 五子棋是我国古代传统的黑白棋种之一&#xff0c;黑白双方依次落子&#xff0c;任意一方在棋盘上形成横向、纵向、斜向的连续相同颜色的五颗棋子的一方为胜。 2.完成界面显示 此次项目将不使用任何图形库进行开发&#xff0c;所有显示的内容都…

五子棋的设计与实现

术&#xff1a;Java等 摘要&#xff1a; 五子棋是一种两人对弈的纯策略型棋类游戏&#xff0c;非常容易上手&#xff0c;老少皆宜。为了更好的推广五子棋&#xff0c;研究简单的人工智能方式&#xff0c;运用Java开发五子棋游戏。主要包含了人机对战&#xff0c;棋盘初始化&…

五子棋-完美解决闪屏问题版-新增悔棋功能(C++实现)

在学C的时候&#xff0c;在做课设的时候做了一个五子棋课设&#xff0c;当时选这个就是觉得挺好玩的&#xff0c;想理解一下游戏是如何实现的&#xff0c;然后就入了这个坑。。。这个我没有用图形化界面来实现&#xff0c;完全是使用字符来实现。然后没有用图形化界面的最大的问…

C++、easyx组合的界面版五子棋(适合新手)

C、easyx组合的五子棋界面版&#xff08;适合新手&#xff09; 点击进入五子棋控制台版本 文章目录 C、easyx组合的五子棋界面版&#xff08;适合新手&#xff09;前言效果图一、游戏规则二、实现逻辑1.绘制棋盘2.落子2.1 鼠标坐标的获取2.2 绘制棋子 三、输赢判定算法四、其…

Windows编程 简易五子棋

简易五子棋&#xff0c;望指正 #include <windows.h> #include<commctrl.h> #define SIZE 20 UINT IDC_BUTTON1200; HWND hwnd; HPEN penbackCreatePen(PS_SOLID,1,RGB(0,0,0)); HPEN penback2CreatePen(PS_SOLID,1,RGB(255,255,255)); HBRUSH brushbackCreateSo…

课程设计-单机版五子棋游戏-Java

一. 项目简介&#xff1a; 五子棋是全国智力运动会竞技项目之一&#xff0c;是一种两人对弈的纯策略型棋类游戏。五子棋的玩法是&#xff1a;双方分别使用黑白两色的棋子&#xff0c;下在棋盘直线与横线的交叉点上&#xff0c;先形成五子连线者获胜。五子棋的棋具与围棋通用…

五子棋人机对弈代码——java版

算法是穷举递归法&#xff0c;只不过用java重新写了一遍 import java.awt.Color;import java.awt.Container;import java.awt.Graphics;import java.awt.event.ActionEvent;import java.awt.event.ActionListener;import java.awt.event.MouseEvent;import java.awt.event.Mou…

c++五子棋人机版

五子棋游戏人机版&#xff1a; 电脑实现了阻挡对方连续的棋子 #include<iostream> #include<iomanip> using namespace std; #include<stdlib.h> #include<time.h> /*class PerPlayer { public:int m;int n;int person(); }; PerPlayer pp;class CompP…