五子棋是程序猿比较熟悉的一款小游戏,相信很多人大学时期就用多种语言写过五子棋小游戏.笔者工作闲暇之余,试着用OC实现了一下,在这里给大家分享一下.有不足之处,欢迎大家提供建议和指点!!!GitHub源码链接https://github.com/HelloYeah/Gomoku-Game

先上效果图

  • 功能展示

    1.gif

  • 初高级棋盘切换效果

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAIAAACQd1PeAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyBpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMC1jMDYwIDYxLjEzNDc3NywgMjAxMC8wMi8xMi0xNzozMjowMCAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNSBXaW5kb3dzIiB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOkJDQzA1MTVGNkE2MjExRTRBRjEzODVCM0Q0NEVFMjFBIiB4bXBNTTpEb2N1bWVudElEPSJ4bXAuZGlkOkJDQzA1MTYwNkE2MjExRTRBRjEzODVCM0Q0NEVFMjFBIj4gPHhtcE1NOkRlcml2ZWRGcm9tIHN0UmVmOmluc3RhbmNlSUQ9InhtcC5paWQ6QkNDMDUxNUQ2QTYyMTFFNEFGMTM4NUIzRDQ0RUUyMUEiIHN0UmVmOmRvY3VtZW50SUQ9InhtcC5kaWQ6QkNDMDUxNUU2QTYyMTFFNEFGMTM4NUIzRDQ0RUUyMUEiLz4gPC9yZGY6RGVzY3JpcHRpb24+IDwvcmRmOlJERj4gPC94OnhtcG1ldGE+IDw/eHBhY2tldCBlbmQ9InIiPz6p+a6fAAAAD0lEQVR42mJ89/Y1QIABAAWXAsgVS/hWAAAAAElFTkSuQmCC" alt="" data-ratio="1.7783783783783784" data-src="http://mmbiz.qpic.cn/mmbiz_gif/g4uoJOMA38JlicjXTyoGhdicmPibLWUyOMURtPo1jT0fabaIWdkj1JuJSfDFBvuPY2KDQuqZ7DNphmwLQQZJ3Anxw/0?wx_fmt=gif" data-type="gif" data-w="555" />

    2.gif

实现思路及主要代码详解

1.绘制棋盘

利用Quartz2D绘制棋盘.代码如下

- (void)drawBackground:(CGSize)size{    
   self.gridWidth = (size.width - 2 * kBoardSpace) / self.gridCount;    //1.开启图像上下文    
   UIGraphicsBeginImageContext(size);    //2.获取上下文    
   CGContextRef ctx = UIGraphicsGetCurrentContext();    CGContextSetLineWidth(ctx, 0.8f);    //3.1 画16条竖线    
   for (int i = 0; i <= self.gridCount; i ++) {        
       CGContextMoveToPoint(ctx, kBoardSpace + i * self.gridWidth , kBoardSpace);        CGContextAddLineToPoint(ctx, kBoardSpace + i * self.gridWidth , kBoardSpace + self.gridCount * self.gridWidth);    
   }    //3.1 画16条横线    
   for (int i = 0; i <= self.gridCount; i ++) {        
       CGContextMoveToPoint(ctx, kBoardSpace, kBoardSpace  + i * self.gridWidth );        CGContextAddLineToPoint(ctx, kBoardSpace + self.gridCount * self.gridWidth , kBoardSpace + i * self.gridWidth);    
   }    
   CGContextStrokePath(ctx);    //4.获取生成的图片    
   UIImage *image=UIGraphicsGetImageFromCurrentImageContext();    //5.显示生成的图片到imageview    
   UIImageView * imageView = [[UIImageView alloc]initWithImage:image];    
   [self addSubview:imageView];    
   UIGraphicsEndImageContext();
}
2.点击棋盘落子

1.根据落子位置求出该棋子的行号与列号.
2.判断落子位置是否已经有棋子,有则不能下.如果没有,将棋子保存在字典中,以列号和行号组合起来的字符串为key值.
 代码如下

