Функция Haskell, возвращающая пустой список - PullRequest
2 голосов
/ 20 марта 2011

Я на самом деле абсолютный новичок в Haskell, так что я в полной растерянности относительно того, как отлаживать некоторые написанные мной функции. Когда я звоню shuntingYard ["3+4"], я возвращаюсь [], тогда как я хочу вернуть [34+]. Любая помощь будет очень, очень признательна.

import Char

isOperator :: Char -> Bool
isOperator x = elem x ['+','-','*','/','%','^','!','=','<','>']

associativityOf :: Char -> String
associativityOf x = if elem x ['+','-','*','/','%']
                    then "Left"
                    else "Right"

precedenceOf :: Char -> Int
precedenceOf x
    | elem x "=<>"   = 1 
    | elem x "+-"    = 2
    | elem x "*/%"   = 3
    | elem x "^!"    = 4
    | otherwise      = 0

operatorActions :: [[Char]] -> [[Char]] -> [[Char]]
operatorActions stmt stack
    | ( tokenAssoc == "Left"  && tokenPrecedence <= stackPrecedence ) ||        
      ( tokenAssoc == "Right" && tokenPrecedence <  stackPrecedence ) =
        [stackOper] : _shuntingYard stmt (tail stack)
    | otherwise   = _shuntingYard (tail stmt) ((head stmt) : stack)
    where tokenAssoc       = associativityOf (head (head stmt))
          tokenPrecedence  = precedenceOf    (head (head stmt))
          stackOper        = if (not (null stack))
                           then (head (head stack))
                           else '='
          stackPrecedence  = precedenceOf stackOper

stackOperations :: [[Char]] -> [[Char]]                                
stackOperations stack
    | ((not (null stack)) && (head (head stack)) == '(') = 
      error "Unbalanced parens."
    | null stack = []
    | otherwise  = (head stack) : _shuntingYard [] (tail stack)

_shuntingYard :: [[Char]] -> [[Char]] -> [[Char]]
_shuntingYard stmt stack
    | null stmt          = stackOperations stack
    | all isDigit (head stmt) = (head stmt) : _shuntingYard (tail stmt) stack
    | isOperator  (head (head stmt)) = operatorActions stmt stack
    | (head (head stmt)) == '('=
      _shuntingYard (tail stmt) ((head stmt) : stack)
    | (head (head stmt)) == ')' = if (head (head stack)) == '('
                            then _shuntingYard (tail stmt) (tail stack)
                            else (head stack) : _shuntingYard stmt (tail stack)
    | otherwise = _shuntingYard (tail stmt) stack

shuntingYard :: [[Char]] -> [[Char]]
shuntingYard stmt = _shuntingYard stmt []

Ответы [ 2 ]

5 голосов
/ 20 марта 2011

В качестве общей техники отладки вы можете использовать модуль Debug.Trace, чтобы узнать, какие функции вызываются и каковы их входные данные.Используя взгляд на состояние вашего алгоритма после каждого шага.

import Debug.Trace

-- Show arguments each time _shuntingYard is called
_shuntingYard :: [[Char]] -> [[Char]] -> [[Char]]
_shuntingYard stmt stack = traceShow (stmt, stack) $ __shuntingYard stmt stack

__shuntingYard stmt stack
  | null stmt = stackOperations stack
  {- etcetera -}

Это печатает:

(["3+4"],[])
([],[])

Хм, вы потеряли все после первого вызова.Глядя на охранников в __shuntingYard, кажется, что вызывается случай «иначе».

Может быть, вы хотели позвонить shuntingYard ["3", "+", "4"]?

3 голосов
/ 20 марта 2011

Хорошо, давайте просто поиграем в то, что происходит, когда вы звоните shuntingYard ["3+4"]:

  1. Он вызывает _shuntingYard ["3+4"] []
  2. Он проходит через охранников _shuntingYard:
    1. null stmt = null ["3+4"] = false
    2. all isDigit (head stmt) = all isDigit "3+4" = false как + не является цифрой
    3. isOperator (head (head stmt)) = isOperator '3' = false
    4. Также ложно как '3' /= '('
    5. Также ложно, как '3' /= ')'
  3. Поскольку ни один из охранников не совпал, мы переходим к делу по умолчанию и вызываем _shuntingYard (tail stmt) stack = _shuntingYard [] []
  4. На этот раз первый охранник (null stmt = null []) совпадений, поэтому мы вызываем stackOperations [] и получаем [].
...