leetcode 1895. 最大的幻方 中等

一个 k x k幻方 指的是一个 k x k 填满整数的方格阵,且每一行、每一列以及两条对角线的和 全部 相等 。幻方中的整数 不需要互不相同 。显然,每个 1 x 1 的方格都是一个幻方。

给你一个 m x n 的整数矩阵 grid ,请你返回矩阵中 最大幻方尺寸 (即边长 k)。

示例 1:

复制代码
输入:grid = [[7,1,4,5,6],[2,5,1,6,4],[1,5,4,3,2],[1,2,7,3,4]]
输出:3
解释:最大幻方尺寸为 3 。
每一行,每一列以及两条对角线的和都等于 12 。
- 每一行的和:5+1+6 = 5+4+3 = 2+7+3 = 12
- 每一列的和:5+5+2 = 1+4+7 = 6+3+3 = 12
- 对角线的和:5+4+3 = 6+4+2 = 12

示例 2:

复制代码
输入:grid = [[5,1,3,1],[9,3,3,1],[1,3,3,8]]
输出:2

提示:

  • m == grid.length
  • n == grid[i].length
  • 1 <= m, n <= 50
  • 1 <= grid[i][j] <= 10^6

分析:先求出水平、垂直、主对角线、反对角线的前缀和,再枚举幻方边长,遍历矩阵的每个点作为左上角点时幻方是否满足条件。

cpp 复制代码
int largestMagicSquare(int** grid, int gridSize, int* gridColSize) {
    int n=gridSize,m=gridColSize[0],ans=1;
    int sum_hor[n+5][m+5],sum_ver[n+5][m+5],sum_lr[n+5][m+5],sum_rl[n+5][m+5];
    for(int i=0;i<=n;++i)
        for(int j=0;j<=m;++j)
            sum_hor[i][j]=sum_ver[i][j]=sum_lr[i][j]=sum_rl[i][j]=0;

    for(int i=0;i<n;++i)
        for(int j=m-1;j>=0;--j)
            sum_hor[i][j]=grid[i][j]+sum_hor[i][j+1];
    for(int j=0;j<m;++j)
        for(int i=n-1;i>=0;--i)
            sum_ver[i][j]=grid[i][j]+sum_ver[i+1][j];
    for(int i=n-1;i>=0;--i)
        for(int j=m-1;j>=0;--j)
            sum_lr[i][j]=grid[i][j]+sum_lr[i+1][j+1];
    for(int i=n-1;i>=0;--i)
        for(int j=0;j<m;++j)
        {
            sum_rl[i][j]=grid[i][j];
            if(j)sum_rl[i][j]+=sum_rl[i+1][j-1];
        }
    
    int maxn=fmin(n,m);
    for(int l=1;l<maxn;++l)
    {
        for(int i=0;i<n-l;++i)
        {
            for(int j=0;j<m-l;++j)
            {
                int temp=sum_lr[i][j]-sum_lr[i+l+1][j+l+1],f1=0;
                if(j-1>=0)
                {
                    if(sum_rl[i][j+l]-sum_rl[i+l+1][j-1]==temp)f1=1;
                }
                else if(sum_rl[i][j+l]==temp)f1=1;

                for(int k=0;k<=l&&f1;++k)
                {
                    if(sum_hor[i+k][j]-sum_hor[i+k][j+l+1]!=temp)f1=0;
                    if(sum_ver[i][j+k]-sum_ver[i+l+1][j+k]!=temp)f1=0;
                }
                if(f1)ans=fmax(ans,l+1);
            }
        }
    }
    
    return ans;
}
相关推荐
摸个小yu14 小时前
【力扣LeetCode热题h100】链表、二叉树
算法·leetcode·链表
skywalker_1116 小时前
力扣hot100-5(盛最多水的容器),6(三数之和)
算法·leetcode·职场和发展
生信研究猿16 小时前
leetcode 226.翻转二叉树
算法·leetcode·职场和发展
XWalnut16 小时前
LeetCode刷题 day9
java·算法·leetcode
6Hzlia17 小时前
【Hot 100 刷题计划】 LeetCode 39. 组合总和 | C++ 回溯算法与 startIndex 剪枝
c++·算法·leetcode
宵时待雨17 小时前
优选算法专题1:双指针
数据结构·c++·笔记·算法·leetcode
We་ct17 小时前
LeetCode 172. 阶乘后的零:从暴力到最优,拆解解题核心
开发语言·前端·javascript·算法·leetcode·typescript
老虎062718 小时前
LeetCode热题100 刷题笔记(第五天)双指针法 「 三数之和 」
笔记·算法·leetcode
美式请加冰18 小时前
简单多状态问题
数据结构·算法·leetcode
佑白雪乐18 小时前
<LeetCode>二叉树前/中/后/层遍历**递归&&非递归**
算法·leetcode·深度优先