本文是利用C#实现连连看的小例子,以供学习分享使用。如有不足之处,还望指正。

思路:

  1. 初始化布局(横竖十行十列,共100个单元格,每一个格一个按钮,背景图为水果图片,随机生成) 。
  2. 初始化对应棋盘(用二维数组表示【0表示空白,非0表示界面对象】)和页面相对应,同步操作。
  3. 判断点击的图片是否可以消掉(转化为二维数组【以水平方向,垂直方向,一个拐角,两个拐角的步骤进行判断】)。
  4. 如可以消掉,隐藏图片,增加分数。
  5. 时间限制,采用倒计时方式。

涉及知识点:

  1. 线程:Thread,后台运行时间控制【倒计时方式】。
  2. 界面闪烁:当界面中的控件较多,且有背景图时,界面就会出现闪烁【解决方式:1,双缓冲方式 2. 设置控件创建样式,统一刷新】。
  3. TableLayoutPanel:表示一个面板,它可以在一个由行和列组成的网格中对其内容进行动态布局【新增元素,设置行列,以及样式】。
  4. 资源文件:Resources 用于存放图片及其他资源。
  5. Button:FlatAppearance获取用于指示选中状态和鼠标状态的边框外观和颜色。

效果图

(一)【开始,初始化后,倒计时功能,停止功能】:

效果图(二)【时间结束】

核心算法

连连看核心算法代码如下:

     /// <summary>
