我是靠谱客的博主 怕黑麦片,最近开发中收集的这篇文章主要介绍python123九宫格输入法_手机屏幕解锁模式,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

36

可以通过DFS的方式先计算出刚好按下n个键时有多少种组合,然后求出S[n]至S[M]的和。

DFS的主要难度在于,下一步可以与当前的位置不直接相连。这时分两种情况:

1. 普通的八个方向 (上下左右以及其45度夹角方向):

若这八个方向都已经越界或走过了,则这时无路可走。若是普通的DFS则返回,但是九宫格解锁可以跳过相邻的一格。注意只能在这八个方向跳多一步,相当于踩着已经被按下的位置再沿着相同方向走一步。

2.其余的八个方向

其余的八个方向虽然不直接与当前位置直接相连,但是它与当前位置的连线不会触碰到其他位置,因此也可以直接到达。

以下为DFS代码

int dir[16][2] = {//16个方向

{ -1, 0 }, { -1, 1 }, { 0, 1 }, { 1, 1 },

{ 1, 0 }, { 1, -1 }, { 0, -1 }, { -1, -1 },

{-2, 1 }, { -1, 2 }, { 1, 2 }, { 2, 1 },

{ 2, -1 }, { 1, -2 }, { -1, -2 }, { -2, -1 }

};

int isVisit[5][5];//是否已按下

bool canVisit(int i, int j){//判断能否按下

if (i 3 || j 3 || isVisit[i][j]) return false;

return true;

}

int times[10];

//d:已经被选中的键的个数(深度)

void DFS(int i, int j, int d){

if (d == 9){

return;

}

isVisit[i][j] = true;

times[d++] ++;

//选择下一个键

for (int y = 0; y 

int ni = i + dir[y][0], nj = j + dir[y][1];

if (canVisit(ni, nj)){//该点未被选择

DFS(ni, nj, d);

}

else if (y 

ni += dir[y][0];

nj += dir[y][1];

if (canVisit(ni, nj) ){//该点未被选择

DFS(ni, nj, d);

}

}

}

isVisit[i][j] = false;

return;

} solution:

class Solution {

public:

int solution(int m, int n) {

if(m > n){    //易被忽略

return 0;

}

m = (m<0? 0: m);    //参数检查必须有

n = (n>9? 9:n);

int tmp[] = {0, 9, 56, 320, 1624, 7152, 26016, 72912, 140704, 140704 };

int ans = 0;

for(int i=m; i<=n; i++){

ans += tmp[i];

}

return ans;

}

};

发表于 2020-04-01 21:43:16

回复(5)

12

# Python3 dfs

# 所有方向

di = [(1,0), (-1,0), (0,1), (0,-1), (1,1), (-1,-1), (1,-1), (-1,1), (1,2), (1,-2), (-1,2), (-1,-2), (2,1), (2,-1),(-2,1),(-2,-1)]

# 可跨一个点的方向

ds = [(1,0), (-1,0), (0,1), (0,-1), (1,1), (-1,-1), (1,-1), (-1,1)]

# 9个点

nodes = {(0,0), (0,1), (0,2), (1,0), (1,1), (1,2), (2,0), (2,1), (2,2)}

def dfs(x, y, visited, count):

visited.add((x, y))

count -= 1

ans = 0

if count == 0:

ans += 1

else:

for d in di:

if (x+d[0], y+d[1]) in visited or (x+d[0], y+d[1]) not in nodes:

if d not in ds:

continue

else:

dx = d[0] * 2

dy = d[1] * 2

if (x+dx, y+dy) in nodes and (x+dx, y+dy) not in visited:

ans += dfs(x+dx, y+dy, visited, count)

else:

ans += dfs(x+d[0], y+d[1], visited, count)

visited.remove((x, y))

return ans

ans = [0] * 10

for count in range(1, 10):

for i in range(3):

for j in range(3):

visited = set()

ans[count] += dfs(i, j, visited, count)

# ans[i]即为i个键的结果数

# ans = [0, 9, 56, 320, 1624, 7152, 26016, 72912, 140704, 140704]

print(ans)

编辑于 2020-04-02 18:52:08

回复(2)

7

