Jak określić wzorzec wyrażenia regularnego w Haskell?

Nov 28 2020

Próbuję zastąpić wyrażenie regularne następującym kodem

import Text.RE.Replace
import Text.RE.TDFA.String

onlyLetters :: String -> String
onlyLetters s = replaceAll "" $ s *=~ [re|$([^a-zA-Z])|]

Naprawdę ciężko mi było znaleźć jakąkolwiek zrozumiałą dokumentację na ten temat. Powoduje to błąd kompilacji:

    src\Pangram.hs:6:53: error: parse error on input `]'
  |
6 | onlyLetters s = replaceAll "" $ (s *=~ [re|[a-zA-Z]|]) | ^ Progress 1/2 -- While building package pangram-2.0.0.12 (scroll up to its section to see the error) using: C:\sr\setup-exe-cache\x86_64-windows\Cabal-simple_Z6RU0evB_3.0.1.0_ghc-8.8.4.exe --builddir=.stack-work\dist\29cc6475 build lib:pangram test:test --ghc-options " -fdiagnostics-color=always" Process exited with code: ExitFailure 1 PS C:\Users\mcleg\Exercism\haskell\pangram> stack test pangram> configure (lib + test) Configuring pangram-2.0.0.12... pangram> build (lib + test) Preprocessing library for pangram-2.0.0.12.. Building library for pangram-2.0.0.12.. [1 of 2] Compiling Pangram src\Pangram.hs:7:56: error: parse error on input `]' | 7 | onlyLetters s = replaceAll "" $ s *=~ [re|$([^a-zA-Z])|]
  |                                                        ^

Progress 1/2

--  While building package pangram-2.0.0.12 (scroll up to its section to see the error) using:
      C:\sr\setup-exe-cache\x86_64-windows\Cabal-simple_Z6RU0evB_3.0.1.0_ghc-8.8.4.exe --builddir=.stack-work\dist\29cc6475 build lib:pangram test:test --ghc-options " -fdiagnostics-color=always"
    Process exited with code: ExitFailure 1

Jaki jest problem z tym wspornikiem i jak mam to zrobić poprawnie? Dziękuję -Skye

Odpowiedzi

3 WillemVanOnsem Nov 28 2020 at 18:56

Jest […|…|]to składnia quasi cudzysłowu [haskell-wiki] . Jest to rozszerzenie składni Haskella i nie jest domyślnie włączone.

Możesz to włączyć za pomocą LANGUAGEpragmy:

{-# LANGUAGE QuasiQuotes #-}

import Text.RE.Replace
import Text.RE.TDFA.String

onlyLetters :: String -> String
onlyLetters s = replaceAll "" $ s *=~ [re|$([^a-zA-Z])|]

Quasi-cudzysłowy wygenerują kod Haskella, który jest następnie używany w programie Haskell. Oznacza to, że za pomocą quasi-cudzysłowów walidację wyrażenia regularnego można przeprowadzić w czasie kompilacji, a nawet nieznacznie zoptymalizować wydajność w porównaniu z kompilacją wyrażenia regularnego w czasie wykonywania.

Dla danej onlyLettersfunkcji otrzymujemy:

*Main> onlyLetters "fo0b4r"
"fobr"
2 JamesBrock Nov 29 2020 at 21:02

Odpowiedź Willema Van Onsema jest lepszą odpowiedzią na to pytanie, ale ja zasugeruję odpowiedź „spróbuj tego zamiast tego” .

W ten sposób możesz dokonać zamiany tekstu w zwykłym Haskellu bez komplikacji związanych z quasi-cytatami wyrażeń regularnych.

Z https://hackage.haskell.org/package/replace-megaparsec/docs/Replace-Megaparsec.html#v:streamEdit

{-# LANGUAGE TypeFamilies #-}

import Text.Megaparsec
import Text.Megaparsec.Char
import Replace.Megaparsec
import Data.Void

-- | Invert a single-token parser “character class”.
-- | For example, match any single token except a letter or whitespace: `anySingleExcept (letterChar <|> spaceChar)`
anySingleExcept :: (MonadParsec e s m, Token s ~ Char) => m (Token s) -> m (Token s)
anySingleExcept p = notFollowedBy p *> anySingle

-- | A parser monad pattern which matches anything except letters.
nonLetters :: Parsec Void String String
nonLetters = many (anySingleExcept letterChar) 

onlyLetters :: String -> String
onlyLetters = streamEdit nonLetters (const "")

onlyLetters "fo0b4r"
"fobr"