题意:
给你一个二维字符网格数组 grid ,大小为 m x n ,你需要检查 grid 中是否存在 相同值 形成的环。
一个环是一条开始和结束于同一个格子的长度 大于等于 4 的路径。对于一个给定的格子,你可以移动到它上、下、左、右四个方向相邻的格子之一,可以移动的前提是这两个格子有 相同的值 。
同时,你也不能回到上一次移动时所在的格子。比方说,环 (1, 1) -> (1, 2) -> (1, 1) 是不合法的,因为从 (1, 2) 移动到 (1, 1) 回到了上一次移动时的格子。
如果 grid 中有相同值形成的环,请你返回 true ,否则返回 false 。
样例1 输入:grid = [["a","a","a","a"],["a","b","b","a"],["a","b","b","a"],["a","a","a","a"]] 输出:true 解释:如下图所示,有 2 个用不同颜色标出来的环:
样例2
输入:grid = [["c","c","c","a"],["c","d","c","c"],["c","c","e","c"],["f","c","c","c"]] 输出:true 解释:如下图所示,只有高亮所示的一个合法环:
样例3
输入:grid = [["a","b","b"],["b","z","b"],["b","b","a"]] 输出:false
解题思路;
使用并查集判断环
class Solution {
private $fa = array();
/**
* @param String[][] $grid
* @return Boolean
*/
function containsCycle($grid) {
$n = count($grid);
$m = count($grid[0]);
$k = 0;
for($i = 0; $i < $n; $i++) {
for($j = 0; $j < $m; $j++) {
$this->fa[$k] = $i * $m + $j;
$k++;
}
}
for($i = 0; $i < $n; $i++) {
for($j = 0; $j < $m; $j++) {
if(($i+1 < $n) && $grid[$i][$j] == $grid[$i+1][$j]) {
if(!$this->merge($i*$m+$j, ($i+1)*$m+$j)) {
return true;
}
}
if(($j+1 < $m) && $grid[$i][$j] == $grid[$i][$j+1]) {
if(!$this->merge($i*$m+$j, $i*$m+$j+1)) {
return true;
}
}
}
}
return false;
}
function get($i) {
if($i != $this->fa[$i]) {
return $this->fa[$i] = $this->get($this->fa[$i]);
}
return $i;
}
function merge($i, $j) {
$x = $this->get($i);
$y = $this->get($j);
if($x != $y) {
$this->fa[$x] = $y;
return true; // 如果需要合并返回true
} else {
return false; // 已经在一个集合中,说明它是多余的边
}
}
}