/// 连连看帮助类
/// </summary>
public class LinkHelper
{
/// <summary>
/// 连连看,看板
/// </summary>
public int[,] LinkBoard { get; set; } /// <summary>
/// 连线成功事件
/// </summary>
public event EventHandler SucClick; /// <summary>
/// 连接失败事件
/// </summary>
public event EventHandler FailClick; private int col = ; public int Col
{
get
{
return col;
} set
{
col = value;
}
} private int row = ; public int Row
{
get
{
return row;
} set
{
row = value;
}
} /// <summary>
/// 尝试连线
/// </summary>
public void LinkLine(Point first, Point second)
{
EventArgs e = new EventArgs();
if (checkLink(first, second))
{
//连线成功
this.LinkBoard[first.X, first.Y] = ;
this.LinkBoard[second.X, second.Y] = ;
if (this.SucClick != null)
{
SucClick(this, e);
}
}
else {
//连线失败
if (this.FailClick != null)
{
FailClick(this, e);
}
}
} /// <summary>
/// 是否赋值
/// </summary>
/// <param name="p"></param>
/// <returns></returns>
public bool IsChecked(Point p)
{
bool flag = false;
if (p.X != - && p.Y != -)
{
flag = true;
}
return flag;
} #region 核心算法 /// <summary>
/// 判断是否连线成功
/// </summary>
/// <param name="a">第一个点击对象</param>
/// <param name="b">第二个点击对象</param>
/// <returns></returns>
private bool checkLink(Point a, Point b)
{
if (!Point.Equals(a, b))
{
if (this.LinkBoard[a.X, a.Y] == this.LinkBoard[b.X, b.Y])
{
if (a.X == b.X && horizon(a, b))
{
return true;
}
if (a.Y == b.Y && vertical(a, b))
{
return true;
}
if (oneCorner(a, b))
{
return true;
}
else
{
return twoCorner(a, b);
}
}
else {
//如果点击的不是同一个图案,直接返回false
return false;
}
}
else {
//如果点击的是同一个位置的图案,直接返回false;
return false;
}
} /// <summary>
/// 水平连线
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
/// <returns></returns>
private bool horizon(Point a, Point b)
{
int col_start = a.Y < b.Y ? a.Y : b.Y; //获取a,b中较小的y值
int col_end = a.Y < b.Y ? b.Y : a.Y; //获取a,b中较大的值 //遍历a,b之间是否通路,如果一个不是就返回false;
for (int i = col_start + ; i < col_end; i++)
{
if (this.LinkBoard[a.X, i] != )
{
return false;
}
}
return true;
} /// <summary>
/// 垂直连线
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
/// <returns></returns>
private bool vertical(Point a, Point b)
{
int row_start = a.X < b.X ? a.X : b.X;
int row_end = a.X < b.X ? b.X : a.X;
for (int i = row_start + ; i < row_end; i++)
{
if (this.LinkBoard[i, a.Y] != )
{
return false;
}
}
return true;
} /// <summary>
/// 一个拐角
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
/// <returns></returns>
private bool oneCorner(Point a, Point b)
{
Point c = new Point(b.X, a.Y);
Point d = new Point(a.X, b.Y);
//判断C点是否有元素
if (this.LinkBoard[c.X, c.Y] == )
{
bool path1 = horizon(b, c) && vertical(a, c);
return path1;
}
//判断D点是否有元素
if (this.LinkBoard[d.X, d.Y] == )
{
bool path2 = horizon(a, d) && vertical(b, d);
return path2;
}
else
{
return false;
}
} /// <summary>
/// 两个拐角
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
/// <returns></returns>
private bool twoCorner(Point a, Point b)
{
List<Line> ll = scan(a, b);
if (ll.Count == )
{
return false;
}
for (int i = ; i < ll.Count; i++)
{
Line tmpLine = ll[i];
if (tmpLine.direct == )
{ if (vertical(a, tmpLine.a) && vertical(b, tmpLine.b))
{
return true;
}
}
else if (tmpLine.direct == )
{
if (horizon(a, tmpLine.a) && horizon(b, tmpLine.b))
{
return true;
}
}
}
return false;
} /// <summary>
/// 扫描A与B之间的连接点组成的线
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
/// <returns></returns>
private List<Line> scan(Point a, Point b)
{
List<Line> linkList = new List<Line>();
//检测a点,b点的左侧是否能够垂直直连
for (int i = a.Y; i >= ; i--)
{
if (this.LinkBoard[a.X, i] == && this.LinkBoard[b.X, i] == && vertical(new Point(a.X, i), new Point(b.X, i)))
{
linkList.Add(new Line(new Point(a.X, i), new Point(b.X, i), ));
}
}
//检测a点,b点的右侧是否能够垂直直连
for (int i = a.Y; i < Col; i++)
{
if (this.LinkBoard[a.X, i] == && this.LinkBoard[b.X, i] == && vertical(new Point(a.X, i), new Point(b.X, i)))
{
linkList.Add(new Line(new Point(a.X, i), new Point(b.X, i), ));
}
}
//检测a点,b点的上侧是否能够水平直连
for (int j = a.X; j >= ; j--)
{
if (this.LinkBoard[j, a.Y] == && this.LinkBoard[j, b.Y] == && horizon(new Point(j, a.Y), new Point(j, b.Y)))
{
linkList.Add(new Line(new Point(j, a.Y), new Point(j, b.Y), ));
}
}
//检测a点,b点的下侧是否能够水平直连
for (int j = a.X; j < Row; j++)
{
if (this.LinkBoard[j, a.Y] == && this.LinkBoard[j, b.Y] == && horizon(new Point(j, a.Y), new Point(j, b.Y)))
{
linkList.Add(new Line(new Point(j, a.Y), new Point(j, b.Y), )); }
} return linkList;
} #endregion
}

关于连连看核心算法讲解,请参考链接

源码下载

