棋盘和棋子采用GDI+画上去的。棋盘18*18。棋子是用DrawElipse画的,白棋和黑棋分两个List集合存储,方便判断五子连线的情况。
主要说一下,五子连线的思路,把集合按行和按列以及按正斜和反斜进行分组。然后在依次遍历被分组的集合,如果其中有一个集合出现了5子相连的情况就胜利。而五子相连的判断是:通过判断前一个和后一个棋子的距离是否是等于棋盘的格子宽度。按行列分组好判断,就 x y相等的分别分组。但是正斜和反斜的分组,我接用的是一元一次方程来做的。详细如代码所示:

private bool IsWin(List<Point> pt)
{
bool isWin = false;
//横
var yGroup=pt.GroupBy(p => p.Y);
foreach (var h in yGroup)
{
var lst = h.ToList().Select(a => a.X).ToList();
lst.Sort(); if (lst.Count < 5)
{
continue;
}
int hasCount = 1; for (int j = 0; j < lst.Count; j++)
{
var x=lst[j];
for (int m = j+1; m <lst.Count&&m< j+5; m++)
{ var xNext=lst[m]; if (xNext - x <= 30*(hasCount))
{
hasCount++;
}
else
{
hasCount = 1;
break;
} if (hasCount == 5)
{
return true;
} } if (hasCount != 5)
{
hasCount = 1;
}
} } //纵
var xGroup=pt.GroupBy(p => p.X);
foreach (var z in xGroup)
{
var lst = z.ToList().Select(a => a.Y).ToList();
lst.Sort(); if (lst.Count < 5)
{
continue;
}
int hasCount = 1; for (int j = 0; j < lst.Count; j++)
{
var y = lst[j];
for (int m = j + 1; m < lst.Count && m < j + 5; m++)
{ var yNext = lst[m]; if (yNext - y <= 30 * (hasCount))
{
hasCount++;
}
else
{
hasCount = 1;
break;
} if (hasCount == 5)
{
return true;
} } if (hasCount != 5)
{
hasCount = 1;
}
} } for (int b = -30*18; b < 30*18; b+=30)
{
var lst = pt.Where(a => -1 * a.Y == -1 * a.X + b).Select(a => a.X).ToList();
lst.Sort(); if (lst.Count < 5)
{
continue;
}
int hasCount = 1; for (int j = 0; j < lst.Count; j++)
{
var y = lst[j];
for (int m = j + 1; m < lst.Count && m < j + 5; m++)
{ var yNext = lst[m]; if (yNext - y <= 30 * (hasCount))
{
hasCount++;
}
else
{
hasCount = 1;
break;
} if (hasCount == 5)
{
return true;
} } hasCount = 1;
}
} //右斜 for (int mq=1;mq<18*18;mq++)
{
int b = mq * -10;
var lst = pt.Where(a =>-1*a.Y == a.X + b).Select(a => a.X).ToList();
lst.Sort(); if (lst.Count < 5)
{
continue;
}
int hasCount = 1; for (int j = 0; j < lst.Count; j++)
{
var y = lst[j];
for (int m = j + 1; m < lst.Count && m < j + 5; m++)
{ var yNext = lst[m]; if (yNext - y <= 30 * (hasCount))
{
hasCount++;
}
else
{
hasCount = 1;
break;
} if (hasCount == 5)
{
return true;
} } hasCount = 1;
}
}
return isWin; }
 
这是简单的适合人和人玩的方法。
目前,想加入人机,想采用极大极小alphabat剪枝的博弈算法来做机器人,但对我来说,有点复杂,肯定要花很多时间的,我没有信心完成,所以就没做了。还想做局域网对战,以前做过一次五子棋局域网的,记得是在18年的时候,是用button作为棋子,下个子要等很久,而且五连判断还没写好。比起那时候自己做的五子棋,这次这个要好很多了。
我发现在网络里,能发现无数个比自己厉害的人,特别是在B站,博客园什么的,很多厉害的人,他们在大学时候甚至才高中,初中,小学,就已经实现了我现在想实现的五子棋功能。我觉得,自己虽然已毕业多年,但在编程方面确实要下功夫才行。

