I have implemented a sorting algorithm for a custom string that represents either time or distance data for track & field events. Below is the format
’10:03.00 – Either ten minutes and three seconds or 10 feet, three inches
The result of the sort is that for field events, the longest throw or jump would be the first element while for running events, the fastest time would be first. Below is the code I am currently using for field events. I didn’t post the running_event_sort since it is the same logic with the greater than/less than swapped. While it works, it just seems overly complex and needs to be refactored. I am open to suggestions. Any help would be great.
event_participants.sort!{ |a, b| Participant.field_event_sort(a, b) }
class Participant
def self.field_event_sort(a, b)
a_parts = a.time_distance.scan(/'([\d]*):([\d]*).([\d]*)/)
b_parts = b.time_distance.scan(/'([\d]*):([\d]*).([\d]*)/)
if(a_parts.empty? || b_parts.empty?)
0
elsif a_parts[0][0] == b_parts[0][0]
if a_parts[0][1] == b_parts[0][1]
if a_parts[0][2] > b_parts[0][2]
-1
elsif a_parts[0][2] < b_parts[0][2]
1
else
0
end
elsif a_parts[0][1] > b_parts[0][1]
-1
else
1
end
elsif a_parts[0][0] > b_parts[0][0]
-1
else
1
end
end
end
This is a situation where
#sort_bycould simplify your code enormously:Here, I parse the relevant times into an array of integers, and use those as a sorting key for the data. Array comparisons are done entry by entry, with the first being the most significant, so this works well.
One thing you don’t do is convert the digits to integers, which you most likely want to do. Otherwise, you’ll have issues with
"100" < "2" #=> true. This is why I added the#mapstep.Also, in your regex, the square brackets around
\dare unnecessary, though you do want to escape the period so it doesn’t match all characters.One way the code I gave doesn’t match the code you gave is in the situation where a line doesn’t contain any distances. Your code will compare them as equal to surrounding lines (which may get you into trouble if the sorting algorithm assumes equality is transitive. That is
a == b,b == cimpliesa ==c, which is not the case for your code : for examplea = "'10:00.1",b = "frog",c="'9:99:9").#sort_bysorts in ascending order, so the call to#reversewill change it into descending order.#sort_byalso has the advantage of only parsing out the comparison values once, whereas your algorithm will have to parse each line for every comparison.