-
Notifications
You must be signed in to change notification settings - Fork 68
/
Copy pathStringMix.rb
97 lines (79 loc) · 3.17 KB
/
StringMix.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
=begin
Given two strings s1 and s2, we want to visualize how different the two strings
are. We will only take into account the lowercase letters (a to z). First let us
count the frequency of each lowercase letters in s1 and s2.
s1 = "A aaaa bb c"
s2 = "& aaa bbb c d"
s1 has 4 'a', 2 'b', 1 'c'
s2 has 3 'a', 3 'b', 1 'c', 1 'd'
So the maximum for 'a' in s1 and s2 is 4 from s1; the maximum for 'b' is 3 from
s2. In the following we will not consider letters when the maximum of their
occurrences is less than or equal to 1.
We can resume the differences between s1 and s2 in the following string:
"1:aaaa/2:bbb" where 1 in 1:aaaa stands for string s1 and aaaa because the
maximum for a is 4. In the same manner 2:bbb stands for string s2 and bbb
because the maximum for b is 3.
The task is to produce a string in which each lowercase letters of s1 or s2
appears as many times as its maximum if this maximum is strictly greater than 1;
these letters will be prefixed by the number of the string where they appear
with their maximum value and :. If the maximum is in s1 as well as in s2 the
prefix is =:.
In the result, substrings will be in decreasing order of their length and when
they have the same length sorted alphabetically (more precisely sorted by
codepoint); the different groups will be separated by '/'.
Hopefully other examples can make this clearer.
s1 = "my&friend&Paul has heavy hats! &"
s2 = "my friend John has many many friends &"
mix(s1, s2) "2:nnnnn/1:aaaa/1:hhh/2:mmm/2:yyy/2:dd/2:ff/2:ii/2:rr/=:ee/=:ss"
s1 = "mmmmm m nnnnn y&friend&Paul has heavy hats! &"
s2 = "my frie n d Joh n has ma n y ma n y frie n ds n&"
mix(s1, s2) "1:mmmmmm/=:nnnnnn/1:aaaa/1:hhh/2:yyy/2:dd/2:ff/2:ii/2:rr/=:ee/=:ss"
s1="Are the kids at home? aaaaa fffff"
s2="Yes they are here! aaaaa fffff"
mix(s1, s2) --> "=:aaaaaa/2:eeeee/=:fffff/1:tt/2:rr/=:hh"
=end
# My Solution
def mix(s1, s2)
s1_hash = Hash.new {|v,k| v[k] = 0}
s2_hash = Hash.new {|v,k| v[k] = 0}
s1.split("").each {|x| s1_hash[x] += 1 if x =~ /[a-z]/}
s2.split("").each {|x| s2_hash[x] += 1 if x =~ /[a-z]/}
result = []
high = 0
("a".."z").each do |x|
result << [1, x, s1_hash[x]] if s1_hash[x] > s2_hash[x] && s1_hash[x] > 1
result << [2, x, s2_hash[x]] if s2_hash[x] > s1_hash[x] && s2_hash[x] > 1
result << [3, x, s1_hash[x]] if s1_hash[x] == s2_hash[x] && s1_hash[x] > 1
high = s1_hash[x] if s1_hash[x] > high
high = s2_hash[x] if s2_hash[x] > high
end
output = []
(high).downto(2) do |n|
temp = []
result.each do |x|
if x[2] == n
temp << [x[0],x[1],x[2]]
end
end
temp = temp.sort_by! {|a,b,c| a}
temp.each do |x|
x[0] == 3 ? m = "=" : m = x[0].to_s
output << "#{m}:#{x[1]*x[2]}"
end
end
output.join("/")
end
# Better Solution
def mix(s1, s2)
selection = ('a'..'z').to_a.select { |letter| s1.count(letter) > 1 || s2.count(letter) > 1 }
selection.map! do |selection|
if s1.count(selection) > s2.count(selection)
"1:#{selection * s1.count(selection)}"
elsif s1.count(selection) < s2.count(selection)
"2:#{selection * s2.count(selection)}"
else
"=:#{selection * s1.count(selection)}"
end
end
selection.sort_by { |x| [-x.size, x[0], x[-1]] }.join("/")
end