//点击棋盘,下棋
- (void)tapBoard:(UITapGestureRecognizer *)tap{    
   CGPoint point = [tap locationInView:tap.view];    //计算下子的列号行号    
   NSInteger col = (point.x - kBoardSpace + 0.5 * self.gridWidth) / self.gridWidth;    NSInteger row = (point.y - kBoardSpace + 0.5 * self.gridWidth) / self.gridWidth;    NSString * key = [NSString stringWithFormat:@"%ld-%ld",col,row];    if (![self.chessmanDict.allKeys containsObject:key]) {        UIView * chessman = [self chessman];        
   chessman.center = CGPointMake(kBoardSpace + col * self.gridWidth, kBoardSpace + row * self.gridWidth);        
   [self addSubview:chessman];        
   [self.chessmanDict setValue:chessman forKey:key];        self.lastKey = key;        //检查游戏结果        
   [self checkResult:col andRow:row andColor:self.isBlack];        self.isBlack = !self.isBlack;    
   }
}
3.检测游戏结果

每落一个棋子就要多游戏结果进行一次检查,判断四个方向上是否有大于等于5个同色的棋子连成一线,有则提示游戏输赢结果,无则游戏继续.算法为,从当前棋子位置向前遍历,直到遇到与自己不同色的棋子,累加同色棋子的个数,再往后遍历,直到遇到与自己不同色的棋子,累加同色棋子的个数.得到该方向相连同色棋子的总个数
代码如下

//判断是否大于等于五个同色相连
- (BOOL)checkResult:(NSInteger)col andRow:(NSInteger)row andColor:(BOOL)isBlack andDirection:(GmkDirection)direction{    
   if (self.sameChessmanArray.count >= 5) {        
       return YES;    
   }    
   UIColor * currentChessmanColor = [self.chessmanDict[[NSString stringWithFormat:@"%ld-%ld",col,row]] backgroundColor];    
   [self.sameChessmanArray addObject:self.chessmanDict[self.lastKey]];    
   switch (direction) {        //水平方向检查结果        
       case GmkHorizontal:{            //向前遍历            
           for (NSInteger i = col - 1; i > 0; i --) {                
               NSString * key = [NSString stringWithFormat:@"%ld-%ld",i,row];                
               if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor) break;                
               [self.sameChessmanArray addObject:self.chessmanDict[key]];            
           }            //向后遍历            
           for (NSInteger i = col + 1; i < kGridCount; i ++) {                NSString * key = [NSString stringWithFormat:@"%ld-%ld",i,row];                                if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor) break;                
               [self.sameChessmanArray addObject:self.chessmanDict[key]];            
           }            
           if (self.sameChessmanArray.count >= 5) {                
               [self alertResult];                
               return YES;            
           }            
           [self.sameChessmanArray removeAllObjects];          
      }            
           break;        
      case GmkVertical:{            //向前遍历            
          for (NSInteger i = row - 1; i > 0; i --) {                NSString * key = [NSString stringWithFormat:@"%ld-%ld",col,i];                if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor) break;                
              [self.sameChessmanArray addObject:self.chessmanDict[key]];            
           }            //向后遍历            
           for (NSInteger i = row + 1; i < kGridCount; i ++) {                NSString * key = [NSString stringWithFormat:@"%ld-%ld",col,i];                if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor) break;                
               [self.sameChessmanArray addObject:self.chessmanDict[key]];            
           }            
           if (self.sameChessmanArray.count >= 5) {                
               [self alertResult];                
               return YES;            
           }            
           [self.sameChessmanArray removeAllObjects];          
      }            
          break;        
       case GmkObliqueDown:{            //向前遍历            
          NSInteger j = col - 1;            
          for (NSInteger i = row - 1; i >= 0; i--,j--) {                NSString * key = [NSString stringWithFormat:@"%ld-%ld",j,i];                if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor || j < 0) break;                
                  [self.sameChessmanArray addObject:self.chessmanDict[key]];            
          }            //向后遍历            
          j = col + 1;            
          for (NSInteger i = row + 1 ; i < kGridCount; i++,j++) {                NSString * key = [NSString stringWithFormat:@"%ld-%ld",j,i];                if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor || j > kGridCount) break;                
              [self.sameChessmanArray addObject:self.chessmanDict[key]];            
          }            
          if (self.sameChessmanArray.count >= 5) {                
              [self alertResult];                
              return YES;            
           }            
           [self.sameChessmanArray removeAllObjects];           }            
           break;        
       case GmkObliqueUp:{            //向前遍历            
           NSInteger j = col + 1;            
           for (NSInteger i = row - 1; i >= 0; i--,j++) {                NSString * key = [NSString stringWithFormat:@"%ld-%ld",j,i];                if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor || j > kGridCount) break;                
               [self.sameChessmanArray addObject:self.chessmanDict[key]];            
            }            //向后遍历            
            j = col - 1;            
           for (NSInteger i = row + 1 ; i < kGridCount; i++,j--) {                NSString * key = [NSString stringWithFormat:@"%ld-%ld",j,i];                if (![self.chessmanDict.allKeys containsObject:key] || [self.chessmanDict[key] backgroundColor] != currentChessmanColor || j < 0) break;                
                [self.sameChessmanArray addObject:self.chessmanDict[key]];            
            }            
            if (self.sameChessmanArray.count >= 5) {                
                [self alertResult];                
                return YES;            
             }            
             [self.sameChessmanArray removeAllObjects];          
        }            
            break;    
    }    
   return NO;
}
4.对外提供,重新开始,悔棋,切换初高级棋盘的三个接口

