不可重复的字符串比较,迫使elsif失败

时间:2009-05-30 13:51:55

标签: ruby string-comparison ruby-1.8

在回答this code golf question时,我的回答中遇到了一个问题。

我一直在测试这个,我甚至不能在代码中使用这两个比较,尽管事实上IRB有正确的行为。我真的需要一些帮助。

下面的代码是对问题的解释。

def solve_expression(expr)
  chars = expr.split '' # characters of the expression
  parts = []  # resulting parts
  s,n = '','' # current characters

  while(n = chars.shift)
    if (s + n).match(/^(-?)[.\d]+$/) || (!chars[0].nil? && chars[0] != ' ' && n == '-') # only concatenate when it is part of a valid number
      s += n
    elsif (chars[0] == '(' && n[0] == '-') || n == '(' # begin a sub-expression
      p n # to see what it breaks on, ( or -
      negate = n[0] == '-'
      open = 1
      subExpr = ''
      while(n = chars.shift)
        open += 1 if n == '('
        open -= 1 if n == ')'
        # if the number of open parenthesis equals 0, we've run to the end of the
        # expression.  Make a new expression with the new string, and add it to the
        # stack.
        subExpr += n unless n == ')' && open == 0
        break if open == 0
      end
      parts.push(negate ? -solve_expression(subExpr) : solve_expression(subExpr))
      s = ''
    elsif n.match(/[+\-\/*]/)
      parts.push(n) and s = ''
    else
      parts.push(s) if !s.empty?
      s = ''
    end
  end
  parts.push(s) unless s.empty? # expression exits 1 character too soon.

  # now for some solutions!
  i = 1
  a = parts[0].to_f # left-most value is will become the result
  while i < parts.count
    b,c = parts[i..i+1]
    c = c.to_f
    case b
      when '+': a = a + c
      when '-': a = a - c
      when '*': a = a * c
      when '/': a = a / c
    end
    i += 2
  end
  a
end

问题出现在negate

的分配中

当表达式之前的字符是破折号时,我需要否定为真,但条件甚至不起作用。引用形式n == '-'n[0] == '-'都无关紧要,每次都会出现FALSE。然而,我一直在使用这种精确的比较,n == '('每次都能正常工作!

发生了什么事? n == '-'何时,n == '('为什么不起作用?这是用UTF-8编码的,没有BOM,UNIX换行符。

我的代码出了什么问题?

1 个答案:

答案 0 :(得分:3)

你有:

if (s + n).match(/^(-?)[.\d]+$/) || (!chars[0].nil? && chars[0] != ' ' && n == '-')
      s += n
elsif (chars[0] == '(' && n[0] == '-') || n == '('

由于n始终是一个字符的字符串,如果(chars[0] == '(' && n[0] == '-'))为真,则之前的条件(!chars[0].nil? && chars[0] != ' ' && n == '-')也将为真。如果if,您的代码将永远不会进入n[0]=='-'的第二部分。

如果您的p n行正在输出短划线,请确保它与您要查找的字符完全相同,而不是某些字符看起来像短划线。 Unicode有很多种破折号,也许你的代码或输入中有一个奇怪的unicode破折号字符。