算法训练Day30 回溯算法专题 | LeetCode332. 重新安排行程;51.N皇后(棋盘问题);37.解数独(二维的递归)

news2025/1/17 23:25:37

前言:

算法训练系列是做《代码随想录》一刷,个人的学习笔记和详细的解题思路,总共会有60篇博客来记录,计划用60天的时间刷完。 

内容包括了面试常见的10类题目,分别是:数组,链表,哈希表,字符串,栈与队列,二叉树,回溯算法,贪心算法,动态规划,单调栈。

博客记录结构上分为 思路,代码实现,复杂度分析,思考和收获,四个方面。

如果这个系列的博客可以帮助到读者,就是我最大的开心啦,一起LeetCode一起进步呀;)

目录

LeetCode332. 重新安排行程

1. 思路

2. 代码实现

3. 复杂度分析

4. 思考与收获

Leetcode51. N皇后

1. 思路

2. 代码实现

3. 复杂度分析

4. 思考与收获

Leetcode37. 解数独

1. 思路

2. 代码实现

3. 复杂度分析

4. 思考与收获


LeetCode332. 重新安排行程

链接:332. 重新安排行程 - 力扣(LeetCode) 

1. 思路

直觉上来看 这道题和回溯法没有什么关系,更像是图论中的深度优先搜索。

实际上确实是深搜,但这是深搜中使用了回溯的例子,在查找路径的时候,如果不回溯,怎么能查到目标路径呢。

所以我倾向于说本题应该使用回溯法,那么我也用回溯法的思路来讲解本题,其实深搜一般都使用了回溯法的思路,在图论系列中我会再详细讲解深搜。

这里就是先给大家拓展一下,原来回溯法还可以这么玩!

这道题目有几个难点:

  1. 一个行程中,如果航班处理不好容易变成一个圈,成为死循环
  2. 有多种解法,字母序靠前排在前面,让很多同学望而退步,如何该记录映射关系呢 ?
  3. 使用回溯法(也可以说深搜) 的话,那么终止条件是什么呢?
  4. 搜索的过程中,如何遍历一个机场所对应的所有机场。

针对以上问题我来逐一解答!

如何理解死循环?

为什么要举这个例子呢,就是告诉大家,出发机场和到达机场也会重复的,如果在解题的过程中没有对集合元素处理好,就会死循环;

记录映射关系?

有多种解法,字母序靠前排在前面,让很多同学望而退步,如何该记录映射关系呢 ?

一个机场映射多个机场,机场之间要靠字母序排列,一个机场映射多个机场,可以使用std::unordered_map,如果让多个机场之间再有顺序的话,就是用std::map 或者std::multimap 或者 std::multiset。

如果对map 和 set 的实现机制不太了解,也不清楚为什么 map、multimap就是有序的同学,可以看这篇文章**关于哈希表,你该了解这些! (opens new window)**。

这样存放映射关系可以定义为 unordered_map<string, multiset<string>> targets 或者 unordered_map<string, map<string, int>> targets

含义如下:

unordered_map<string, multiset> targets:unordered_map<出发机场, 到达机场的集合> targets

unordered_map<string, map<string, int>> targets:unordered_map<出发机场, map<到达机场, 航班次数>> targets

这两个结构,我选择了后者,因为如果使用unordered_map<string, multiset<string>> targets 遍历multiset的时候,不能删除元素,一旦删除元素,迭代器就失效了。

再说一下为什么一定要增删元素呢,正如开篇我给出的图中所示,出发机场和到达机场是会重复的,搜索的过程没及时删除目的机场就会死循环。

所以搜索的过程中就是要不断的删multiset里的元素,那么推荐使用unordered_map<string, map<string, int>> targets

在遍历 unordered_map<出发机场, map<到达机场, 航班次数>> targets的过程中,可以使用"航班次数"这个字段的数字做相应的增减,来标记到达机场是否使用过了。

如果“航班次数”大于零,说明目的地还可以飞,如果如果“航班次数”等于零说明目的地不能飞了,而不用对集合做删除元素或者增加元素的操作。

相当于说我不删,我就做一个标记!

回溯法

这道题目我使用回溯法,那么下面按照我总结的回溯模板来:

void backtracking(参数) {
    if (终止条件) {
        存放结果;
        return;
    }

    for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {
        处理节点;
        backtracking(路径,选择列表); // 递归
        回溯,撤销处理结果
    }
}

本题以输入:[["JFK", "KUL"], ["JFK", "NRT"], ["NRT", "JFK"]为例,抽象为树形结构如下:

2. 代码实现

2.1 递归函数参数

在讲解映射关系的时候,已经讲过了,使用unordered_map<string, map<string, int>> targets; 来记录航班的映射关系,我定义为全局变量。

当然把参数放进函数里传进去也是可以的,我是尽量控制函数里参数的长度。

参数里还需要ticketNum,表示有多少个航班(终止条件会用上)。

代码如下:

class Solution(object):
    def __init__(self):
        self.path = ["JFK"]
        self.tickets_dict = collections.defaultdict(list)

    def findItinerary(self, tickets):

		def backtracking(self,tickets,startPoint):

注意函数返回值我用的是bool!

我们之前讲解回溯算法的时候,一般函数返回值都是void,这次为什么是bool呢?

因为我们只需要找到一个行程,就是在树形结构中唯一的一条通向叶子节点的路线;所以找到了这个叶子节点了直接返回,这个递归函数的返回值问题我们在讲解二叉树的系列的时候,在这篇**二叉树:递归函数究竟什么时候需要返回值,什么时候不要返回值? (opens new window)**详细介绍过。

当然本题的targets和result都需要初始化;

2.2 递归终止条件

拿题目中的示例为例,输入: [["MUC", "LHR"], ["JFK", "MUC"], ["SFO", "SJC"], ["LHR", "SFO"]] ,这是有4个航班,那么只要找出一种行程,行程里的机场个数是5就可以了;所以终止条件是:我们回溯遍历的过程中,遇到的机场个数,如果达到了(航班数量+1),那么我们就找到了一个行程,把所有航班串在一起了。

if len(path) == len(tickets) + 1:
      return True

已经看习惯回溯法代码的同学,到叶子节点了习惯性的想要收集结果,但发现并不需要,本题的result相当于 回溯算法:求组合总和! (opens new window) 中的path,也就是本题的result就是记录路径的(就一条),在如下单层搜索的逻辑中result就添加元素了;

2.3 单层搜索的逻辑

回溯的过程中,如何遍历一个机场所对应的所有机场呢?

C++ 语言特性的讨论:(一刷没看)

这里刚刚说过,在选择映射函数的时候,不能选择unordered_map<string, multiset<string>> targets, 因为一旦有元素增删multiset的迭代器就会失效,当然可能有牛逼的容器删除元素迭代器不会失效,这里就不在讨论了。

可以说本题既要找到一个对数据进行排序的容器,而且还要容易增删元素,迭代器还不能失效

所以我选择了unordered_map<string, map<string, int>> targets 来做机场之间的映射;

可以看出 通过unordered_map<string, map<string, int>> targets 里的int字段来判断 这个集合里的机场是否使用过,这样避免了直接去删元素;

Python 写法:

# 单层搜索逻辑
for _ in self.tickets_dict[startPoint]:
    # 必须及时删除,避免出现死循环
    endPoint = self.tickets_dict[startPoint].pop(0)
    self.path.append(endPoint)
    if self.backtracking(tickets,endPoint):
        return True
    self.tickets_dict[startPoint].append(endPoint)
    self.path.pop()

2.4 整体代码实现

import collections
class Solution(object):
    def __init__(self):
        self.path = ["JFK"]
        self.tickets_dict = collections.defaultdict(list)

    def findItinerary(self, tickets):
        """
        :type tickets: List[List[str]]
        :rtype: List[str]
        """
        # 把ticket数组中的机票都放进dictionary里面
        for item in tickets:
            if item[0] in self.tickets_dict:
                self.tickets_dict[item[0]].append(item[1])
            else: self.tickets_dict[item[0]] = [item[1]]
        # 把每个出发地后面的数组排序
        for airport in self.tickets_dict: 
            self.tickets_dict[airport].sort()
        self.backtracking(tickets,"JFK")
        return self.path
        
    def backtracking(self,tickets,startPoint):
        # 终止条件
        if len(self.path) == len(tickets)+1:
            return True 
        # 单层搜索逻辑
        for _ in self.tickets_dict[startPoint]:
            # 必须及时删除,避免出现死循环
            endPoint = self.tickets_dict[startPoint].pop(0)
            self.path.append(endPoint)
            if self.backtracking(tickets,endPoint):
                return True
            self.tickets_dict[startPoint].append(endPoint)
            self.path.pop()

3. 复杂度分析

(自己分析的,不确定)

  • 时间复杂度O(N logN )

    N为tickets数组里面机票的个数,首先需要遍历存入数组中O(N),然后需要排序,O(NlogN),然后在backtracking里面,需要遍历每一个pair处理O(N);总体来说O(NlogN);

  • 空间复杂度:O(N)

    主要是递归的栈的深度,最大为O(N);

4. 思考与收获

  1. Python中的defaultdict():

    1,collections.defaultdict类的介绍:

    defaultdict是Python内建dict类的一个子类,第一个参数为default_factory属性提供初始值,默认为None。它覆盖一个方法并添加一个可写实例变量。它的其他功能与dict相同,但会为一个不存在的键提供默认值,从而避免KeyError异常。

    2,一般的dict类型会导致KeyError异常:

    一般dict类型:

         KeyError异常:

         defaultdict类避免KeyError异常:

  1. 因为这个是找到一个满足条件的立即返回即可,所以要在backtracking的递归中加上一个if 条件,满足了就立即返回;

Reference:

  1. Python中的defaultdict方法 - 知乎 (zhihu.com)
  2. 代码随想录 (programmercarl.com)

本题学习时间:80分钟。


Leetcode51. N皇后

链接:51. N 皇后 - 力扣(LeetCode)

1. 思路

都知道n皇后问题是回溯算法解决的经典问题,但是用回溯解决多了组合、切割、子集、排列问题之后,之前解决的问题都是一维的回溯问题,遇到这种二维矩阵还会有点不知所措;

首先来看一下皇后们的约束条件:

  1. 不能同行
  2. 不能同列
  3. 不能同斜线

确定完约束条件,来看看究竟要怎么去搜索皇后们的位置,其实搜索皇后的位置,可以抽象为一棵树;下面用一个 3 * 3 的棋盘,将搜索过程抽象为一棵树,如图:

从图中,可以看出,深度就是矩阵的高度N,宽度就是树形结构中就是矩阵的宽度N,那么我们用皇后们的约束条件,来回溯搜索这棵树,只要搜索到了树的叶子节点,说明就找到了皇后们的合理位置了;

2. 代码实现

按照总结的如下回溯模板,我们来依次分析:

void backtracking(参数) {
    if (终止条件) {
        存放结果;
        return;
    }
    for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {
        处理节点;
        backtracking(路径,选择列表); // 递归
        回溯,撤销处理结果
    }
}

2.1 递归函数参数

我依然是定义全局变量二维数组result来记录最终结果;

参数n是棋盘的大小,然后用row来记录当前遍历到棋盘的第几层了;

class Solution:
    def solveNQueens(self, n: int) -> List[List[str]]:
    def backtracking(board, row, n):

2.2 递归终止条件

在如下树形结构中:

可以看出,当递归到棋盘最底层(也就是叶子节点)的时候,就可以收集结果并返回了;代码如下:

if row == n:
    temp_res = []
    for temp in board:
        temp_str = "".join(temp)
        temp_res.append(temp_str)
    res.append(temp_res)

2.3 单层搜索的逻辑

递归深度就是row控制棋盘的行,每一层里for循环的col控制棋盘的列,一行一列,确定了放置皇后的位置;每次都是要从新的一行的起始位置开始搜,所以都是从0开始;

 for col in range(n):
	    if not isVaild(board, row, col):
	        continue
	    board[row][col] = 'Q'
	    backtracking(board, row+1, n)
	    board[row][col] = '.'

2.4 验证棋盘是否合法

按照如下标准去重:

  1. 不能同行
  2. 不能同列
  3. 不能同斜线 (45度和135度角)

代码如下:

def isVaild(board,row, col):
    #判断同一列是否冲突
    for i in range(len(board)):
        if board[i][col] == 'Q':
            return False
    # 判断左上角是否冲突
    i = row -1
    j = col -1
    while i>=0 and j>=0:
        if board[i][j] == 'Q':
            return False
        i -= 1
        j -= 1
    # 判断右上角是否冲突
    i = row - 1
    j = col + 1
    while i>=0 and j < len(board):
        if board[i][j] == 'Q':
            return False
        i -= 1
        j += 1
    return True

在这份代码中,细心的同学可以发现为什么没有在同行进行检查呢?

因为在单层搜索的过程中,每一层递归,只会选for循环(也就是同一行)里的一个元素,所以不用去重了;

2.5 整体代码实现

# 回溯算法 N皇后问题
# time: O(N!);space:O(N)
class Solution(object):
    def __init__(self):
        self.result = []
        self.board = []
    def solveNQueens(self, n):
        """
        :type n: int
        :rtype: List[List[str]]
        """
        if n==0: return []
        self.board = [["."] * n for _ in range(n)]
        self.backtracking(0,n)
        return self.result
    
    def backtracking(self,row,n):
        # 终止条件,如果走到最后一行,说明已经找到一个解
        if row == n:
            temp_res = []
            for temp in self.board:
                temp_str = "".join(temp)
                temp_res.append(temp_str)
            self.result.append(temp_res)
        for col in range(n):
            if not self.isValid(row,col,n):
                continue
            self.board[row][col] = "Q"
            self.backtracking(row+1,n)
            self.board[row][col] = "."
    
    def isValid(self,row,col,n):
        # 判断同一列
        for i in range(n):
            if self.board[i][col] == "Q":
                return False 
        # 判断左上角那条线
        i = row -1
        j = col -1
        while i >= 0 and j>=0:
            if self.board[i][j] == "Q":
                return False
            i -= 1
            j -= 1
        # 判断右上角那条线
        i = row -1
        j = col +1
        while i >=0 and j <n:
            if self.board[i][j] == "Q":
                return False
            i -= 1
            j += 1
        return True

3. 复杂度分析

  • 时间复杂度:O(n!)

    其实如果看树形图的话,直觉上是O(n^n),但皇后之间不能见面所以在搜索的过程中是有剪枝的,最差也就是O(n!),n!表示n * (n-1) * .... * 1;

  • 空间复杂度:O(n)

    递归深度为n,所以系统栈所用空间为O(n),每一层递归所用的空间都是常数级别,注意代码里的result和path都是全局变量,就算是放在参数里,传的也是引用,并不会新申请内存空间,最终空间复杂度为O(n);

4. 思考与收获

  1. 本题是我们解决棋盘问题的第一道题目,如果从来没有接触过N皇后问题的同学看着这样的题会感觉无从下手,可能知道要用回溯法,但也不知道该怎么去搜。这里我明确给出了棋盘的宽度就是for循环的长度,递归的深度就是棋盘的高度,这样就可以套进回溯法的模板里了;

Rreference:代码随想录 (programmercarl.com)

本题学习时间:60分钟


Leetcode37. 解数独

链接:37. 解数独 - 力扣(LeetCode)

1. 思路

棋盘搜索问题可以使用回溯法暴力搜索,只不过这次我们要做的是二维递归

怎么做二维递归呢?

对比之前的题?

大家已经跟着「代码随想录」刷过了如下回溯法题目,例如:77.组合(组合问题) (opens new window)131.分割回文串(分割问题) (opens new window)78.子集(子集问题) (opens new window)46.全排列(排列问题) (opens new window),以及**51.N皇后(N皇后问题) (opens new window)**,其实这些题目都是一维递归。

**N皇后问题 (opens new window)**是因为每一行每一列只放一个皇后,只需要一层for循环遍历一行,递归来来遍历列,然后一行一列确定皇后的唯一位置。

本题就不一样了,本题中棋盘的每一个位置都要放一个数字(而N换后是一行只放一个皇后),并检查数字是否合法,解数独的树形结构要比N皇后更宽更深。

因为这个树形结构太大了,我抽取一部分,如图所示:

2. 代码实现

2.1 递归函数及参数

**递归函数的返回值需要是bool类型,为什么呢?**因为解数独找到一个符合的条件(就在树的叶子节点上)立刻就返回,相当于找从根节点到叶子节点一条唯一路径,所以需要使用bool返回值;

class Solution:
    def solveSudoku(self, board: List[List[str]]) -> None:

    def backtracking(self, board: List[List[str]]) -> bool:

2.2 递归终止条件

本题递归不用终止条件,解数独是要遍历整个树形结构寻找可能的叶子节点就立刻返回。

不用终止条件会不会死循环?

递归的下一层的棋盘一定比上一层的棋盘多一个数,等数填满了棋盘自然就终止(填满当然好了,说明找到结果了),所以不需要终止条件!

**那么有没有永远填不满的情况呢?**这个问题在递归单层搜索逻辑里在来讲!

2.3 递归单层搜索逻辑

在树形图中可以看出我们需要的是一个二维的递归(也就是两个for循环嵌套着递归);一个for循环遍历棋盘的行,一个for循环遍历棋盘的列,一行一列确定下来之后,递归遍历这个位置放9个数字的可能性;

 def backtracking(self, board: List[List[str]]) -> bool:
    # 若有解,返回True;若无解,返回False
    for i in range(len(board)): # 遍历行
        for j in range(len(board[0])):  # 遍历列
            # 若空格内已有数字,跳过
            if board[i][j] != '.': continue
            for k in range(1, 10):
                if self.is_valid(i, j, k, board):
                    board[i][j] = str(k)
                    if self.backtracking(board): return True
                    board[i][j] = '.'
            # 若数字1-9都不能成功填入空格,返回False无解
            return False
    return True # 有解

注意这里return false的地方,这里放return false 是有讲究的

因为如果一行一列确定下来了,这里尝试了9个数都不行,说明这个棋盘找不到解决数独问题的解!那么会直接返回, 这也就是为什么没有终止条件也不会永远填不满棋盘而无限递归下去;

2.4 判断棋盘是否合法

判断棋盘是否合法有如下三个维度:

  • 同行是否重复
  • 同列是否重复
  • 9宫格里是否重复
def is_valid(self, row: int, col: int, val: int, board: List[List[str]]) -> bool:
	    # 判断同一行是否冲突
	    for i in range(9):
	        if board[row][i] == str(val):
	            return False
	    # 判断同一列是否冲突
	    for j in range(9):
	        if board[j][col] == str(val):
	            return False
	    # 判断同一九宫格是否有冲突
	    start_row = (row // 3) * 3
	    start_col = (col // 3) * 3
	    for i in range(start_row, start_row + 3):
	        for j in range(start_col, start_col + 3):
	            if board[i][j] == str(val):
	                return False
	    return True

2.5 整体代码实现

# 回溯算法  解数独
# time:O(9^m),m为“.”的数目;space:O(N^2)
class Solution(object):
    def solveSudoku(self, board):
        """
        :type board: List[List[str]]
        :rtype: None Do not return anything, modify board in-place instead.
        """
        self.backtracking(board)
        return board
    def backtracking(self,board):
        for i in range(len(board)):  # 遍历行
            for j in range(len(board[0])): # 遍历列
                if board [i][j] != "." : continue 
                for k in range(1,10):
                    if self.isValid(i,j,k,board):
                        board[i][j] = str(k)
                        if self.backtracking(board): return True 
                        board[i][j] = "."
                # 如果1-9都不能成功填入空格,return False无解
                return False 
        # 两层for循环都遍历完了,表格全部填满了,有解了
        return True 
    def isValid(self,row,col,val,board):
        # 判断同一行是否冲突
        for i in range(9):
            if board[row][i] == str(val): 
                return False
        # 判断同一列是否冲突
        for j in range(9):
            if board[j][col] == str(val):
                return False
        # 判断同一个九宫格是否冲突
        startRow = (row//3)*3
        startCol = (col//3)*3
        for i in range(startRow,startRow+3):
            for j in range(startCol,startCol+3):
                if board[i][j] == str(val):
                    return False
        return True

3. 复杂度分析

  • 时间复杂度:O(9^m)

    m是'.'的数目。

  • 空间复杂度:O(n^2)

    递归的深度是n^2,N为数独的长或者宽

4. 思考与收获

  1. 解数独可以说是非常难的题目了,如果还一直停留在单层递归的逻辑中,这道题目可以让大家瞬间崩溃,所以我在开篇就提到了二维递归,这也是Carl自创词汇,希望可以帮助大家理解解数独的搜索过程,一波分析之后,在看代码会发现其实也不难,唯一难点就是理解二维递归的思维逻辑。这样,解数独这么难的问题,也被我们攻克了

Reference:代码随想录 (programmercarl.com)

本题学习时间:60分钟。


本篇学习时间为3个多小时,总结字数10000+;本篇的三道题都很难,重新安排行程给我们打开了回溯问题的其他思路;N皇后和解数独这两个棋盘问题,涉及到二维的回溯算法解题。(求推荐!)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/44591.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

android 自定义view: 跑马灯-光圈

本系列自定义View全部采用kt **系统: **mac android studio: 4.1.3 **kotlin version:**1.5.0 gradle: gradle-6.5-bin.zip 本篇效果: 前沿 最近在bilibili看到一个跑马灯光圈效果挺好, 参考着思路写了一下. bilibili地址,美中不足的是这是html代码 QaQ 实现思路 将效果…

Spark系列之Spark启动与基础使用

title: Spark系列 第三章 Spark启动与基础使用 3.1 Spark Shell 3.1.1 Spark Shell启动 安装目录的bin目录下面&#xff0c;启动命令&#xff1a; spark-shell$SPARK_HOME/bin/spark-shell \ --master spark://hadoop10:7077 \ --executor-memory 512M \ --total-executor-…

长短期记忆网络(LSTM)重点!(自己汇集了很多资料在这篇博客)

文章目录参考资料推荐基础知识评论区精髓代码实现底层实现简洁实现参考资料推荐 心心念念 学了这么久 &#xff0c;终于学到第57集了。 参考一篇掘金的图文LSTM 李宏毅老师的手撕视频配套课件 27:39 开始手撕 看完了李沐老师的LSTM又去找了李宏毅老师的课程然后发现又多了个导…

西门子机床联网

一、设备信息确认 1、确认型号 数控面板拍照确认&#xff1a; 此系统为&#xff1a;西门子828D 还有一种情况是面板无任何版本信息&#xff0c;这时就需要进入系统里面再确认。 2、确认通讯接口 1、数控面板的后面 X130为网络标号 2、其他位置 其他位置一般是前面位置用…

H3C mstp+vrrp实验 新华三杯拆解

H3C mstpvrrp实验一、实验拓扑二、实验要求局域网规划&#xff1a;可靠性&#xff1a;三、实验配置&#xff08;一&#xff09;链路聚合1.创建链路聚合组2.检查&#xff08;二&#xff09;VLAN1.创建vlan2.放行vlan3.检查&#xff08;三&#xff09;MSTP1.配置MSTP域2.配置主备…

通俗易懂的java设计模式(1)-单例模式

什么是单例模式&#xff1f; 单例模式是java中最简单的一种设计模式 需要注意的问题&#xff1a; 1.单例类有且只能有一个实例 2.单例类必须自己创建出这个实例&#xff0c;并提供给外界 那么如何自己创建实例而不让外界创建呢&#xff1f;很简单&#xff0c;我们将无参的构造函…

麦芽糖-刀豆球蛋白A,maltose-ConcanavalinA,刀豆球蛋白A-PEG-麦芽糖

麦芽糖-刀豆球蛋白A,maltose-ConcanavalinA,刀豆球蛋白A-PEG-麦芽糖 中文名称&#xff1a;麦芽糖-刀豆球蛋白A 英文名称&#xff1a;maltose-ConcanavalinA 别称&#xff1a;刀豆球蛋白A修饰麦芽糖&#xff0c;ConA-麦芽糖 还可以提供PEG接枝修饰麦芽糖&#xff0c;麦芽糖…

SpringCloud-alibaba-Nacos 从理论到落地使用

Nacos: Dynamic Naming and Configuration Service 就是&#xff1a; 一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。、 下面是生态图&#xff1a; Nacos EurekaConfig Bus 即 Nacos就是注册中心 配置中心的组合 他能干什么呢&#xff1f; 1、替代Eu…

防火墙firewalld

RHEL7中有几种防火墙共存&#xff1a;firewalld、iptables、ebtables等。基于iptables的防火墙默认不启动&#xff0c;但仍然可以继续使用。RHEL7默认使用firewalld作为防火墙&#xff0c;管理工具是firewall-cmd。RHEL7的内核版本是3.10&#xff0c;在此版本的内核里防火墙的包…

论文阅读2 Learning Hierarchy-Aware Knowledge Graph Embeddings for Link Prediction

目录 问题 创新 1、Introduction 2、相关工作 3、HAKE模型 原文&#xff1a;[1911.09419] Learning Hierarchy-Aware Knowledge Graph Embeddings for Link Prediction (arxiv.org) 问题 现有的知识图嵌入模型主要关注对称/反对称、反转和复合symmetry/antisymmetry, i…

EquiVSet

又搬来一个于最优子集的神经集合函数学习方法 集合函数被广泛应用于各种场景之中&#xff0c;例如商品推荐、异常检测和分子筛选等。在这些场景中&#xff0c;集合函数可以被视为一个评分函数&#xff1a;其将一个集合作为输入并输出该集合的分数。我们希望从给定的集合中选取…

【新知实验室-TRTC开发】实时音视频之欢度世界杯

目录 一、什么是TRTC 二、用5分钟跑通一个demo 1、开通腾讯云-TRTC 2、获取demo必须的两把钥匙 2.1输入应用名称 2.2下载对应的源码包&#xff08;手机、web、小程序等&#xff09; 2.3拿到钥匙 2.4完成 三、搭建一起看世界杯应用 1、解压源码&#xff08;耗时30S&#x…

Linux下top命令详解

Linux下top命令用法详解 作为一名Linux软件攻城狮&#xff0c;top命令大家应该并不陌生。top命令是Linux下常用的性能分析工具&#xff0c;能够实时显示系统中各个进程的资源占用状况。top可以动态显示过程,不断刷新当前状态。top命令提供了实时的对系统处理器的状态监视。它将…

Kotlin高仿微信-第5篇-主页-通讯录

Kotlin高仿微信-项目实践58篇详细讲解了各个功能点&#xff0c;包括&#xff1a;注册、登录、主页、单聊(文本、表情、语音、图片、小视频、视频通话、语音通话、红包、转账)、群聊、个人信息、朋友圈、支付服务、扫一扫、搜索好友、添加好友、开通VIP等众多功能。 Kotlin高仿…

【博客547】keepalived实现vip的原理剖析

keepalived实现vip的原理剖析 keepalived实现vip的原理&#xff1a;vrrp gratuitous arp 1、vrrp vrrp更多细节参考&#xff1a;vrrp技术白皮书 相关术语&#xff1a; VRRP工作过程 VRRP的工作过程为&#xff1a; 1、虚拟路由器中的路由器根据优先级选举出Master。Master…

【Python】五、程序循环结构

文章目录实验目的一、掌握while语句二、掌握for循环和range()内建函数三、掌握循环语句嵌套四、掌握break语句和continue语句五、编写程序实现猜数字的游戏1.设计思路2.设计算法3.参考代码4.实验截图实验目的 掌握循环结构&#xff1b;培养学生动手查阅资料能力和解决实际问题的…

matlab实现线性参数的最小二乘法处理

一、实验目的 最小二乘法原理是一种在多学科领域中获得广泛应用的数据处理方法。通过实验要求掌握最小二乘法基本原理、正规方程以及组合测量的最小二乘法处理办法。 二、实验原理 &#xff08;1&#xff09;测量结果的最可信赖值应在残余误差平方和为最小的条件下求出&#…

破局模块总结 -- 宁向东的清华管理学课总结

1. 管理学就是要破局而出 为什么需要管理学&#xff1f;德鲁克说&#xff0c;我们需要选择正确的事情去做&#xff0c;并且把事情作对。 管理学是一个分析、权衡和决策的学问&#xff0c;分析、权衡和选择的目的是什么&#xff1f;就是要破局而出。 管理學是研究人類管理活動…

《Linux运维总结:基于快照模式迁移单节点elasticsearch数据(方案二)》

一、背景信息 说明&#xff1a;由于整个系统需要从互联网迁移到政务外网&#xff0c;elasticsearch作为其中一个组件&#xff0c;也需要将 所有索引数据 迁移到政务外网。 由于数据量比较大&#xff0c;所以使用快照的模式对elasticsearch数据进行备份及恢复操作&#xff0c;提…

[附源码]计算机毕业设计springboot病人跟踪治疗信息管理系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…