leetcode 36 有效的数独 哈希表 unordered_set unordersd_map 保存状态 leetcode 37 解数独
leetcode 36
感觉就是遍历。
保存好状态,就是各行各列还有各分区divide的情况
用数组做。
空间小时间大

class Solution {
public:
bool isValidSudoku(vector<vector<char>>& board) {
int row[9][9]={0},col[9][9]={0},div[9][9]={0};
int temp=0,dnum;
for(int i=0;i<9;i++){
for(int j=0;j<9;j++){
if(board[i][j]!='.'){
temp=board[i][j]-'0'-1;
row[i][temp]++;
col[j][temp]++;
dnum=(i/3)*3+j/3;
div[dnum][temp]++;
if(row[i][temp]==2||col[j][temp]==2||div[dnum][temp]==2)
return false;
}
}
}
return true;
}
};
然后
学到了用unordered_set unordersd_map加速。
空间大时间小

class Solution {
public:
bool isValidSudoku(vector<vector<char>>& board) {
vector<unordered_set<int>> row(9), col(9), block(9);
for(int i = 0; i < 9; ++ i){
for(int j = 0; j < 9; ++ j){
int bindex = (i / 3)* 3 + j / 3;
char cur = board[i][j];
if(cur == '.') continue;
if(row[i].count(cur) || col[j].count(cur) || block[bindex].count(cur)) return false;
row[i].insert(cur);
col[j].insert(cur);
block[bindex].insert(cur);
}
}
return true;
}
};
leetcode 37
做之前还百度了一下有没有解数独的技巧。。。 OxO
上一道题保存了行列块的状态,这里也可以保留,保留的不再是出现与否而是可用数字
回溯就完事了
class Solution {
public:
vector<unordered_set<int>> row, col, block;
void solveSudoku(vector<vector<char>>& board) {
// 预处理,初始状态
int t=0;
unordered_set<int> tp={1,2,3,4,5,6,7,8,9};
for(int i=0;i<9;i++)
{row.push_back(tp);col.push_back(tp);block.push_back(tp);}
for( int i = 0; i < 9; i++)
for( int j = 0; j < 9; j++)
if( board[i][j] != '.'){
t = board[i][j] - '0';
row[i].erase(t), col[j].erase(t), block[(i/3)*3 + j/3].erase(t);
}
int flag=0;
dfs(board,0,flag);
if(flag==0) //题目说明有唯一解,不会出现
cout<<"wrong"<<endl;
}
bool check(vector<vector<char>>& board,int i,int j,int num){
if(row[i].find(num)!= row[i].end()&& col[j].find(num) != col[j].end()
&& block[(i/3)*3 + j/3].find(num) != block[(i/3)*3 + j/3].end())
return true;
return false;
}
void dfs(vector<vector<char>>& board,int cnt,int &flag){
if( cnt == 81){
flag = 1;
return ;
}
int i = cnt / 9, j = cnt % 9;
if( board[i][j] == '.'){
for( int k = 1; k < 10; k++)
if(check(board, i, j, k)){
row[i].erase(k), col[j].erase(k), block[(i/3)*3 + j/3].erase(k);
board[i][j] = '0'+k;
dfs(board, cnt+1,flag);
if(flag)
return;
else{
row[i].insert(k), col[j].insert( k), block[(i/3)*3 + j/3].insert(k);
board[i][j] = '.';
}
}
else
continue;
}
else
dfs(board, cnt+1,flag);
return ;
}
};
然后效率很差。。时间和空间上

改成数组
class Solution {
public:
int row[10][10], col[10][10], block[10][10];
void solveSudoku(vector<vector<char>>& board) {
// 预处理,初始状态
int t=0;
for( int i = 0; i < 9; i++)
for( int j = 1; j < 10; j++)
{
row[i][j]=0, col[i][j]=0, block[i][j]=0;
}
for( int i = 0; i < 9; i++)
for( int j = 0; j < 9; j++)
if( board[i][j] != '.'){
t = board[i][j] - '0';
row[i][t]=1, col[j][t]=1, block[(i/3)*3 + j/3][t]=1;
}
int flag=0;
dfs(board,0,flag);
if(flag==0) //题目说明有唯一解,不会出现
cout<<"wrong"<<endl;
}
void dfs(vector<vector<char>>& board,int cnt,int &flag){
if( cnt == 81){
flag = 1;
return ;
}
int i = cnt / 9, j = cnt % 9;
if( board[i][j] == '.'){
for( int k = 1; k < 10; k++)
if(row[i][k]==0 && col[j][k]==0 && block[(i/3)*3 + j/3][k]==0){
row[i][k]=1, col[j][k]=1, block[(i/3)*3 + j/3][k]=1;
board[i][j] = '0'+k;
dfs(board, cnt+1,flag);
if(flag)
return;
else{
row[i][k]=0, col[j][k]=0, block[(i/3)*3 + j/3][k]=0;
board[i][j] = '.';
}
}
else
continue;
}
else
dfs(board, cnt+1,flag);
return ;
}
};
我好了

