LeetCode 2033. 获取单值网格的最小操作数(数学,中位数贪心)

题意:
给你一个大小为 m x n 的二维整数网格 grid 和一个整数 x 。
每一次操作,你可以对 grid 中的任一元素 加 x 或 减 x 。

单值网格 是全部元素都相等的网格。

返回使网格化为单值网格所需的 最小 操作数。如果不能,返回 -1 。

数据范围:
m == grid.length
n == grid[i].length
1 <= m, n <= 1e5
1 <= m * n <= 1e5
1 <= x, grid[i][j] <= 1e4
解法:
有解的条件:
任意两个数的差值是x的倍数.
先特判一下.

-----

最后的数字取中位数时最优.

假设中位数为mid,<mid的数有q个,>mid的数有e个,
那么mid每左移x,答案变化-q+e,
mid每右移x,答案变化q-e.

可以尽量使q=e才能让答案最小,
因此mid取中位数.
code:
func minOperations(grid [][]int, x int) int {
    a:=[]int{}
    for i:=0;i<len(grid);i++{
        for j:=0;j<len(grid[0]);j++{
            a=append(a,grid[i][j])
        }
    }
    //差值不是x的倍数则无解
    n:=len(a)
    sort.Ints(a)
    for i:=1;i<n;i++{
        if (a[i]-a[0])%x!=0{
            return -1
        }
    }
    //
    mid:=a[n/2]
    ans:=0
    abs:=func(i int)int{
        if i<0{
            return -i
        }
        return i
    }
    for i:=0;i<n;i++{
        ans+=abs(a[i]-mid)/x
    }
    return ans
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值