36. 有效的数独(Valid Sudoku)题解(C语言)

题目描述

判断一个 9×9 的数独是否有效。只需要根据以下规则验证已经填入的数字是否有效即可:

  1. 数字 1-9 在每一行只能出现一次;

  2. 数字 1-9 在每一列只能出现一次;

  3. 数字 1-9 在每一个以粗实线分隔的 3×3 宫内只能出现一次。

注意:

  • 空白格用 '.' 表示;

  • 一个有效的数独不一定是可解的,只需要验证已经填入的数字是否有效即可。

示例 1:

复制代码
输入:
board = 
[["5","3",".",".","7",".",".",".","."]
,["6",".",".","1","9","5",".",".","."]
,[".","9","8",".",".",".",".","6","."]
,["8",".",".",".","6",".",".",".","3"]
,["4",".",".","8",".","3",".",".","1"]
,["7",".",".",".","2",".",".",".","6"]
,[".","6",".",".",".",".","2","8","."]
,[".",".",".","4","1","9",".",".","5"]
,[".",".",".",".","8",".",".","7","9"]]

输出:true

示例 2:

复制代码
输入:
board = 
[["8","3",".",".","7",".",".",".","."]
,["6",".",".","1","9","5",".",".","."]
,[".","9","8",".",".",".",".","6","."]
,["8",".",".",".","6",".",".",".","3"]
,["4",".",".","8",".","3",".",".","1"]
,["7",".",".",".","2",".",".",".","6"]
,[".","6",".",".",".",".","2","8","."]
,[".",".",".","4","1","9",".",".","5"]
,[".",".",".",".","8",".",".","7","9"]]

输出:false
解释:左上角 3×3 宫格内有两个 '8',因此无效。

解题思路

本题的核心是 检查三种约束条件

  1. 每行数字不能重复;

  2. 每列数字不能重复;

  3. 每个 3×3 宫格内数字不能重复。

方法

  • 使用三个二维数组来记录数字是否出现过:

    • row[9][9]:行出现记录

    • col[9][9]:列出现记录

    • box[9][9]:3×3 宫格出现记录

  • 遍历整个棋盘:

    1. 如果当前格是 '.',跳过;

    2. 否则将字符转为数字索引 num = board[i][j] - '1'

    3. 计算当前格所属的宫格索引 boxIndex = (i / 3) * 3 + j / 3

    4. 如果 row[i][num]col[j][num]box[boxIndex][num] 已经标记过,则返回 false

    5. 否则将其标记为已出现。

  • 遍历完成后仍未发现冲突,则数独有效,返回 true


C语言实现

复制代码
#include <stdbool.h>

bool isValidSudoku(char** board, int boardSize, int* boardColSize) {
    int row[9][9] = {0};
    int col[9][9] = {0};
    int box[9][9] = {0};

    for(int i = 0; i < 9; i++){
        for(int j = 0; j < 9; j++){
            if(board[i][j] == '.') continue;

            int num = board[i][j] - '1';       // 数字转换为索引 0~8
            int boxIndex = (i / 3) * 3 + j / 3; // 计算 3x3 宫格索引

            if(row[i][num] || col[j][num] || box[boxIndex][num])
                return false;

            row[i][num] = 1;
            col[j][num] = 1;
            box[boxIndex][num] = 1;
        }
    }

    return true;
}

算法分析

  • 时间复杂度:O(9×9) = O(1),固定大小棋盘,遍历每个格子一次;

  • 空间复杂度:O(9×9) = O(1),使用固定大小的三个二维数组存储状态。


总结

本题属于 经典的状态记录题,核心技巧是:

  1. 使用辅助数组记录行、列、宫格的状态;

  2. 宫格索引计算公式:boxIndex = (i/3)*3 + j/3

  3. 遇到 '.' 跳过即可。

面试时,如果能讲出这一思路,并且写出简洁的 C 语言实现,基本可以轻松通过。


相关推荐
小肝一下2 小时前
每日两道力扣,day5
数据结构·c++·算法·leetcode·职场和发展·hot100
jiang_changsheng2 小时前
亚马逊的2026年最新算法变革自然流量分发机制“文本匹配”到“多模态意图理解”的范式革命
大数据·算法·推荐算法
OOJO7 小时前
c++---list介绍
c语言·开发语言·数据结构·c++·算法·list
别或许8 小时前
1、高数----函数极限与连续(知识总结)
算法
派大星~课堂8 小时前
【力扣-142. 环形链表2 ✨】Python笔记
python·leetcode·链表
田梓燊8 小时前
code 560
数据结构·算法·哈希算法
笨笨饿8 小时前
29_Z变换在工程中的实际意义
c语言·开发语言·人工智能·单片机·mcu·算法·机器人
kobesdu9 小时前
综合强度信息的激光雷达去拖尾算法解析和源码实现
算法·机器人·ros·slam·激光雷达
艾为电子9 小时前
【技术帖】让接口不再短命:艾为 C-Shielding™ Type-C智能水汽防护技术解析
c语言·开发语言
weixin_413063219 小时前
记录 MeshFlow-Online-Video-Stabilization 在线稳像
算法·meshflow·实时防抖