class Solution {

public:

void move(vector >& board, int i, int j, int k, int m, int n, int& ans){

// 如果已经走过的点数大于等于m,则是有效路径,ans++

if(k >= m) ans ++;

// 如果已经走过的点数等于n,则不需要继续探索,故返回

if(k == n) return;

// 如果已经走过的点数小于n,则还可以继续探索

for(int dx=-2; dx<=2; dx++){

for(int dy=-2; dy<=2; dy++){

if(i+dx>=0 && i+dx<=2 && j+dy>=0 && j+dy<=2 && board[i+dx][j+dy]==0){

// 如果两点之间没有第三个点(条件:dx%2 || dy%2),则无需判断是否经过“已经过”的点

// 如果两点之间有第三个点,则需要判断这个点是否是已经走过的点

if(dx%2 || dy%2 || (!(dx%2) && !(dy%2) && board[i+dx/2][j+dy/2]==1)){

board[i+dx][j+dy] = 1;

move(board, i+dx, j+dy, k+1, m, n, ans);

board[i+dx][j+dy] = 0;

}

}

}

}

return;

}

int solution(int m, int n) {

// write code here

vector > board(3, vector(3, 0));

int ans = 0;

// 如果n等于0,则直接返回0

if(n == 0) return ans;

// 选择棋盘上任意一点作为起点

for(int i=0; i<3; i++){

for(int j=0; j<3; j++){

board[i][j] = 1;

move(board, i, j, 1, m, n, ans);

board[i][j] = 0;

}

}

return ans;

}

};

发表于 2020-04-21 19:06:47

回复(2)

4

Java版本,比较容易想到,但是调试了很多遍。 public static int solution (int m, int n) {

// 递归实现

int count = 0;

n = n>9 ? 9 : n;

for(int i=m;i<=n;i++) {

boolean[][] flags = new boolean[3][3];

for(int j=0;j<3;j++) {

for(int t=0;t<3;t++) {

count += findNext(flags, j, t, 0, i);

}

}

}

return count;

}

public static int findNext(boolean[][] flags,int curRow, int curCol, int steps, int n) {

int count = 0;

steps ++;

flags[curRow][curCol] = true;

// 步数走完返回

if(steps == n)

return 1;

// 如果可以走,那么返回 1

for(int i=0;i<3;i++) {

for(int j=0;j<3;j++) {

if(flags[i][j] == false && canStep(flags, curRow, curCol, i, j)) {

count += findNext(flags, i, j, steps, n);

// 恢复状态

flags[i][j] = false;

}

}

}

flags[curRow][curCol] = false;

return count;

}

public static boolean canStep(boolean[][] flags, int curRow, int curCol, int targetRow, int targetCol) {

// 在同一行

if(curRow == targetRow) {

int low = curCol < targetCol?curCol:targetCol;

if(Math.abs(curCol - targetCol) >1 && flags[curRow][low+1] == false)

return false;

}

// 在同一列

if(curCol == targetCol) {

int low = curRow < targetRow?curRow:targetRow;

if(Math.abs(curRow - targetRow) >1 && flags[low+1][curCol] == false)

return false;

}

// 斜对角

if(Math.abs(curRow-targetRow)==2 && Math.abs(curCol-targetCol)==2 && flags[1][1] == false)

return false;

return true;

}

发表于 2020-04-24 12:58:52

回复(5)

2

import java.util.*;

/*

图形密码即使图案相同,但是顺序不同的话也算作一种

dfs,以每个位置作为起点,然后走 [n, m] 个点

当走到无路可走的时候,如果走的点小于 m,那么该路径不满足要求

当已经走的点 > n 的时候,那么表示接下来的路径也不满足要求,直接返回

当我们发现周围 几个点都已经走过了,如果是普通的 dfs,这时候已经返回了

但是,我们可以越过已经访问过的点,继续往下一个点走,因此我们需要判断

8 个方向第 2 个点是否已经访问过了,如果没有,那么可以继续访问

注意:只有我们发现某个方向上的点访问过了才可以越过该点

如果该方向上的第一个点还没有访问,那么我们不能直接越过

*/

