原来的代码全部删除,进行了深层次重构,得其意而忘其言。得意之处有二:

1.关于显示

以 StoneSize 属性为依托,在 set 中加了一句:Width = Height = m_StoneSize * 19;以此来控制棋盘大小。所有的对象在 Init() 方法中创建,而具体的渲染在 Redraw() 方法中完成。这种创建与重绘相分开的办法,使调整大小时进行重绘更简单易行。这两个方法的代码如下:

 void Init()
{
// 线
for (int i = ; i < ; i++) {
m_LinesH[i] = new Line();
m_LinesH[i].Stroke = Brushes.Black;
m_Canvas.Children.Add(m_LinesH[i]); m_LinesV[i] = new Line();
m_LinesV[i].Stroke = Brushes.Black;
m_Canvas.Children.Add(m_LinesV[i]);
} // 星
for (int j = ; j < ; j++) {
for (int i = ; i < ; i++) {
m_Stars[i, j] = new Ellipse();
m_Stars[i, j].Fill = Brushes.Black;
m_Canvas.Children.Add(m_Stars[i, j]);
}
} for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
m_Stones[i, j] = new Ellipse();
m_Stones[i, j].Visibility = Visibility.Hidden;
m_Canvas.Children.Add(m_Stones[i, j]); m_Numbers[i, j] = new TextBlock();
m_Numbers[i, j].Background = Brushes.Transparent;
m_Numbers[i, j].Visibility = Visibility.Hidden;
m_Canvas.Children.Add(m_Numbers[i, j]); m_Steps[i, j] = new Step();
m_Steps[i, j].Row = i;
m_Steps[i, j].Col = j;
m_EmptySteps.Add(m_Steps[i, j]);
m_AllSteps.Add(m_Steps[i, j]);
}
} // 当前标志
m_CurrentRect.Visibility = System.Windows.Visibility.Hidden;
m_CurrentRect.Fill = Brushes.Red;
m_Canvas.Children.Add(m_CurrentRect); for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
Rectangle rect = new Rectangle();
rect.Visibility = System.Windows.Visibility.Hidden;
m_EmptyRects[i, j] = rect;
m_Canvas.Children.Add(m_EmptyRects[i,j]); }
}
}

Init()

 public void Redraw()
{
// 画线
for (int i = ; i < ; i++) {
Line l = m_LinesH[i];
int y = i * StoneSize + StoneSize / ;
l.X1 = StoneSize / ;
l.Y1 = y;
l.X2 = * StoneSize - StoneSize / ;
l.Y2 = y; l = m_LinesV[i];
int x = i * StoneSize + StoneSize / ;
l.X1 = x;
l.Y1 = StoneSize / ;
l.X2 = x;
l.Y2 = * StoneSize - StoneSize / ;
} // 画星
for (int j = ; j < ; j++) {
for (int i = ; i < ; i++) {
Ellipse e = m_Stars[i, j];
e.Width = e.Height = StoneSize / ;
double left = * StoneSize + j * * StoneSize - StoneSize / - e.Width / ;
double top = * StoneSize + i * * StoneSize - StoneSize / - e.Height / ;
Canvas.SetLeft(e, left);
Canvas.SetTop(e, top);
}
} // Stones and Numbers
for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
var stone = m_Stones[i, j];
stone.Width = stone.Height = StoneSize;
Canvas.SetLeft(stone, j * StoneSize);
Canvas.SetTop(stone, i * StoneSize); ShowNumber(i, j, m_Steps[i, j].StepCount);
}
} // 点目标志
if (IsShowMesh)
for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
var rect = m_EmptyRects[i, j];
rect.Width = rect.Height = m_CurrentRect.Width;
double offset = (StoneSize - rect.Width) / 2.0;
Canvas.SetLeft(rect, j * StoneSize + offset);
Canvas.SetTop(rect, i * StoneSize + offset);
}
}
} public bool NextOne(int row, int col)
{
if (m_Steps[row, col].StoneColor != StoneColor.Empty)
return false;
if (m_BanOnce.Row == row && m_BanOnce.Col == col) {
return false;
}
m_BanOnce.Row = m_BanOnce.Col = -; DrawStep(row, col);
bool isBlack;
if (m_Steps[row, col].StoneColor == StoneColor.Black) {
m_BlackSteps.Add(m_Steps[row, col]);
isBlack = true;
} else {
m_WhiteSteps.Add(m_Steps[row, col]);
isBlack = false;
}
m_EmptySteps.Remove(m_Steps[row, col]); UpdateBlackBlocks();
UpdateWhiteBlocks();
if (isBlack) {
if (!UpdateDeadBlocks(m_WhiteBlocks))
UpdateDeadBlocks(m_BlackBlocks);
} else {
if (!UpdateDeadBlocks(m_BlackBlocks))
UpdateDeadBlocks(m_WhiteBlocks);
} MoveCurrentRect(); m_StepCount++; StoneColor selfColor = isBlack ? StoneColor.Black : StoneColor.White;
bool isKillSelf = m_DeadBlocks.ContainsKey(m_StepCount - )
&& m_DeadBlocks[m_StepCount - ].Steps.Count ==
&& m_DeadBlocks[m_StepCount - ].Steps[].StoneColor == selfColor;
if (isKillSelf) {
m_DeadBlocks.Remove(m_StepCount - );
BackOne();
return false;
} return true;
}

