Module: Text::Levenshtein

Extended by:
Levenshtein
Included in:
Levenshtein
Defined in:
lib/text/levenshtein.rb

Overview

:nodoc:

Instance Method Summary collapse

Instance Method Details

#distance(str1, str2) ⇒ Object

Calculate the Levenshtein distance between two strings str1 and str2.

In Ruby 1.8, str1 and str2 should be ASCII, UTF-8, or a one-byte-per character encoding such as ISO-8859-*. They will be treated as UTF-8 if $KCODE is set appropriately (i.e. ā€˜uā€™). Otherwise, the comparison will be performed byte-by-byte. There is no specific support for Shift-JIS or EUC strings.

In Ruby 1.9+, the strings will be processed as UTF-8.

When using Unicode text, be aware that this algorithm does not perform normalisation. If there is a possibility of different normalised forms being used, normalisation should be performed beforehand.



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
# File 'lib/text/levenshtein.rb', line 32

def distance(str1, str2)
  prepare =
    if "ruby".respond_to?(:encoding)
      lambda { |str| str.encode(Encoding::UTF_8).unpack("U*") }
    else
      rule = $KCODE.match(/^U/i) ? "U*" : "C*"
      lambda { |str| str.unpack(rule) }
    end

  s, t = [str1, str2].map(&prepare)
  n = s.length
  m = t.length
  return m if n.zero?
  return n if m.zero?

  d = (0..m).to_a
  x = nil

  n.times do |i|
    e = i + 1
    m.times do |j|
      cost = (s[i] == t[j]) ? 0 : 1
      x = [
        d[j+1] + 1, # insertion
        e + 1,      # deletion
        d[j] + cost # substitution
      ].min
      d[j] = e
      e = x
    end
    d[m] = x
  end

  return x
end