第13课-字典树和并查集

字典树 Trie

内容

  1. 字典树的数据结构
  2. 字典树的核心思想
  3. 字典树的基本性质

在这里插入图片描述

在这里插入图片描述

基本结构

在这里插入图片描述

基本性质

  1. 结点本身不存完整单词;
  2. 从根结点到某一结点,路径上经过的字符连接起来,为该结点对应的 字符串;
  3. 每个结点的所有子结点路径代表的字符都不相同。

在这里插入图片描述
在这里插入图片描述

核心思想

  • Trie 树的核心思想是空间换时间。
  • 利用字符串的公共前缀来降低查询时间的开销以达到提高效率的目的。

实战题目

  1. https://leetcode-cn.com/problems/implement-trie-prefix-tree/
  2. https://leetcode-cn.com/problems/word-search-ii/
  3. Search suggestion - system design
 class Trie(object):
  def __init__(self):
    self.root = {}
    self.end_of_word = "#"
  def insert(self, word):
    node = self.root
    for char in word:
      node = node.setdefault(char, {})
    node[self.end_of_word] = self.end_of_word
  def search(self, word):
    node = self.root
    for char in word:
      if char not in node:
        return False
      node = node[char]
    return self.end_of_word in node
  def startsWith(self, prefix):
    node = self.root
    for char in prefix:
      if char not in node:
        return False
      node = node[char]
    return True

Trie: https://leetcode-cn.com/problems/word-search-ii/

	dx = [-1, 1, 0, 0]
    dy = [0, 0, -1, 1]
    END_OF_WORD = "#"
    class Solution(object):
       def findWords(self, board, words):
         if not board or not board[0]: return []
         if not words: return []
         self.result = set()
# 构建trie
         root = collections.defaultdict()
         for word in words:
           node = root
           for char in word:
             node = node.setdefault(char, collections.defaultdict())
           node[END_OF_WORD] = END_OF_WORD
         self.m, self.n = len(board), len(board[0])
         for i in xrange(self.m):
           for j in xrange(self.n):
             if board[i][j] in root:
               self._dfs(board, i, j, "", root)
         return list(self.result)
 def _dfs(self, board, i, j, cur_word, cur_dict):
  cur_word += board[i][j]
  cur_dict = cur_dict[board[i][j]]
  if END_OF_WORD in cur_dict:
   self.result.add(cur_word)
  tmp, board[i][j] = board[i][j], '@'
  for k in xrange(4):
   x, y = i + dx[k], j + dy[k]
   if 0 <= x < self.m and 0 <= y < self.n \
     and board[x][y] != '@' and board[x][y] in cur_dict:
     self._dfs(board, x, y, cur_word, cur_dict)
  board[i][j] = tmp

并查集 Disjoint Set

适用场景

• 组团、配对问题
• Group or not ?

基本操作

• makeSet(s):建立一个新的并查集,其中包含 s 个单元素集合。
• unionSet(x, y):把元素 x 和元素 y 所在的集合合并,要求 x 和 y 所在
的集合不相交,如果相交则不合并。
• find(x):找到元素 x 所在的集合的代表,该操作也可以用于判断两个元 素是否位于同一个集合,只要将它们各自的代表比较一下就可以了。

初始化

在这里插入图片描述

查询、合并

在这里插入图片描述

路径压缩

在这里插入图片描述

Java 实现

class UnionFind {
  private int count = 0;
  private int[] parent;
  public UnionFind(int n) {
   count = n;
   parent = new int[n];
   for (int i = 0; i < n; i++) {
     parent[i] = i;
   }
  }
  public int find(int p) {
   while (p != parent[p]) {
     parent[p] = parent[parent[p]];
     p = parent[p];
   }
   return p; 
  }

  public void union(int p, int q) {
   int rootP = find(p);
   int rootQ = find(q);
   if (rootP == rootQ) return;
   parent[rootP] = rootQ;
count--; }
}
 

Python实现

def init(p):
  # for i = 0 .. n: p[i] = i;
  p = [i for i in range(n)]
def union(self, p, i, j):
  p1 = self.parent(p, i)
  p2 = self.parent(p, j)
  p[p1] = p2
def parent(self, p, i):
  root = i
  while p[root] != root:
root = p[root]
while p[i] != i: # 路径压缩 ?
   x = i; i = p[i]; p[x] = root
  return root

实战题目

https://leetcode-cn.com/problems/friend-circles
https://leetcode-cn.com/problems/number-of-islands/
https://leetcode-cn.com/problems/surrounded-regions/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值