AtCoder Beginner Contest C - Connect Cities Difficulty: 363
AtCoder Beginner Contest D - Friends Difficulty: 676
Ce thème, Union Find
Cette fois, j'utilise «Union Find (DSU)» dedans. C - Connect Cities
ruby.rb
# frozen_string_literal: true
# Disjoint Set Union
class DSU
  def initialize(n = 0)
    # root node: -1 * component size
    # otherwise: parent
    @n = n
    @parent_or_size = Array.new(n, -1)
  end
  def merge(a, b)
    x = leader(a)
    y = leader(b)
    return x if x == y
    x, y = y, x if -@parent_or_size[x] < -@parent_or_size[y]
    @parent_or_size[x] += @parent_or_size[y]
    @parent_or_size[y] = x
    x
  end
  def same(a, b)
    leader(a) == leader(b)
  end
  def leader(a)
    @parent_or_size[a].negative? ? a : (@parent_or_size[a] = leader(@parent_or_size[a]))
  end
  def size(a)
    -@parent_or_size[leader(a)]
  end
  def groups
    leader_buf = Array.new(@n, 0)
    group_size = Array.new(@n, 0)
    (0...@n).each do |i|
      leader_buf[i] = leader(i)
      group_size[leader_buf[i]] += 1
    end
    result = Array.new(@n) { [] }
    (0...@n).each do |i|
      result[leader_buf[i]].push i
    end
    result.delete_if(&:empty?)
    result
  end
  alias root leader
  alias find leader
  alias unite merge
  alias same? same
end
UnionFind        = DSU
UnionFindTree    = DSU
DisjointSetUnion = DSU
n, m = gets.split.map(&:to_i)
ab = m.times.map { gets.split.map(&:to_i) }
u = UnionFind.new(n)
ab.each do |a, b|
  u.merge(a - 1, b - 1)
end
puts u.groups.size - 1
D - Friends
ruby.rb
# frozen_string_literal: true
# Disjoint Set Union
class DSU
  def initialize(n = 0)
    # root node: -1 * component size
    # otherwise: parent
    @n = n
    @parent_or_size = Array.new(n, -1)
  end
  def merge(a, b)
    x = leader(a)
    y = leader(b)
    return x if x == y
    x, y = y, x if -@parent_or_size[x] < -@parent_or_size[y]
    @parent_or_size[x] += @parent_or_size[y]
    @parent_or_size[y] = x
    x
  end
  def same(a, b)
    leader(a) == leader(b)
  end
  def leader(a)
    @parent_or_size[a].negative? ? a : (@parent_or_size[a] = leader(@parent_or_size[a]))
  end
  def size(a)
    -@parent_or_size[leader(a)]
  end
  def groups
    leader_buf = Array.new(@n, 0)
    group_size = Array.new(@n, 0)
    (0...@n).each do |i|
      leader_buf[i] = leader(i)
      group_size[leader_buf[i]] += 1
    end
    result = Array.new(@n) { [] }
    (0...@n).each do |i|
      result[leader_buf[i]].push i
    end
    result.delete_if(&:empty?)
    result
  end
  alias root leader
  alias find leader
  alias unite merge
  alias same? same
end
UnionFind        = DSU
UnionFindTree    = DSU
DisjointSetUnion = DSU
n, m = gets.split.map(&:to_i)
ab = m.times.map { gets.split.map(&:to_i) }
u = UnionFind.new(n)
ab.each do |a, b|
  u.merge(a - 1, b - 1)
end
puts u.groups.map { _1.size }.max
Par rapport à la première source, seule la dernière ligne est modifiée.
Site référencé ac-library-rb - GitHub