连连看核心算法oc实现

在学习ios开发的过程中,用一般的方式用oc写了一个练练看的小游戏,没有用到cocos2d编程.自己做的思路如下:

程序的关键在于判断用户连续点击的两个图案能否消除。两个图片可以消除的条件有两个: 

(1) 图案相同 

(2) 图案间连线的转角数不得超过2

所以连通的算法分为:

(1) 直连型 (2) 一个拐角连通 (3) 两个拐角连通

将图片所在的view看为一个棋盘,然后根据棋盘的行列进行相关的判断.

首先探讨下直连型:分为水平直连(横坐标相等)和竖直直连(纵坐标相等),且两者之间没有其他的图案.

连连看算法 python 连连看算法思路_i++

连连看算法 python 连连看算法思路_i++_02

可以看出A -- B横纵向连通,oc代码如下:(dict是根据图案的坐标,判断该位置是否有图案,有的话值为1,否则为0)

/** 1.直连型 */
 - (BOOL)verticalWith:(CGPoint)pointA point:(CGPoint)pointB{
      
      int rowA = [[NSString stringWithFormat:@"%f",pointA.x] intValue];
      int colA = [[NSString stringWithFormat:@"%f",pointA.y] intValue];
      
      int rowB = [[NSString stringWithFormat:@"%f",pointB.x] intValue];
      int colB = [[NSString stringWithFormat:@"%f",pointB.y] intValue];
      
     //1.直连型
     if (rowA == rowB){//同一行
         int minCol = colA < colB ? colA : colB;//最小列号
         int maxCol = colA > colB ? colA : colB;//最大列号
         
         for (int j = minCol+1; j<maxCol; j++) {
             CGPoint point = CGPointMake(rowA, j);
             NSString *key = NSStringFromCGPoint(point);
             if ([self.dict[key] intValue] != 0) {//两个图案之间存在其他的图案
                 return NO;
             }
         }
         return YES;
         
     }else if(colA == colB){//同一列
         int minRow = rowA < rowB ? rowA : rowB;//最小行号
         int maxRow = rowA > rowB ? rowA : rowB;//最大行号
         for (int i = minRow+1; i<maxRow; i++) {
             CGPoint point = CGPointMake(i, colA);
             NSString *key = NSStringFromCGPoint(point);
             if ([self.dict[key] intValue] != 0) {//两个图案之间存在其他的图案
                 return NO;
            }
         }
        return YES;
     } else{
         return NO;
     }
 }

 

 

2. 一折连通:其实相当于两个图片划出一个矩形,这两个图片是一对对角顶点,另外两个顶点如果可以同时和这两个棋子直连,那就说明可以"一折连通"。

连连看算法 python 连连看算法思路_调用函数_03

找出C,D两点,然后判断C,D处是否有图案,若存在图案直接返回no.如果其中一个没有,判断与A,B是否连通
/** 2.一折型 */
- (BOOL)oneCorner:(CGPoint)pointA button:(CGPoint)pointB{
    
    //找出拐角点的坐标
    CGPoint pointC = CGPointMake(pointA.x, pointB.y);
    
    CGPoint pointD = CGPointMake(pointB.x, pointA.y);

    //判断C点是否有元素
    if ([self.dict[NSStringFromCGPoint(pointC)] intValue] == 0) {
        
        return [self verticalWith:pointA point:pointC] && [self verticalWith:pointC point:pointB];
    }
    //判断D点是否有元素
    if ([self.dict[NSStringFromCGPoint(pointD)] intValue] == 0) {
        
        return [self verticalWith:pointA point:pointD] && [self verticalWith:pointD point:pointB];
    }
    
    //其他情况
    return NO;
}