重新开始

- (void)newGame{    
   self.isOver = NO;    
   self.lastKey = nil;    
   [self.sameChessmanArray removeAllObjects];    
   self.userInteractionEnabled = YES;    
   [self.chessmanDict removeAllObjects];    
   for (UIView * view in self.subviews) {        
       if ([view isKindOfClass:[UIImageView class]]) {            
           continue;        
       }        
       [view removeFromSuperview];    
    }    
    self.isBlack = NO;
}

悔棋

//撤回至上一步棋
- (void)backOneStep:(UIButton *)sender{    
   if(self.isOver) return;    
       if (self.lastKey == nil) {        
       sender.enabled = NO;        
       CGFloat width = SCREEN_WIDTH * 0.4 * SCREEN_WIDTH_RATIO;        UIView * tip = [[UIView alloc]initWithFrame:CGRectMake(0, 0, width, 0.6 * width)];         tip.backgroundColor = [UIColor colorWithWhite:1 alpha:0.8];        
       tip.layer.cornerRadius = 8.0f;        
       [self addSubview:tip];        
       tip.center = CGPointMake(self.width * 0.5, self.height * 0.5);        UILabel * label = [[UILabel alloc]init];        
       label.text = self.chessmanDict.count > 0 ? @"只能悔一步棋!!!" : @"请先落子!!!";        
       label.font = [UIFont systemFontOfSize:15];        
       [label sizeToFit];        
       label.center = CGPointMake(tip.width * 0.5, tip.height * 0.5);        
       [tip addSubview:label];        
       self.userInteractionEnabled = NO;        
       dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{            self.userInteractionEnabled = YES;            
       sender.enabled = YES;            
       [tip removeFromSuperview];          
       });        
       return;    
   }    
   [self.chessmanDict removeObjectForKey:self.lastKey];    
   [self.subviews.lastObject removeFromSuperview];    
   self.isBlack = !self.isBlack;    
   self.lastKey = nil;
}

切换初高级键盘

//改变键盘级别
- (void)changeBoardLevel{    
     for (UIView * view in self.subviews) {        
         [view removeFromSuperview];    
   }    
   [self newGame];    
   self.isHighLevel = !self.isHighLevel;    
   [self drawBackground:self.bounds.size];
}

Demo中的一个小技巧

用字典存放棋子,以棋子的列号和行号组合起来的字符串为key值,value值为棋子view.这样处理,在判断某行某列是否有棋子就非常简单了.
GitHub源码地址https://github.com/HelloYeah/Gomoku-Game

感谢分享