然后,不传引用而是copy可以再节省一点空间
然后剪枝,没考虑 OxO
leetcode 36 有效的数独 哈希表 unordered_set unordersd_map 保存状态 leetcode 37 解数独的更多相关文章
- Leetcode之回溯法专题-37. 解数独(Sudoku Solver)
Leetcode之回溯法专题-37. 解数独(Sudoku Solver) 编写一个程序,通过已填充的空格来解决数独问题. 一个数独的解法需遵循如下规则: 数字 1-9 在每一行只能出现一次.数字 1 ...
- Java实现 LeetCode 37 解数独
37. 解数独 编写一个程序,通过已填充的空格来解决数独问题. 一个数独的解法需遵循如下规则: 数字 1-9 在每一行只能出现一次. 数字 1-9 在每一列只能出现一次. 数字 1-9 在每一个以粗实 ...
- [leetcode] 37. 解数独(Java)(dfs,递归,回溯)
37. 解数独 1A 这个题其实15分钟左右就敲出来并且对了...但是由于我输错了一个数..导致我白白debug一个多小时.. 没啥难度,练递归-dfs的好题 class Solution { pri ...
- [LeetCode] #1# Two Sum : 数组/哈希表/二分查找/双指针
一. 题目 1. Two SumTotal Accepted: 241484 Total Submissions: 1005339 Difficulty: Easy Given an array of ...
- [LeetCode] Fraction to Recurring Decimal 哈希表
Given two integers representing the numerator and denominator of a fraction, return the fraction in ...
- leetcode刷题-37解数独
题目 编写一个程序,通过已填充的空格来解决数独问题. 一个数独的解法需遵循如下规则: 数字 1-9 在每一行只能出现一次.数字 1-9 在每一列只能出现一次.数字 1-9 在每一个以粗实线分隔的 3x ...
- C++ 哈希表 (hashtable) 用于保存简单的数据,及数据查找,数据删除
/*hashtable.h*/ #include<iostream> #include <string> #include<vector> using namesp ...
- Leetcode——37.解数独 [##]
@author: ZZQ @software: PyCharm @file: leetcode37_solveSudoku.py @time: 2018/11/20 16:41 思路:递归回溯 首先, ...
- C#LeetCode刷题-哈希表
哈希表篇 # 题名 刷题 通过率 难度 1 两数之和 C#LeetCode刷题之#1-两数之和(Two Sum) 42.8% 简单 3 无重复字符的最长子串 24.2% 中等 18 四数之和 ...
随机推荐
- Java自学笔记之学生管理系统
实现:学生管理系统,实现学生信息的添加.修改.查询和删除功能 涉及:集合的基础知识(集合遍历,值得获取与替换,set/get方法) 代码如下: Student文件 1 package Demo_120 ...
- 三十三:WEB漏洞-逻辑越权之水平垂直越权
水平和垂直越权 水平越权:可以获得同级别用户权限 垂直权限:享受高几个层次的用户权限 解释,原理,检测,利用,防御 通过更换的某个ID之类的身份标识,从而使得A账号获取(修改,删除)B账号的数据,通过 ...
- Golang拼接字符串的5种方法及其效率_Chrispink-CSDN博客_golang 字符串拼接效率 https://blog.csdn.net/m0_37422289/article/details/103362740
Different ways to concatenate two strings in Golang - GeeksforGeeks https://www.geeksforgeeks.org/di ...
- 公共错误码 - 支付宝开放平台 https://opendocs.alipay.com/open/common/105806
公共错误码 - 支付宝开放平台 https://opendocs.alipay.com/open/common/105806
- int ping = 11; 限流 客户端验证与服务端是连接的
int ping = 11; ZooKeeper Programmer's Guide https://zookeeper.apache.org/doc/r3.1.2/zookeeperProgram ...
- Tensorflow-各种优化器总结与比较
优化器总结 机器学习中,有很多优化方法来试图寻找模型的最优解.比如神经网络中可以采取最基本的梯度下降法. 梯度下降法(Gradient Descent) 梯度下降法是最基本的一类优化器,目前主要分为三 ...
- Pod和容器的LimitRange原理和实践总结
一.背景介绍 通常情况下,Pod中的容器可以无限制的使用节点上的CPU和内存资源,在共享资源和资源有限的情况下,若不加以限制,某个集群或命名空间的资源可能会消耗殆尽,导致其他节点上优先级低的Pod发生 ...
- Axure RP 9版本最新版授权码和密钥 亲测可用
分享Axure RP 9版本最新版授权码和密钥 亲测可用 声明:以下资源的获取来源为网络收集,仅供学习参考,不作商业用途,如有侵权请联系博主删除,谢谢! 自新的Axure RP 9.0 Beta版发布 ...
- LOJ10097和平委员会
POI 2001 根据宪法,Byteland民主共和国的公众和平委员会应该在国会中通过立法程序来创立. 不幸的是,由于某些党派代表之间的不和睦而使得这件事存在障碍. 此委员会必须满足下列条件: 每个党 ...
- js异步、事件循环(EventLoop)小结
单线程 众所周知,JS是单线程的语言,之所以是单线程,用一句烂大街的话就是,如果两个线程同时操作一个DOM节点,那么该以哪个为准呢,虽然多线程也有办法解决,但是js毕竟是浏览器脚本语言,不需要那么复杂 ...