[2016][ruby][1] refactoring

profile
Alpha Chen 8 years ago
parent b77d565b97
commit ddbface9e5

@ -1,39 +1,49 @@
include Math
def follow_instructions(input)
seq = input.split(', ')
Instruction = Struct.new(:direction, :blocks) do
def initialize(direction, blocks)
super(direction, blocks.to_i)
end
end
# start facing north at the origin
dir = -PI/2
locations = [[0,0]]
seq.each.with_object(locations) do |step, locations|
/(?<turn>\w)(?<blocks>\d+)/ =~ step
Turtle = Struct.new(:orientation, :location) do
def self.run(input)
instructions = parse(input)
turtles = [Turtle.new(-PI/2, [0,0])]
case turn
when ?R
dir += PI/2
when ?L
dir -= PI/2
else
raise "unexpected turn: #{turn}"
instructions.each.with_object(turtles) do |instruction, turtles|
turtles << turtles.last.turn(instruction.direction).step
(instruction.blocks - 1).times do
turtles << turtles.last.step
end
end
blocks.to_i.times do
location = locations.last
x = location[0] + cos(dir).to_i
y = location[1] + sin(dir).to_i
locations << [x, y]
turtles
end
def self.parse(input)
input.split(', ').map {|step|
Instruction.new(*step.scan(/(?<turn>\w)(?<blocks>\d+)/)[0])
}
end
locations
end
def turn(direction)
angle = case direction
when ?R
PI/2
when ?L
-PI/2
else
raise "unexpected direction: #{direction}"
end
Turtle.new(orientation + angle, location)
end
def first_dupe(ary)
seen = Set.new
ary.find { |elem|
seen.include?(elem) || (seen << elem && false)
}
def step
x = location[0] + cos(orientation).to_i
y = location[1] + sin(orientation).to_i
Turtle.new(orientation, [x, y])
end
end
require 'minitest'
@ -45,23 +55,17 @@ class TestInstructions < Minitest::Test
assert_distance 12, 'R5, L5, R5, R3'
end
def test_first_dupe
assert_equal 1, first_dupe([1, 2, 3, 1])
assert_equal 2, first_dupe([1, 2, 3, 2])
assert_equal 3, first_dupe([1, 2, 3, 3])
locations = follow_instructions('R8, R4, R4, R8')
assert_equal [4, 0], first_dupe(locations)
end
def assert_distance(expected, instructions)
locations = follow_instructions(instructions)
assert_equal expected, locations.last.map(&:abs).inject(:+)
assert_equal expected, Turtle.run(instructions).last.location.map(&:abs).inject(:+)
end
end
if __FILE__ == $0
puts first_dupe(follow_instructions(DATA.read))
turtles = Turtle.run(DATA.read)
p turtles.last
locations = turtles.map(&:location)
p locations.find {|location| locations.count(location) > 1 }
end
__END__

Loading…
Cancel
Save