2850. 将石头分散到网格图的最少移动次数

题目描述:

给你一个大小为 3 * 3 ,下标从 0 开始的二维整数矩阵 grid ,分别表示每一个格子里石头的数目。网格图中总共恰好有 9 个石头,一个格子里可能会有 多个 石头。

每一次操作中,你可以将一个石头从它当前所在格子移动到一个至少有一条公共边的相邻格子。

请你返回每个格子恰好有一个石头的 最少移动次数 。

示例 1:

输入:grid = [[1,1,0],[1,1,1],[1,2,1]]
输出:3
解释:让每个格子都有一个石头的一个操作序列为:
1 - 将一个石头从格子 (2,1) 移动到 (2,2) 。
2 - 将一个石头从格子 (2,2) 移动到 (1,2) 。
3 - 将一个石头从格子 (1,2) 移动到 (0,2) 。
总共需要 3 次操作让每个格子都有一个石头。
让每个格子都有一个石头的最少操作次数为 3 。

示例 2:

输入:grid = [[1,3,0],[1,0,0],[1,0,3]]
输出:4
解释:让每个格子都有一个石头的一个操作序列为:
1 - 将一个石头从格子 (0,1) 移动到 (0,2) 。
2 - 将一个石头从格子 (0,1) 移动到 (1,1) 。
3 - 将一个石头从格子 (2,2) 移动到 (1,2) 。
4 - 将一个石头从格子 (2,2) 移动到 (2,1) 。
总共需要 4 次操作让每个格子都有一个石头。
让每个格子都有一个石头的最少操作次数为 4 。

提示:

  • grid.length == grid[i].length == 3
  • 0 <= grid[i][j] <= 9
  • grid 中元素之和为 9

解题分析及思路:

方法:深度优先搜索

思路:

首先找到所有不符合规则的格子,即找到不为1的格子,分为两组:

  • 一组为所有石头数大于1的格子
  • 一组为所有石头数等于0的格子

那么,要做的是,将所有石头数大于1的格子,都移动到所有石头数等于0的格子中,使得每个格子都有石头且都为1。

利用深度优先搜索的思想,所有石头数为0的格子都可以从石头数大于1的格子中选择一个格子,并将其移动到石头数等于0的格子中,使得每个格子都有石头且都为1。

从所有的方案中,选择最小的移动次数。

import "math"

func minimumMoves(grid [][]int) int {
	var num1 [][]int
	var num2 [][]int
	for i := 0; i < len(grid); i++ {
		for j := 0; j < len(grid[i]); j++ {
			if grid[i][j] > 1 {
				num1 = append(num1, []int{i, j, grid[i][j]})
			} else if grid[i][j] == 0 {
				num2 = append(num2, []int{i, j, grid[i][j]})
			}
		}
	}
	var minCount = math.MaxInt
	var dfs func(count, zeroCount int)
	dfs = func(count, zeroCount int) {
		if zeroCount == 0 {
			minCount = min(minCount, count)
			return
		}
		for index1 := 0; index1 < len(num1); index1++ {
			for index2 := 0; index2 < len(num2); index2++ {
				if num1[index1][2] > 1 && num2[index2][2] == 0 {
					num1[index1][2]--
					num2[index2][2]++
					dfs(count+abs(num1[index1][0]-num2[index2][0])+abs(num1[index1][1]-num2[index2][1]), zeroCount-1)
					num1[index1][2]++
					num2[index2][2]--
				}
			}
		}
		return
	}
	dfs(0, len(num2))
	if minCount == math.MaxInt {
		return 0
	}
	return minCount
}

func min(num1, num2 int) int {
	if num1 < num2 {
		return num1
	}
	return num2
}

func abs(num int) int {
	if num > 0 {
		return num
	}
	return -num
}

复杂度:

  • 时间复杂度:O(2n), 其中 n 为网格矩阵的元素个数。
  • 空间复杂度:O(n)

执行结果:

  • 执行耗时:108 ms,击败了16.67% 的Go用户
  • 内存消耗:2.7 MB,击败了50.00% 的Go用户

通过次数 16.1K 提交次数 27.7K 通过率 58.1%

Related Posts

1026. 节点与其祖先之间的最大差值

## 题目描述:给定二叉树的根节点 root,找出存在于 不同 节点 A 和 B 之间的最大值 V,其中 V = |A.val - B.val|,且 A 是 B 的祖先。(如果 A 的任何子节点之一为 B,或者 A 的任何子节点是 B 的祖先,那么我们认为 A 是 B 的祖先)示例 1: ![](/img/leetcode/1026节点与其祖先之间的最大差值/tmp-tre

read more

1038. 从二叉搜索树到更大和树

## 题目描述:给定一个二叉搜索树 root (BST),请将它的每个节点的值替换成树中大于或者等于该节点值的所有节点值之和。提醒一下, 二叉搜索树 满足下列约束条件:- 节点的左子树仅包含键 小于 节点键的节点。 - 节点的右子树仅包含键 大于 节点键的节点。 - 左右子树也必须是二叉搜索树。*示例 1:***![](/img/leetcode/

read more

105. 从前序与中序遍历序列构造二叉树

## 题目描述:给定两个整数数组 preorder 和 inorder ,其中 preorder 是二叉树的先序遍历, inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。示例 1: ``` 输入: preorder = [3,9,20,15,7], inorder = [

read more

106. 从中序与后序遍历序列构造二叉树

## 题目描述:给定两个整数数组 inorder 和 postorder ,其中 inorder 是二叉树的中序遍历, postorder 是同一棵树的后序遍历,请你构造并返回这颗 二叉树 。示例 1: ``` 输入:inorder = [9,3,15,20,7], postorder

read more

109. 有序链表转换二叉搜索树

## 题目描述:给定一个单链表的头节点 head ,其中的元素 按升序排序 ,将其转换为高度平衡的二叉搜索树。本题中,一个高度平衡二叉树是指一个二叉树每个节点 的左右两个子树的高度差不超过 1。示例 1: ``` 输入: head = [-10,-3,0,5,9] 输出: [0,-3,9

read more

114. 二叉树展开为链表

## 题目描述:给你二叉树的根结点 root ,请你将它展开为一个单链表:- 展开后的单链表应该同样使用 TreeNode ,其中 right 子指针指向链表中下一个结点,而左子指针始终为 null 。 - 展开后的单链表应该与二叉树 先序遍历 顺序相同。示例 1: ``` 输入:root

read more