开启辅助访问 切换到宽版

精易论坛

 找回密码
 注册

QQ登录

只需一步,快速开始

用微信号发送消息登录论坛

新人指南 邀请好友注册 - 我关注人的新帖 教你赚取精币 - 每日签到


求职/招聘- 论坛接单- 开发者大厅

论坛版规 总版规 - 建议/投诉 - 应聘版主 - 精华帖总集 积分说明 - 禁言标准 - 有奖举报

查看: 2434|回复: 21
收起左侧

[易源码分享] 求解数独 改自超级用户 侵删

[复制链接]

结帖率:100% (47/47)
发表于 2023-4-9 00:22:10 | 显示全部楼层 |阅读模式   江西省南昌市
分享源码
界面截图: -
是否带模块: -
备注说明: -
求解使用状态压缩+回溯。
状态压缩
  • 使用 bitset<9> 来压缩存储每一行、每一列、每一个 3x3 宫格中 1-9 是否出现
  • 这样每一个格子就可以计算出所有不能填的数字,然后得到所有能填的数字 getPossibleStatus()
  • 填入数字和回溯时,只需要更新存储信息
  • 每个格子在使用时,会根据存储信息重新计算能填的数字


回溯
  • 每次都使用 getNext() 选择能填的数字最少的格子开始填,这样填错的概率最小,回溯次数也会变少
  • 使用 fillNum() 在填入和回溯时负责更新存储信息
  • 一旦全部填写成功,一路返回 true ,结束递归



1fb1c64cfddb5c66b61bd769224724a05027172d6486feb19b3a16d9473372ee-图片.png

[C++] 纯文本查看 复制代码
class Solution {

private:
    bitset<9> getPossibleStatus(int x, int y)
    {
        return ~(rows[x] | cols[y] | cells[x / 3][y / 3]);
    }

    vector<int> getNext(vector<vector<char>>& board)
    {
        vector<int> ret;
        int minCnt = 10;
        for (int i = 0; i < board.size(); i++)
        {
            for (int j = 0; j < board.size(); j++)
            {
                if (board[j] != '.') continue;
                auto cur = getPossibleStatus(i, j);
                if (cur.count() >= minCnt) continue;
                ret = { i, j };
                minCnt = cur.count();
            }
        }
        return ret;
    }

    void fillNum(int x, int y, int n, bool fillFlag)
    {
        rows[x][n] = (fillFlag) ? 1 : 0;
        cols[y][n] = (fillFlag) ? 1 : 0;
        cells[x / 3][y / 3][n] = (fillFlag) ? 1 : 0;
    }

    bool dfs(vector<vector<char>>& board, int cnt)
    {
        if (cnt == 0) return true;

        auto next = getNext(board);
        auto bits = getPossibleStatus(next[0], next[1]);
        for (int n = 0; n < bits.size(); n++)
        {
            if (!bits.test(n)) continue;
            fillNum(next[0], next[1], n, true);
            board[next[0]][next[1]] = n + '1';
            if (dfs(board, cnt - 1)) return true;
            board[next[0]][next[1]] = '.';
            fillNum(next[0], next[1], n, false);
        }
        return false;
    }

    vector<bitset<9>> rows;
    vector<bitset<9>> cols;
    vector<vector<bitset<9>>> cells;

public:

    void solve(vector<vector<char>>& board)
    {
        rows = vector<bitset<9>>(9, bitset<9>());
        cols = vector<bitset<9>>(9, bitset<9>());
        cells = vector<vector<bitset<9>>>(3, vector<bitset<9>>(3, bitset<9>()));

        int cnt = 0;
        for (int i = 0; i < board.size(); i++)
        {
            for (int j = 0; j < board.size(); j++)
            {
                cnt += (board[j] == '.');
                if (board[j] == '.') continue;
                int n = board[j] - '1';
                rows |= (1 << n);
                cols[j] |= (1 << n);
                cells[i / 3][j / 3] |= (1 << n);
            }
        }
        dfs(board, cnt);
    }
};




QQ截图20230409001602.png

QQ截图20230409001614.png

秒出结果

数独.zip (228.31 KB, 下载次数: 55)

评分

参与人数 3好评 +3 精币 +4 收起 理由
易语言资源网 + 1 + 3 开源精神必须支持~
光影魔术 + 1 + 1 支持开源~!感谢分享
睿思Online + 1 YYDS~!

查看全部评分


本帖被以下淘专辑推荐:

结帖率:0% (0/5)

签到天数: 1 天

发表于 2024-4-16 23:35:32 | 显示全部楼层   贵州省黔东南苗族侗族自治州
DLL 易语言源码有吗
回复 支持 反对

使用道具 举报

结帖率:0% (0/5)

签到天数: 1 天

发表于 2024-4-16 23:28:25 | 显示全部楼层   贵州省黔东南苗族侗族自治州
不能解其他答案?
回复 支持 反对

使用道具 举报

结帖率:100% (7/7)

签到天数: 15 天

发表于 2024-1-29 09:46:37 | 显示全部楼层   广东省佛山市
支持开源~!感谢分享
回复 支持 反对

使用道具 举报

发表于 2023-4-23 12:25:06 | 显示全部楼层   四川省成都市

好,太好了
回复 支持 反对

使用道具 举报

结帖率:0% (0/1)

签到天数: 11 天

发表于 2023-4-18 19:55:44 | 显示全部楼层   广东省深圳市
牛,
回复 支持 反对

使用道具 举报

发表于 2023-4-13 21:53:48 | 显示全部楼层   广东省佛山市
求解数独 改自超级用户
回复 支持 反对

使用道具 举报

结帖率:100% (3/3)
发表于 2023-4-13 12:02:59 | 显示全部楼层   福建省泉州市
已经顶贴,感谢您对论坛的支持!
回复 支持 反对

使用道具 举报

结帖率:50% (1/2)

签到天数: 1 天

发表于 2023-4-13 10:09:13 | 显示全部楼层   浙江省温州市
好,太好了
回复 支持 反对

使用道具 举报

结帖率:92% (213/232)

签到天数: 19 天

发表于 2023-4-11 11:12:28 | 显示全部楼层   黑龙江省伊春市
支持一下,学习看看~~~
回复 支持 反对

使用道具 举报

结帖率:98% (107/109)

签到天数: 22 天

发表于 2023-4-10 21:40:27 | 显示全部楼层   山东省青岛市
谢谢分享学习一下
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则 致发广告者

发布主题 收藏帖子 返回列表

sitemap| 易语言源码| 易语言教程| 易语言论坛| 易语言模块| 手机版| 广告投放| 精易论坛
拒绝任何人以任何形式在本论坛发表与中华人民共和国法律相抵触的言论,本站内容均为会员发表,并不代表精易立场!
论坛帖子内容仅用于技术交流学习和研究的目的,严禁用于非法目的,否则造成一切后果自负!如帖子内容侵害到你的权益,请联系我们!
防范网络诈骗,远离网络犯罪 违法和不良信息举报电话0663-3422125,QQ: 793400750,邮箱:wp@125.la
网站简介:精易论坛成立于2009年,是一个程序设计学习交流技术论坛,隶属于揭阳市揭东区精易科技有限公司所有。
Powered by Discuz! X3.4 揭阳市揭东区精易科技有限公司 ( 粤ICP备12094385号-1) 粤公网安备 44522102000125 增值电信业务经营许可证 粤B2-20192173

快速回复 返回顶部 返回列表