五子棋-b的更多相关文章

  1. 自己写的HTML5 Canvas + Javascript五子棋

    看到一些曾经只会灌水的网友,在学习了前端之后,已经能写出下载量几千几万的脚本.样式,帮助大众,成为受欢迎的人,感觉满羡慕的.我也想学会前端技术,变得受欢迎呀.于是心血来潮,开始学习前端知识,并写下了这 ...

  2. java swing 双人五子棋源代码

    import java.awt.Color; import java.awt.Font; import java.awt.Graphics; import java.awt.Toolkit; impo ...

  3. HTML5 五子棋 - JS/Canvas 游戏

    背景介绍 因为之前用c#的winform中的gdi+,java图形包做过五子棋,所以做这个逻辑思路也就驾轻就熟,然而最近想温故html5的canvas绘图功能(公司一般不用这些),所以做了个五子棋,当 ...

  4. [收藏]C++简单五子棋

    #include<iostream> #include<iomanip> using namespace std; ; //棋盘行数 ; //棋盘列数 char p[X][Y] ...

  5. jQuery网页版五子棋小游戏源码下载

    体验效果:http://hovertree.com/texiao/game/4/ 网页五子棋源代码: <!DOCTYPE html> <html> <head> & ...

  6. js+html5双人五子棋(源码下载)

    代码如下: <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" c ...

  7. jquery在线五子棋

    在线五子棋试玩地址:http://keleyi.com/game/12/ 以下是完整代码,保存到html文件打开也可以玩: <!DOCTYPE html> <html> < ...

  8. 五子棋AI清月连珠开源

    经过差不多两年的业余时间学习和编写,最近把清月连珠的无禁手部分完善得差不多了.这中间进行了很多思考,也有很多错误认识,到现在有一些东西还没有全面掌握,所以想通过开源于大家共同交流. 最近一直发表一些五 ...

  9. PyQt写的五子棋

    技术路线 GUI的实现 使用PyQt技术作为基础.PyQt是一个支持多平台的客户端开发SDK,使用它实现的客户端可以运行在目前几乎所有主流平台之上. 使用PyQt,Qt设计器实现UI,通过pyuic4 ...

  10. 浅析基本AI五子棋算法

    五子棋是所有棋类博弈中比较简单的了,这里介绍的也只是一种非常基本的AI策略.其实,包括之前的AI贪吃蛇,感觉这两个AI其实体现的都是一种建模思想,把一个现实中的问题模型化,抽象化,得到其一般特征,再设 ...

随机推荐

  1. sendStickyBroadcast和sendStickyOrderedBroadcast

    sendStickyBroadcast和sendStickyOrderedBroadcast发出的广播会一直滞留(等待),以便有人注册这则广播消息后能尽快的收到这条广播.其他功能与sendBroadc ...

  2. careercup-数学与概率 7.6

    7.6 在二维平面上,有一些点,请找出经过点数最多的那条线. 解法: 类似于leetcode:Max Points on a Line 我们只需在任意两点之间“画”一条无限长的直线(也即不是线段),并 ...

  3. oracle学习----行级锁的理解

    通过实验来理解行级锁的发生 1.创建需要的表 SQL> conn / as sysdba已连接.SQL> create table dept as select * from scott. ...

  4. 深入理解计算机系统第二版习题解答CSAPP 2.3

    填写空白.单字节可以用两个十六进制数表示. 十进制 二进制 十六进制 0 0000 0000 0x00 167 1010 0111 0xA7 62 0011 1110 0x3E 188 1011 11 ...

  5. 7.Composer的安装和使用

    1.安装Composer: 局部安装 要真正获取 Composer,我们需要做两件事.首先安装 Composer (同样的,这意味着它将下载到你的项目中): curl -sS https://getc ...

  6. CentOS 6.3 安装ATI显卡驱动

    环境: centos 6.3  内核(GNU貌似大家都知道) 显卡:ATI Mobility Radeon HD 3470 安装显卡流程: 1.官网下载最新驱动 http://support.amd. ...

  7. Environment variable:"PATH" 状态 失败

    问题截图: 问题内容: 未能满足某些最低安装要求.请复查并修复下表中列出的问题,然后重新检查系统. Checks    Environment Variable: "PATH"  ...

  8. T-SQL基础 (子查询,连接查询,交叉查询,事务|| 笔记0807)

    一: A.子查询: 1.select 字段名 from table where 字段名=(select 字段名 from table 条件)  //只能做1个匹配 2.select 字段名 from ...

  9. 解决OOM小记

    跟猜想的一样是OOM.一回来遇一不怎么熟悉的sb,给我气的....算了.....哥哥也是种种原因回的合肥.继续看问题. 这个地方的界面是这样的 划红线的地方是三个LinearLayout,每次oncl ...

  10. 学习笔记_Java get和post区别(转载_GET一般用于获取/查询资源信息,而POST一般用于更新资源信息)

    转载自:[hyddd(http://www.cnblogs.com/hyddd/)] 总结一下,      Get是向服务器发索取数据的一种请求      而Post是向服务器提交数据的一种请求,在F ...