mirror of
https://github.com/TheAlgorithms/Ruby
synced 2024-12-29 22:24:07 +01:00
62 lines
1.5 KiB
Ruby
62 lines
1.5 KiB
Ruby
require 'set'
|
|
|
|
##
|
|
# This class aims to represent unweighted graphs
|
|
# (i.e. graphs for which edges between nodes have no specific weight associated to them).
|
|
#
|
|
# Both directed (i.e. an edge between node U and node V does not imply an edge in the opposite direction)
|
|
# and undirected graphs are supported, depending on the constructor invocation.
|
|
|
|
class UnweightedGraph
|
|
attr_reader :nodes
|
|
attr_reader :directed
|
|
|
|
def initialize(nodes: [], neighbors: {}, directed: true)
|
|
@nodes = Set[]
|
|
@neighbors = {}
|
|
@directed = directed
|
|
for node in nodes
|
|
add_node(node)
|
|
end
|
|
neighbors.each do |node, neighbors|
|
|
for neighbor in neighbors
|
|
add_edge(node, neighbor)
|
|
end
|
|
end
|
|
end
|
|
|
|
def add_node(node)
|
|
if include?(node)
|
|
raise ArgumentError, "node #{node} already exists in this graph!"
|
|
end
|
|
@nodes.add(node)
|
|
@neighbors[node] = Set[]
|
|
end
|
|
|
|
def add_edge(start_node, end_node)
|
|
if has_neighbor?(start_node, end_node)
|
|
raise ArgumentError, "node #{start_node} already has an edge to #{end_node} in this graph!"
|
|
end
|
|
@neighbors[start_node].add(end_node)
|
|
@neighbors[end_node].add(start_node) unless directed
|
|
end
|
|
|
|
def neighbors(node)
|
|
unless include?(node)
|
|
raise ArgumentError, "node #{node} does not exist in this graph!"
|
|
end
|
|
@neighbors[node]
|
|
end
|
|
|
|
def empty?
|
|
nodes.empty?
|
|
end
|
|
|
|
def include?(node)
|
|
nodes.include?(node)
|
|
end
|
|
|
|
def has_neighbor?(start_node, end_node)
|
|
neighbors(start_node).include?(end_node)
|
|
end
|
|
end
|