C# winform GDI+ 五子棋 (一):基本界面和胜负判断的更多相关文章

  1. 五子棋(无AI winform gdi+)

    之前无意间在博客园看到一篇用深度学习玩马里奥的文章,于是就想做这个小东西来测试人工智能算法(准备用PYTHON的库,对神经网络的梦已经做了好多年了,但是太难了,一直懒得动它),本来是想用WPF做UI, ...

  2. c# winform编程之多线程ui界面资源修改总结篇【转】

    c# winform编程之多线程ui界面资源修改总结篇 单线程的winfom程序中,设置一个控件的值是很easy的事情,直接 this.TextBox1.value = "Hello Wor ...

  3. Winform GDI+

    什么是GDI+ GDI (Graphics Device Interface), 是属于绘图方面的 API (Application Programming Interface). 因为应用程序不能直 ...

  4. WinForm GDI+ 资料收集

    UI(User Interface)编程在整个项目开发过程中是个颇为重要的环节,任何好的解决方案若没有良好的用户界面呈现给最终用户,那么就算包含了最先进的技术也不能算是好程序.UI编程体现在两个方面, ...

  5. WinForm GDI+自定义控件总结(一)

    前言 由于项目的原因好久没写博客了,也正是项目的原因开始系统的学习WinForm,从而接触到自定义控件的开发.自定义控件的开发有一定的难度,对开发者要求比较高,需要了解Windows运行的机制,熟悉w ...

  6. C#winform自定义控件模拟设计时界面鼠标移动和调节大小、选中效果

    要想玩转Winform自定义控件需要对GDI+非常熟悉,对常用的控件有一些了解,好选择合适的基类控件来简化. 要点说明及代码 1)定义接口: using System; using System.Wi ...

  7. C# Winform设计运行时,界面模糊

    程序在Visual Studio设计的很清晰的菜单和界面,运行的时候菜单和控件上字体变得很模糊,界面大小也发生了变化 解决方法是:更改程序的配置文件,使程序运行时自动检测屏幕分辨率,在高分屏时禁用系统 ...

  8. C#-WinForm跨线程修改UI界面

    待解决的问题 在我做WinForm开发的过程中,经常会遇到耗时操作或阻塞操作.他们会引发软件的卡顿甚至假死,严重影响软件的使用. 因此,这类耗时或阻塞的操作一般都会使用异步的方式去执行,不影响主线程( ...

  9. Winform GDI+ 绘图一:绘制2D电池

    winform桌面软件开发,在工业控制领域的使用还是很广泛的,打算好好学习一下GDI+绘图.以前都是用别人的轮子,自己也打算封装一些工业控制领域常用的控件. 今天要将的是,利用缓动函数动态绘制电池. ...

  10. c# winform编程之多线程ui界面资源修改总结篇

    单线程的winfom程序中,设置一个控件的值是很easy的事情,直接 this.TextBox1.value = "Hello World!";就搞定了,但是如果在一个新线程中这么 ...

随机推荐

  1. GPT-3的训练一次成本约为140万美元

    训练GPT模型的成本非常高昂,因为它需要大量的计算资源和时间.具体来说,GPT-3的训练成本约为140万美元,对于一些更大的LLM模型,训练成本介于200万美元至1200万美元之间.此外,OpenAI ...

  2. GAN的实现和一些问题

    GAN的学习是一个二人博弈问题,最终目标是达到纳什平衡.对抗指的是生成网络和判别网络的互相对抗.生成网络尽可能生成逼真样本,判别网络则尽可能去判别该样本是真实样本,还是生成的假样本.示意图如下: 生成 ...

  3. sql 语句系列(计算的进阶)[八百章之第十六章]

    前言 介绍两个实用的sql查询语句. 1.计算平均数时候,去除最大值和最小值. 2.修改累计值. 计算平均数时候,去除最大值和最小值 sql server: select AVG(sal) from( ...

  4. 深入解析Rivest Cipher 4:理论与实践

    第一章:引言 密码学简介: 密码学是研究如何保护通信和信息安全的学科.它涉及加密算法.解密算法.密钥管理等内容,旨在确保信息在传输和存储过程中不被未经授权的人所获取或篡改.密码学可以分为对称加密和非对 ...

  5. huggingface vit训练CIFAR10数据集代码 ,可以改dataset训练自己的数据

    上代码,使用hugging face fineturn vit模型 自己写的代码 from transformers import ViTImageProcessor, ViTForImageClas ...

  6. flask通过线程池实现异步

    from flask import Flask from time import sleep from concurrent.futures import ThreadPoolExecutor # D ...

  7. 浅析Golang map的实现原理

    Golang中的map底层使用的数据结构是hash table,基本原理就和基础的散列表一致,重点是Golang在设计中采用了分桶(Bucket),每个桶里面支持多个key-value元素的这种思路, ...

  8. Go命令行工具cobra

    关于 Cobra 是 Go 的 CLI 框架.它包含一个用于创建功能强大的现代 CLI 应用程序的库,以及一个用于快速生成基于 Cobra 的应用程序和命令文件的工具. Cobra 由 Go 项目成员 ...

  9. 【笔记】Oracle使用笔记 0-sql injection&&&result of string concatenation is too long

    报错:数据库操作错误."27,34006/v1:0-sql injection(SQL注入) 出现这个报错的情况背景是使用后端函数进行前端SQL语句组合进行数据插入的时候的提示 不太清楚是因 ...

  10. 力扣628(java)-三个数的最大乘积(简单)

    题目: 给你一个整型数组 nums ,在数组中找出由三个数组成的最大乘积,并输出这个乘积. 示例 1: 输入:nums = [1,2,3]输出:6示例 2: 输入:nums = [1,2,3,4]输出 ...