C# 实现连连看功能的更多相关文章

  1. 软件设计之基于Java的连连看小游戏(三)——所有功能的实现

    新年快乐!期末接二连三的考试实在太忙了忘记连连看没有更新完,今天想要学习生信时才发现.所以这次直接把连连看所有功能全部放上. 在传统的连连看的基础上,我增加了上传头像的功能,即可以自行上传图片作为游戏 ...

  2. 新功能WBS

    项目名:连连看 组名:天天向上 组长:王森 组员:张政.张金生.林莉.胡丽娜 代码地址:HTTPS:https://git.coding.net/jx8zjs/llk.git SSH:git@git. ...

  3. 连连看的设计与实现——四人小组项目(NABCD)

    小组名称:天天向上 成员:王森.张政,张金生,栾骄阳 题目:连连看游戏 NABCD N(需求) 游戏最大的乐趣在于玩法,我们要想在众多的连连看游戏当中脱颖而出,就需要增加更多富有乐趣.吸引用户的玩法. ...

  4. 【好玩的应用】QQ连连看辅助工具

    自己学了这么久的C语言,但没有写出过什么可以用的东西来,总觉得心里不爽.这几天实在是不想干正事,在网上瞎逛逛,结果发现有人写了连连看的外挂.顿时觉得这很有意思啊.于是把代码下载下来,捣鼓了捣鼓.发现还 ...

  5. [置顶] ios 水果连连看游戏源码

    原创文章,转载请注明出处:http://blog.csdn.net/donny_zhang/article/details/9251917 demo功能:水果连连看游戏源码.iphone6.1 测试通 ...

  6. javascrpit开发连连看记录-小游戏

        工作之余,总想做点什么有意思的东西.但是苦于不知道做什么,也就一直没有什么动作.在一个午饭后,跟@jedmeng和@墨尘聊天过程中,发现可以写一些小东西来练练手,有以下几点好处:     1. ...

  7. cocos2d-x3.6 连连看连通画线

    我的博客:http://blog.csdn.net/dawn_moon 网上看到非常多人写的连连看,都没有画连线的实现.事实上要话连线挺简单的.cocos2d-x 提供了一个非常方便的绘图形的类.Dr ...

  8. 本人第一个android游戏《新连连看》上架

    经过艰苦奋战了几天,本人的第一个android游戏<新连连看>最终完毕了第一个版本号,比較简陋.另一部分功能保留没有开放.等第二个版本号再上.用的libgdx框架.可能不是非常出名,可是本 ...

  9. 使用WPF教你一步一步实现连连看

    使用WPF教你一步一步实现连连看(一) 第一步: 问题,怎样动态的建立一个10*10的grid(布局) for (int i = 0; i < 10; i++){ RowDefinition r ...

随机推荐

  1. C语言数组一种巧妙的使用方式

    作为计算机一种比较古老的语言,它并没有随着岁月老去,而是仍旧在整个领域发挥出耀眼的光芒,就像写作,有很多光芒万丈的句子值得我们去珍藏,今天就遇到了个比较巧妙的数组使用方法,做个记录,以供后续使用. # ...

  2. Kubernetes系列之Coredns and Dashboard介绍篇

    本次系列使用的所需部署包版本都使用的目前最新的或最新稳定版,安装包地址请到公众号内回复[K8s实战]获取 介绍 项目地址:https://github.com/coredns/coredns Core ...

  3. 第四课 VMP壳内爆破

    这一课用来演示的软件是文件巴士. 打开网页一搜索,可笑的是搜索到的结果都是破解版,想找个原版的倒费劲了. 好容易找到一个,下好一查壳,还没有... 行吧,自己加一个VMP壳开搞. 第一步 OD载入程序 ...

  4. tensorflow 1.0 学习:用CNN进行图像分类

    tensorflow升级到1.0之后,增加了一些高级模块: 如tf.layers, tf.metrics, 和tf.losses,使得代码稍微有些简化. 任务:花卉分类 版本:tensorflow 1 ...

  5. Oracle客户端连接数据库配置

    配置文件和路径 配置文件:tnsnames.ora 默认路径:%ORACLE_HOME%\network\admin\tnsnames.ora,%ORACLE_HOME%通常在环境变量中使用. 我的路 ...

  6. 从零开始学 Web 之 Vue.js(四)Vue的Ajax请求和跨域

    大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...

  7. 【深度学习与TensorFlow 2.0】入门篇

    注:因为毕业论文需要用到相关知识,借着 TF 2.0 发布的时机,重新捡起深度学习.在此,也推荐一下优达学城与 TensorFlow 合作发布的TF 2.0入门课程,下面的例子就来自该课程. 原文发布 ...

  8. SHELL脚本--shell数组基础

    bash&shell系列文章:http://www.cnblogs.com/f-ck-need-u/p/7048359.html 数组和变量的区别是:变量在内存中占用的空间是离散的,数组在内存 ...

  9. Go基础系列:数据类型转换(strconv包)

    Go不会对数据进行隐式的类型转换,只能手动去执行转换操作. 简单的转换操作 转换数据类型的方式很简单. valueOfTypeB = typeB(valueOfTypeA) 例如: // 浮点数 a ...

  10. Go标准库:Go template用法详解

    本文只介绍template的语法和用法,关于template包的函数.方法.template的结构和原理,见:深入剖析Go template. 入门示例 以下为test.html文件的内容,里面使用了 ...