3 两折连通:判断图案A与图案B能否经过有两个转角的路径连通,实质上可以转化为判断能否找到一个点C,这个C点与A可以直线连通,且C与B可以通过有一个转角的路径连通。若能找到这样一个C点,那么A与B就可以经过有两个转角的路径连通 。 

  判断是否经两个转角连通的算法需要做两个方向上的扫描:水平扫描和垂直扫描。 

  水平扫描。如下图所示,为了判断A,B能否通过2个转角连通,则从A开始在水平方向上向左右扫描,并判断经过的点能否与B点经过1个转角连通。显然C点能与B点经1个转角连通,故A,B能经2个转角连通。

连连看算法 python 连连看算法思路_i++_04

垂直扫描。如下图所示,为了判断A,B能否通过2个转角连通,则从A开始在垂直方向上下扫描,并判断经过的点能否与B点经过1个转角连通。显然C点能与B点经1个转角连通,故A,B能经2个转角连通。

连连看算法 python 连连看算法思路_i++_05

oc代码:

 

/** 3.两折型 */
/* 判断A和B是否两折连通
 * (1) 水平方向: 从A水平方向左右扫描,并判断经过的点能否与B通过1折连通;
 * (2) 垂直方向: 从A垂直方向上下扫描,并判断经过的点能否与B通过1折连通;
 */
- (BOOL)twoCorner:(CGPoint)pointA button:(CGPoint)pointB maxRow:(int)maxRow maxCol:(int)maxCol{
    //得到A点的所在行列
    int rowA = [[NSString stringWithFormat:@"%f",pointA.x] intValue];
    int colA = [[NSString stringWithFormat:@"%f",pointA.y] intValue];
    
    //初始化C点
    CGPoint pointC = CGPointMake(0, 0);
    
    // 1. 水平方向(列)
    // 1.1 左
    if (colA != 0){//A点不在最左边
        for (int i = colA - 1; i >= 0; i--) {
            pointC = CGPointMake(rowA, i);
            //判断C点是否为空
            if ([self.dict[NSStringFromCGPoint(pointC)] intValue] == 0) {
                if ([self oneCorner:pointC button:pointB]){
                    return YES;
                }
            }else{//不为空的话,直接跳出
                break;
            }
        }
    }

    // 1.2 右
    if (colA != maxCol - 1){//A点不在最右边
        for (int i = colA + 1; i < maxCol; i++) {
            pointC = CGPointMake(rowA, i);
            //判断C点是否为空
            if ([self.dict[NSStringFromCGPoint(pointC)] intValue] == 0) {
                if ([self oneCorner:pointC button:pointB]){
                    return YES;
                }
            }else{//不为空的话,直接跳出
                break;
            }
        }
    }
    
    // 2. 垂直方向(行)
    // 2.1 上
    if (rowA != 0){//A点不在最上边
        for (int i = rowA - 1; i >= 0; i--) {
            pointC = CGPointMake(i, colA);
            //判断C点是否为空
            if ([self.dict[NSStringFromCGPoint(pointC)] intValue] == 0) {
                if ([self oneCorner:pointC button:pointB]){
                    return YES;
                }
            }else{//不为空的话,直接跳出
                break;
            }
        }
    }
    
    // 2.2 下
    if (rowA != maxRow - 1){//A点不在最下边
        for (int i = rowA + 1; i < maxRow; i++) {
            pointC = CGPointMake(i, colA);
            //判断C点是否为空
            if ([self.dict[NSStringFromCGPoint(pointC)] intValue] == 0) {
                if ([self oneCorner:pointC button:pointB]){
                    return YES;
                }
            }else{//不为空的话,直接跳出
                break;
            }
        }
    }

    //其他情况
    return NO;
}

 

总的调用函数来调用:

/** 1.直连型 */
        if ([self verticalWith:pointA point:pointB]) {//是否存在直线,存在返回yes
            //连接成功
            return YES;
        }
        
        /** 2.一折型 */
        if([self oneCorner:pointA button:pointB]){
            return YES;
        }else{
            /** 3.两折型 */
            return [self twoCorner:pointA button:pointB maxRow:rowCount maxCol:columnCount];
        }

 

参考博客

参考博客

 

不积跬步,无以至千里;不积小流,无以成江海。