public class Solution {

/**

* 实现方案

* @param m int整型 最少m个键

* @param n int整型 最多n个键

* @return int整型

*/

public int solution (int m, int n) {

//范围处理

if(m > n){

return 0;

}

m = Math.max(0, m);

n = Math.min(9, n);

if(n == 0){

return 0;

}

// write code here

res = 0;

visited = new boolean[3][3];

for(int i = 0; i 

for(int j = 0; j 

dfs(i, j, m, n, 1);

}

}

return res;

}

static int res;

static boolean[][] visited;

int[][] pos = {

{1, 0},{1, 1},{1, -1}, {-1, 0}, {-1, -1}, {-1, 1}, {0, 1}, {0, -1},

{1, 2}, {2, 1}, {-1, 2}, {-1, -2}, {-2, -1}, {-2, 1}, {1, -2}, {2, -1}

};

private void dfs(int i, int j, int m, int n, int p){

if(p >= m){

res++;

}

//当访问个数大于等于 n 个,那么已经足够了,无需继续访问

if(p >= n){

return;

}

visited[i][j] = true;

//8 个方向走一遍

for(int[] po : pos){

int x = i + po[0];

int y = j + po[1];

if(!isEvil(x, y)){

if(!visited[x][y]){

dfs(x, y, m, n, p + 1);

}else{

//越过同方向上的点

int xx = x + po[0];

int yy = y + po[1];

if(!isEvil(xx, yy) && !visited[xx][yy]){

//越过 (x, y) 点,访问下一个点

dfs(xx, yy, m, n, p + 1);

}

}

}

}

visited[i][j] = false;

}

private boolean isEvil(int i, int j){

return i = 3 || j = 3;

}

}

发表于 2020-08-12 14:07:07

回复(0)

2

1、Python3技巧:使用itertools.permutations生成所有可能路径,逐个检查是否可行

2、路径是否可行,取决于相邻两个点的连线之间是否出现还没有经过的点,例如,如果没有先经过点2,那么不能出现连线(1, 3)

3、路径存在镜像对称性,可以节省大量的搜索过程,比如以3、7、9开头的路径可以对称到以1开头的路径,同理以4、6、8开头的路径可以对称到以2开头的路径。 from itertools import permutations

class Solution:

def solution(self, m, n):

if n == 0: return 0

if m == 0: m = 1

e = {(1, 3), (4, 6), (7, 9),

(1, 7), (2, 8), (3, 9),

(1, 9), (3, 7)}

h, c = {3, 4, 6, 7, 8, 9}, 0

for k in range(m, n + 1):

for a in permutations(range(1, 10), k):

if a[0] in h: continue

t = set()

for i in range(len(a) - 1):

if (a[i], a[i+1]) in e or (a[i+1], a[i]) in e:

if (a[i] + a[i+1]) // 2 not in t: break

t.add(a[i])

else: c += 1 if a[0] == 5 else 4

return c

发表于 2020-06-05 20:00:17

回复(1)

2

公众号“乘风破浪Now”更多内容可查看

分析

这道题与一般的回溯的题目不太一样,这道题的难点在于下一个点可以不是与当前点相邻。这道题回溯的下探规则不能是一个点的相邻点(正上方/正下方/正左方/正右方/左上方/左下方/右上方/右下方)。因为这道题的下探点可以是矩形对角处。如何确定下探规则成为了解这道题的关键,这个下探规则需要适用于所有的点。

现在来确定下探规则:将下探规则分为16个方向前8个用于下探相邻的点,后8个用于下探矩形对角的点。若在 A 方向上下探相邻的点发现该点已经走过,则再在 A 方向上再走多一步则可以下探到与当前点不相邻的点。(A方向为前8个常规方向)

以上下探规则对所有点均适用。

static int[][] directs =

{{-1,1},{0,1},{1,1},{-1,0},{1,0},{-1,-1},{0,-1},

{1,-1},{2,-1},{1,-2},{-2,-1},{-1,-2},{1,2},{2,1},{-2,1},{-1,2}};

