Module: SimplyUseful::Bsearch

Defined in:
lib/simply_useful/bsearch.rb

Instance Method Summary collapse

Instance Method Details

#bfind(range = 0 ... self.length, &block) ⇒ Object



128
129
130
131
132
# File 'lib/simply_useful/bsearch.rb', line 128

def bfind(range = 0 ... self.length, &block)
  pos = self.bsearch_first(range, &block)
  return nil if pos.nil?
  self[pos]
end

#bsearch_first(range = 0 ... self.length, &block) ⇒ Object

This method searches the FIRST occurrence which satisfies a condition given by a block in binary fashion and return the index of the first occurrence. Return nil if not found.



73
74
75
76
77
78
79
80
# File 'lib/simply_useful/bsearch.rb', line 73

def bsearch_first (range = 0 ... self.length, &block)
  boundary = bsearch_lower_boundary(range, &block)
  if boundary >= self.length || yield(self[boundary]) != 0
    return nil
  else
    return boundary
  end
end

#bsearch_last(range = 0 ... self.length, &block) ⇒ Object

This method searches the LAST occurrence which satisfies a condition given by a block in binary fashion and return the index of the last occurrence. Return nil if not found.



108
109
110
111
112
113
114
115
116
117
# File 'lib/simply_useful/bsearch.rb', line 108

def bsearch_last (range = 0 ... self.length, &block)
  # `- 1' for canceling `lower + 1' in bsearch_upper_boundary.
  boundary = bsearch_upper_boundary(range, &block) - 1

  if (boundary <= -1 || yield(self[boundary]) != 0)
    return nil
  else
    return boundary
  end
end

#bsearch_lower_boundary(range = 0 ... self.length, &block) ⇒ Object

Return the lower boundary. (inside)



50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
# File 'lib/simply_useful/bsearch.rb', line 50

def bsearch_lower_boundary (range = 0 ... self.length, &block)
  lower = range.first() -1
  upper = if range.exclude_end? then
            range.last
          else
            range.last + 1
          end
  while lower + 1 != upper
    mid = ((lower + upper) / 2).to_i # for working with mathn.rb (Rational)
    if yield(self[mid]) < 0
      lower = mid
    else
      upper = mid
    end
  end
  return upper
end

#bsearch_range(range = 0 ... self.length, &block) ⇒ Object

Return the search result as a Range object.



122
123
124
125
126
# File 'lib/simply_useful/bsearch.rb', line 122

def bsearch_range (range = 0 ... self.length, &block)
  lower = bsearch_lower_boundary(range, &block)
  upper = bsearch_upper_boundary(range, &block)
  return lower ... upper
end

#bsearch_upper_boundary(range = 0 ... self.length, &block) ⇒ Object

Return the upper boundary. (outside)



85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
# File 'lib/simply_useful/bsearch.rb', line 85

def bsearch_upper_boundary (range = 0 ... self.length, &block)
  lower = range.first() -1
  upper = if range.exclude_end? then
            range.last
          else
            range.last + 1
          end
  while lower + 1 != upper
    mid = ((lower + upper) / 2).to_i # for working with mathn.rb (Rational)
    if yield(self[mid]) <= 0
      lower = mid
    else
      upper = mid
    end
  end
  return lower + 1 # outside of the matching range.
end