Redraw()

2.关于提子

以 LinkSteps()方法为依托,提子不再是上下左右一通乱吃了,而是采用集合的办法,只需看看一块棋有没有气即可。其代码如下:

 //   +
// + + + 与 step 相连的棋子,包含自身
// + 根据 color 参数决定是所有,同色,黑色,白色,还是空色。
List<Step> LinkSteps(Step step, StoneColor color = StoneColor.Empty)
{
List<Step> links = new List<Step>();
for (int i = -; i < ; i++) {
for (int j = -; j < ; j++) {
if (i == j || i == -j) {
continue;
}
if (InRange(step.Row + i, step.Col + j)) {
links.Add(m_Steps[step.Row + i, step.Col + j]);
}
}
}
links.Add(step);
if (color == StoneColor.All) {
return links;
} else {
links.RemoveAll(l => l.StoneColor != color);
return links;
}
}

LinkSteps()

当然,关于劫争,关于悔棋,非深入代码,不能明白。但 LinkSteps()是构成集合的基础。从集合的观点,研究围棋,相信比其他方法更为可行。

x01.Weiqi.8: 一点改进的更多相关文章

  1. x01.Weiqi.12: 定式布局

    定式 下一步当将定式保存到数据库中,如布局中的代码所示,但其初始的代码更有利于理解.以小飞挂为例: // 0 // + 0 0 // + // // + List<Pos> P_LuSta ...

  2. x01.Weiqi.10: 死活问题

    估计得不错,点目后,仅一个方法:UpdateMeshes5() 就完美解决了梅花六.刀把五.斗笠四.盘角曲四等死活问题.先来看看效果图: 其代码如下: void UpdateMeshes5(bool ...

  3. x01.Weiqi.9: 点目功能

    添加点目功能,虽不中,不远也.还是先看看截图吧. 确保其可行,再看一张: 其点目结果,还是比较令人满意的.这主要得益于多遍扫描,如编译器的词法分析阶段,下面的代码可以证明: private void ...

  4. x01.Weiqi.7: 调整重绘

    GitHub 谁方便谁拍,谁重要拍谁.在这个砖头满天飞的时代,一个好的生态显得尤为重要.  红颜小头发,要的很简单. 也许成绝唱,只因鱼断肠. 姚贝福娃的离去,除感叹人生无常外,活着做点有意义的事情, ...

  5. iScroll-5拉动刷新功能实现与iScroll-4上拉刷新的一点改进

    近来在学习移动设备的应用开发,接触了jQuery mobile,在网上查阅相关资料时发现一个叫”iScroll“的小插件.其实这个iScroll插件跟jQuery mobile没有多大关系,并不是基于 ...

  6. x01.Weiqi.11: 神来之笔

    在围棋中,一子两用,可谓妙手,而一子三用,则可称之为神来之笔.在解决征子问题时,一不小心,也来了个神来之笔,其代码如下: // 征子判断,p1, p2 为气,p2 为前进方向,p 为逃跑之子. boo ...

  7. Winform(C#.NET)自动更新组件的使用及部分功能实现(一点改进功能)

    接前两篇继续: Winform(C#.NET)自动更新组件的使用及部分功能实现 Winform(C#.NET)自动更新组件的使用及部分功能实现(续) 借鉴文章:http://www.cnblogs.c ...

  8. x01.Weiqi.13: 鼎力推荐

    鼎力推荐 : 点击后即可观看,小伙子讲的很有深度. 说到深度,自然离不了深度学习.AlphaGo 的横空出世,似乎很有学习的必要. MuGo: 点击下载后,发现是 python,自然免不了一番学习,好 ...

  9. CoreText精彩文字轮廓绘制动画的一点改进

    大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) 原文在: http://oleb.net/blog/2010/ ...

随机推荐

  1. 通过手动创建统计信息优化sql查询性能案例

    本质原因在于:SQL Server 统计信息只包含复合索引的第一个列的信息,而不包含复合索引数据组合的信息 来源于工作中的一个实际问题, 这里是组合列数据不均匀导致查询无法预估数据行数,从而导致无法选 ...

  2. 机器学习 1 regression

    Linear regerssion 线性回归 回归: stock market forecast f(过去10年股票起伏的资料) = 明天道琼指数点数 self driving car f(获取的道路 ...

  3. 一步一步开发Game服务器(二)完成登陆,聊天

    我知道这样的文章在博客园已经多的大家都不想看了,但是这是我的系列文章开始,请各位大神见谅了. 多线程,线程执行器,(详见),socket通信相关 (详见) 本人blog相关文章测试代码,示例,完整版s ...

  4. 故障恢复和恢复模式(Crash Recovery & Recovery Models)

    数据库的恢复模型是否影响故障恢复,在简单恢复模式里,你是否会丢失事务?在今天的文章里我想谈下这点,详细讨论下. 恢复模式(Recovery Models) 对于这个问题的最简单的答案是不会:恢复模型不 ...

  5. MySQL5.7不停业务将传统复制变更为GTID复制

      由于GTID的优势,我们需要将传统基于file-pos的复制更改为基于GTID的复制,如何在线变更成为我们关心的一个点,如下为具体的方法: 目前我们有一个传统复制下的M-S结构: port 330 ...

  6. C# 热敏打印机 Socket 网络链接 打印 图片 (二)

    IPAddress ip = IPAddress.Parse("192.168.1.212"); IPEndPoint iport = );//9100为小票打印机指定端口 Soc ...

  7. C#基础知识四之override和new的区别

    override override是派生类用来重写基类方法的.调用的派生类方法,如需调用基类方法用base关键字 override不能重写非虚方法或静态方法. override重写必须用abstrac ...

  8. 【Win10开发】如何在页面之间传值

    我们知道UWP是通过不同的页面来展示不同的内容的,那么我们该怎么进行页面之间的传值呢? 首先我们在MainPage里面写一个ListView来展示一些英文单词. List<English> ...

  9. 疯狂Android讲义 - 学习笔记(三)

    Android的事件处理 3.1 Android提供了两套事件处理机制:基于监听的事件处理.基于回调的事件处理. 3.2 基于监听的事件处理 3.2.1 监听的处理模型  主要涉及三类对象:Event ...

  10. “三巨头”有变化,BAT还能走多久?

    在腾讯市值超越阿里巴巴后,市场分析多数认为,当年的BAT“三巨头”时代已经彻底结束,进入了“双寡头”时代了 从对外投资来看,BAT不同的投资逻辑可以推测其战略方向 撰文/梁云风 时评员,关注财经与互联 ...