代码 public class Question64 {

static int[][] directs = {{-1,1},{0,1},{1,1},{-1,0},{1,0},{-1,-1},{0,-1},{1,-1},

{2,-1},{1,-2},{-2,-1},{-1,-2},{1,2},{2,1},{-2,1},{-1,2}};

static int[][] dash = { {1,2,3},

{4,5,6},

{7,8,9}};

static int[] nums = new int[10];

static public int solution(int m,int n){

m = m<=9 ? m : 9;

n = n<=9 ? n : 9;

int sum=0;

int[] nums ={0, 9, 56, 320, 1624, 7152, 26016, 72912, 140704, 140704 };

for (int i=m;i<=n;i++){

sum += nums[i];

}

return sum;

}

static public void process(boolean[] V,int count,int x,int y){

if(count==9){

nums[count]++;

return;

}

V[dash[x][y]]=true;

nums[count]++;

for(int i=0;i

int a= x+directs[i][0];

int b= y+directs[i][1];

if(canVisit(V,a,b)){

process(V,count+1,a,b);

}else if(i<8){ // 若是常规方向,则再多走一步则可以走到与当前点不相邻的点

a +=directs[i][0];

b +=directs[i][1];

if(canVisit(V,a,b)){

process(V,count+1,a,b);

}

}

}

V[dash[x][y]]=false;

}

static boolean canVisit(boolean[] V,int i,int j){

if(i<0 || i>=3 || j<0 || j>=3 || V[dash[i][j]]) return false;

return true;

}

public static void main(String[] args) {

for(int i=0;i<3;i++){

for(int j=0;j<3;j++){

process(new boolean[10],1,i,j);

}

}

for (int num : nums) {

System.out.print(num+" ");

}

}

}

发表于 2020-05-31 21:11:35

回复(0)

2

//提交时间:2020-04-29 语言:C++ 运行时间: 50 ms 占用内存:504K 状态:答案正确

class Solution {

public:

bool check(int i, int j) {

if ((i >= 0 && i <= 2) && (j >= 0 && j <= 2) && !isVisit(i, j)) {

return true;

}

return false;

}

bool isVisit(int i, int j) { return visited[i][j] == 0 ? false : true; }

int visited[5][5] = {0};

int map[16][2] = {{-1, 0}, {-1, 1},  {0, 1},   {1, 1},  {1, 0}, {1, -1},

{0, -1}, {-1, -1}, {-2, 1},  {-1, 2}, {1, 2}, {2, 1},

{2, -1}, {1, -2},  {-1, -2}, {-2, -1}};

int dfs(int i, int j, int depth) {

if (depth == 1) {

return 1;

} else if (depth == 0) {

return 0;

}

int num = 0;

visited[i][j] = 1;

for (int k = 0; k 

int pi = i + map[k][0];

int pj = j + map[k][1];

if (check(pi, pj)) {

num += dfs(pi, pj, depth - 1);

} else if (k 

pi += map[k][0];

pj += map[k][1];

if (check(pi, pj)) {

num += dfs(pi, pj, depth - 1);

}

}

}

visited[i][j] = 0;

return num;

}

/**

* 实现方案

* @param m int整型 最少m个键

* @param n int整型 最多n个键

* @return int整型

*/

int solution(int m, int n) {

int count = 0;

m = m >= 0 && m <= 9 ? m : 0;

n = n >= 0 && n <= 9 ? n : 9;

for (; m <= n; m++) {

for (int i = 0; i 

for (int j = 0; j 

count += dfs(i, j, m);

}

}

}

return count;

}

};

发表于 2020-04-29 14:46:44

回复(0)

1

class Solution {

public:

/**

* 实现方案

* @param m int整型 最少m个键

* @param n int整型 最多n个键

* @return int整型

*/

int solution(int m, int n) {

vector vis(9);

return 4 * (dfs(0, 1,vis, m, n) + dfs(1, 1,vis, m, n)) + dfs(4, 1,vis,m,n);

}

int dfs(int i, int cnt, vector vis, int m, int n) {

if (cnt > n) return 0;

int res = cnt >= m ? 1 : 0;

vis[i] = true;

int x = i / 3, y = i % 3;

for (int j = 0; j 

if (vis[j]) continue;

int xx = j / 3, yy = j % 3;

int dx = abs(x - xx), dy = abs(y - yy);

int a = min(dx, dy), b = max(dx, dy);

if (b <= 1 || a == 1 && b == 2) {

res += dfs(j, cnt + 1, vis, m, n);

}

else {

int xmid = (x + xx) / 2, ymid = (y + yy) / 2;

if (vis[xmid * 3 + ymid]) res += dfs(j, cnt + 1, vis, m, n);

}

}

return res;

}

};

发表于 2020-09-02 12:40:35

回复(0)

1

#打个表通过

class Solution:

def solution(self , m , n ):

# write code here

candi = [0,9, 56, 320, 1624, 7152, 26016, 72912, 140704, 140704]

res = 0

for i in range(m, n+1):

if i == 0: continue

if i>9: break

res += candi[i]

return res

正经答案:(简单易懂DFS,时间超了,通过80%)

class Solution:

def solution(self , m , n ):

# write code here

res = 0

for i in range(m, n+1):

if i==0:continue

res += self.compute(i)

return res

def compute(self, k):

self.cnt = 0

board = [['.']*3 for _ in range(3)]

self.x = -1

self.y = -1

def isValid(board, row, col):

if self.x == -1 and self.y == -1:

return True

if abs(row-self.x) > 1 and col == self.y:

if board[(row+self.x)//2][col] != 'Q':

return False

elif abs(col-self.y) > 1 and row == self.x:

if board[row][(col+self.y)//2] != 'Q':

return False

elif abs(col-self.y) > 1 and abs(row-self.x) > 1:

if board[(row+self.x)//2][(col+self.y)//2] != 'Q':

return False

return True

def trackback(board, k, path):

if k==0:

self.cnt += 1

self.x, self.y = path[-2]

return

for i in range(3):

for j in range(3):

if board[i][j] == 'Q': continue

if not isValid(board, i, j): continue

board[i][j] = 'Q'

path.append((i,j))

self.x, self.y = i, j

trackback(board, k-1, path)

self.x, self.y = path[-2]

path.pop()

board[i][j] = '.'

trackback(board, k, [(-1,-1)])

return self.cnt

供你们参考,和leetcode上面的N皇后思路差不多,DFS+剪枝

发表于 2020-06-07 00:22:18

回复(0)

1

# 实现方案: DFS

# 编码方案:[1 2 3

#          4 5 6

#          7 8 9]

#

class Solution:

def solution(self, m, n):

if m > n or n <= 0:

return 0

# 边界修正(还能有[3, 20]这种测试也是醉了)

if m <= 0:

m = 1

if n > 9:

n = 9

# 初始数据:保存1-9长度路径和,原始结点的set,当前所有路径

ret = [9]

oriSet = set(range(1, 10))

currList = [[1], [2], [3], [4], [5], [6], [7], [8], [9]]

for _ in range(2, n + 1):

nextList = []

for currline in currList:

for i in (oriSet - set(currline)):

if self.isValid(currline, i):

tmp = currline[:] + [i]

nextList.append(tmp)

currList = nextList

ret.append(len(currList))

return sum(ret[m-1:n])

def isValid(self, currline, i):

# 判断currline能否增加i这个结点

# 此时为空,或者中点不是整数

if currline == [] or (currline[-1] + i) % 2 == 1:

return True

# 此时mid必然是整数

mid = (currline[-1] + i) // 2

midList = [{1, 3}, {4, 6}, {7, 9}, {1, 7}, {2, 8}, {3, 9}, {1, 9}, {3, 7}]

if mid in currline&nbs***bsp;set([currline[-1], i]) not in midList:

return True

else:

return False

编辑于 2020-06-05 10:05:19

回复(0)

1

本来以为有什么规律,结果浪费了很多时间,没想到还是暴力DFS 就是剪枝有点麻烦

提交时间:2020-05-08 语言:Java 运行时间: 101 ms 占用内存:13312K 状态:答案正确

import java.util.*;

public class Solution {

/**

* 实现方案

* @param m int整型 最少m个键

* @param n int整型 最多n个键

* @return int整型

*/

private int sum = 0;

public int solution (int m, int n) {

if (m <= n) helper(new boolean[9], 0, m, n, -1);

return sum;

}

private void helper(boolean[] hash, int num, int m, int n, int curr) {

if (num == n) return;

for (int i = 0; i 

if (hash[i] || isCorr(hash, curr, i)) continue;

hash[i] = true;

if (num + 1 >= m) sum++;

helper(hash, num + 1, m, n, i);

hash[i] = false;

}

}

private boolean isCorr(boolean[] hash, int i, int j) {

if (i == -1) return false;

if (i > j) return isCorr(hash, j, i);

if (i == 0) {

if (j == 2 && !hash[1]) return true;

if (j == 6 && !hash[3]) return true;

if (j == 8 && !hash[4]) return true;

}

if (i == 2) {

if (j == 6 && !hash[4]) return true;

if (j == 8 && !hash[5]) return true;

}

if (i == 6 && j == 8 && !hash[7]) return true;

if (i + j == 8 && !hash[4]) return true;

return false;

}

}

发表于 2020-05-08 22:05:29

回复(0)

2

有人跟我一样想用Java的吗

发表于 2020-04-03 20:45:40

回复(1)

0

public  int solution (int m, int n) {

int arr[][] = {{1,2,3},{4,5,6},{7,8,9}} ;

//递归实现

int count = 0;

if(n

return 0;

}

if(m==0){

m=1;

}

n = n > 9 ? 9:n;

for(int i= m ;i <= n;i++){

for(int j=0;j<3;j++){

for(int k=0;k<3;k++){

List list = new ArrayList<>();

list.add(arr[j][k]);

count += findNext(list,arr,j,k,1,i);

}

}

}

return count;

}

public  int findNext(List list,int[][] arr,int j,int k,int step,int stepCount){

if(step >= stepCount){

//System.out.println(list);

return 1;

}

int[][] b= new int[][]{{1, 3,2}, {1, 7,4}, {1, 9,5}, {2, 8,5}, {3, 7,5}, {3, 9,6}, {4, 6,5}, {7, 9,8}};

int count = 0;

//上一步

for(int i= 0;i<3;i++){

for(int z=0;z<3;z++){

int s = step;

int lastStep = arr[j][k];

if(!list.contains(arr[i][z])){

boolean flag = true;

for(int r=0;r<8;r++){

//当之前的步骤中存在 则可以绘制

if(((b[r][0] == lastStep && arr[i][z] == b[r][1]) || (b[r][1] == lastStep && arr[i][z] == b[r][0])) && !list.contains(b[r][2])){

flag = false;

}

}

if(flag){

s ++;

List integers = new ArrayList<>(list);

integers.add(arr[i][z]);

//System.out.println(integers);

count+=  findNext(integers,arr,i,z,s,stepCount);

}

}

}

}

return count;

}

发表于 2020-09-25 15:36:29

回复(2)

0

将九宫格的九个格子当做0-8

如,  0,1,2

3,4,5

6,7,8

数组 d[i][j] = n 的意思是:如果当前格是i,下一格选择j,是否经过了某格,如果是0则说明没有经过任何格子,不为0则是经过的格子n, 比如d[0][6] = 3,说明从0到6需要经过3。

数组sign保存了某个数字是否已经被选择,所以在 i 格 选择 j 格为下一格需要经过 n 格, 只要判断 sign [ n ] 是否已经被选择。

class Solution {

public:

/**

* 实现方案

* @param m int整型 最少m个键

* @param n int整型 最多n个键

* @return int整型

*/

vector> d = {

{0,0,1,0,0,0,3,0,4},

{0,0,0,0,0,0,0,4,0},

{1,0,0,0,0,0,4,0,5},

{0,0,0,0,0,4,0,0,0},

{0,0,0,0,0,0,0,0,0},

{0,0,0,4,0,0,0,0,0},

{3,0,4,0,0,0,0,0,7},

{0,4,0,0,0,0,0,0,0},

{4,0,5,0,0,0,7,0,0}

};

int ans = 0;

int solution(int m, int n) {

// write code here

vector sign(9, false);

for(int i=m;i<=n;i++) {

for(int j=0;j<9;j++) {

sign[j] = true;

backtrack(j, sign, 1, i);

sign[j] = false;

}

}

return ans;

}

void backtrack(int cur, vector&sign, int cnt, int maxi) {

if(cnt==maxi)  {

ans++;

return;

}

for(int i=0;i<9;i++) {

if(i==cur)    continue;

if(d[cur][i]==0) {

if(!sign[i]) {

sign[ i ] = true;

backtrack(i, sign, cnt+1, maxi);

sign[ i ] = false;

}

}

else {

if(sign[ d[cur][i] ] && !sign[i]) {

sign[ i ] = true;

backtrack(i, sign, cnt+1, maxi);

sign[i] = false;

}

}

}

}

};

发表于 2020-08-15 19:04:07

回复(0)

0

#

# 实现方案

# @param m int整型 最少m个键

# @param n int整型 最多n个键

# @return int整型

#

class Solution:

def solution(self , m , n ):

# write code here

# deps[i][j] := i to j is ok when deps[i][j] exists

deps = {

1<<0:{1<<2:1<<1,1<<6:1<<3,1<<8:1<<4},

1<<1:{1<<7:1<<4},

1<<2:{1<<0:1<<1,1<<6:1<<4,1<<8:1<<5},

1<<3:{1<<5:1<<4},

1<<4:{},

1<<5:{1<<3:1<<4},

1<<6:{1<<0:1<<3,1<<2:1<<4,1<<8:1<<7},

1<<7:{1<<1:1<<4},

1<<8:{1<<0:1<<4,1<<2:1<<5,1<<6:1<<7},

}

def getlen(nums):

cnt = 0

while nums>0:

nums -= nums & -nums

cnt += 1

return cnt

def getbit(nums):

while nums > 0:

yield nums & -nums

nums -= nums & -nums

ans = [0] * 10

tables = {}

for state in range(1,1<<9):

tables[state] = {}

length = getlen(state)

if length == 1:

tables[state][state] = 1

ans[length] += 1

continue

for u in getbit(state):

tables[state][u] = 0

old_state = state - u

for v in getbit(old_state):

if u not in deps[v]&nbs***bsp;(old_state & deps[v][u]) > 0:

tables[state][u] += tables[old_state][v]

ans[length] += tables[state][u]

n = min(n,9)

total = 0

for i in range(m,n+1):

total += ans[i]

return total

发表于 2020-08-01 11:39:20

回复(0)

0

Python, 38 ms. 在init里面用状态压缩DP把每一个长度的和求出来,query的时候直接查就行了。记得在LC里面用这个方法runtime击败了96%的submission。

class Solution:

def __init__(self):

self.resTable = [0 for _ in range(10)]

premises = {(1<<0):{(1<<2):(1<<1),(1<<8):(1<<4),(1<<6):(1<<3)},

(1<<1):{(1<<7):(1<<4)},

(1<<2):{(1<<0):(1<<1),(1<<6):(1<<4),(1<<8):(1<<5)},

(1<<3):{(1<<5):(1<<4)},(1<<4):{},

(1<<5):{(1<<3):(1<<4)},

(1<<6):{(1<<0):(1<<3),(1<<2):(1<<4),(1<<8):(1<<7)},

(1<<7):{(1<<1):(1<<4)},

(1<<8):{(1<<2):(1<<5),(1<<0):(1<<4),(1<<6):(1<<7)}}

table = {}#mask:out:res

for m in xrange(1,1<<9):

table[m] = {}

m0,length = m,0

while m0>0:

length += 1

m0 -= m0&-m0

if m == (m&-m):

table[m][m] = 1

self.resTable[1] += 1

else:

m0 = m

while m0>0:

out0 = m0&-m0

table[m][out0] = 0

m1 = m-out0

m2 = m1

while m2>0:

out1 = m2&-m2

if out1 not in premises[out0] or (m1&premises[out0][out1]) > 0:

table[m][out0] += table[m1][out1]

m2 -= out1

m0 -= out0

self.resTable[length] += table[m][out0]

def solution(self , m , n ):

# write code here

n = min(n,9)

res = 0

for i in xrange(m,n+1):

res += self.resTable[i]

return res

编辑于 2020-06-30 17:32:05

回复(1)

0

典型的回溯问题。

主要问题是怎么选下一个点,一开始理解错了题目表达的意思。

其实题目表达的意思翻译一下就是,有两种非法访问:

1. 如果当前键到下一个键会经过中间键,那么这个中间键必须已经访问过了,若还没访问那就非法

2. 下一个键若已经访问过了是非法。

import java.util.*;

public class Solution {

public static int pathSum = 0;

static class Pos{

int x;

int y;

Pos(int x,int y){

this.x = x;

this.y = y;

}

}

public int solution(int m,int n){

//异常处理

if(m > n || m  9){

return 0;

}

if(n>9)

n=9;

//保存访问情况

int[][] map = new int[3][3];

while(m <= n){

//从9个点中选择一个起点

for(int i=0;i<3;i++)

for(int j=0;j<3;j++){

for(int k=0;k

Arrays.fill(map[k],0);

Pos start = new Pos(i,j);

map[start.x][start.y] = 1;

BackTrack(start,m,1,map);

}

m++;

}

return pathSum;

}

public void BackTrack(Pos start, int maxLen,int pathLen, int[][] map){

if(pathLen == maxLen){

pathSum++;

return;

}else {

//遍历9个点

for (int i = 0; i <= 2; i++) {

for (int j = 0; j <= 2; j++) {

Pos newStart = new Pos(i, j);

//检验新的键是否符合要求

if (canVisit(map, start, newStart)) {

map[newStart.x][newStart.y] = 1;

//回溯

BackTrack(newStart, maxLen,pathLen+1,map);

map[newStart.x][newStart.y] = 0;

}

}

}

}

}

public boolean canVisit(int[][] map,Pos curPos, Pos newPos){

//首先新键和当前键不能相同

if(!(curPos.x == newPos.x && curPos.y == newPos.y)){

//两个键之间存在中间键

if((curPos.x-newPos.x)%2 == 0 &&(curPos.y-newPos.y)%2 == 0){

Pos mid = new Pos((curPos.x+newPos.x)/2,(curPos.y+newPos.y)/2);

//中间键已经访问过了

if(map[mid.x][mid.y] == 1){

return map[newPos.x][newPos.y] == 0;

}else

return false;

}else {

//其他情况只要是之前没访问过的键都行

return map[newPos.x][newPos.y] == 0;

}

}

return false;

}

}

发表于 2020-06-19 23:18:47

回复(0)

0

DFS,深度遍历。

用一个栈存放键的位置int[]{i,j},遍历到下一个没访问的键。判断这个键的位置与栈顶(peek)键位置关系。

(1)在同一横行,Math.abs(i-peek[0])>1,j==peek[1];

(2)在同一竖行,Math.abs(j-peek[1])>1,i==peek[0];

(3)在对角线上,Math.abs(j-peek[1])>1,Math.abs(i-peek[0])>1;

满足之一,就判断它们的中点是否被访问过:key[(i+temp[0])/2][(j+temp[1])/2]==0?

等于0,不满足有效模式,return;

不等于0,满足有效模式,继续递归;

不满足上面三个位置关系,肯定满足有效模式,继续递归; public class Solution {

int m,n;

int sum=0;

public int solution (int m, int n) {

if(n==0) return 0;

if(m==0) m=1;

this.m=m;

this.n=n;

int[][] key=new int[3][3];

Stack ls=new Stack<>();

dfs(key,ls);

return sum;

}

public void dfs(int[][] key,Stack ls){

if(ls.size()>=n){

sum++;

return;

}

if(ls.size()>=m && ls.size()

sum++;

}

for(int i=0;i<3;i++){

for(int j=0;j<3;j++){

if(key[i][j]!=0) continue;

if(ls.isEmpty()){

key[i][j]=1;

ls.push(new int[]{i,j});

dfs(key,ls);

key[i][j]=0;

ls.pop();

}else{

int[] temp=ls.peek();

if((Math.abs(i-temp[0])>1 && j==temp[1]) ||

(Math.abs(j-temp[1])>1 && i==temp[0]) ||

(Math.abs(j-temp[1])>1 && Math.abs(i-temp[0])>1)){

if(key[(i+temp[0])/2][(j+temp[1])/2]!=0){

key[i][j]=1;

ls.push(new int[]{i,j});

dfs(key,ls);

key[i][j]=0;

ls.pop();

}else{

continue;

}

}else{

key[i][j]=1;

ls.push(new int[]{i,j});

dfs(key,ls);

key[i][j]=0;

ls.pop();

}

}

}

}

}

}

发表于 2020-06-06 22:57:47

回复(0)

最后

以上就是怕黑麦片为你收集整理的python123九宫格输入法_手机屏幕解锁模式的全部内容,希望文章能够帮你解决python123九宫格输入法_手机屏幕解锁模式所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(52)

评论列表共有 0 条评论

立即
投稿
返回
顶部