I am trying to write a program that removes a specific element from a string, but most of the things I use (like filter) only work for [Char]. I really just don't want to have to type "['h','e','l','l','o']" instead of "hello". I realize that technically a String is just a fancy [Char], but how would I unfancify it into a standard [Char]. Also if you have another way to write normal words instead of in an array format please tell me.
CodePudding user response:
As was already said, String is simply a synonym for [Char]
type String = [Char]
so both can be used interchangeably.
In particular, "hello" :: [Char] is exactly the same as "hello" :: String, both are just more elegant ways of writing ['h','e','l','l','o'].
That said, you'll find that not everything that would be a “String” in other languages is a String in Haskell. See, the list implementation is actually really inefficient in particular memory-wise – for an ASCII string, most languages take either 8 or 16 bit per character, but with Haskell's String type each character is a 64-bit Char plus a reference to the next character, for a total 128 bits!
That's why most modern Haskell libraries avoid String, except for short things like file names. (Incidentally,
type FilePath = String
so that is also interchangeable.)
What these libraries use for general string is typically Text, which is indeed a different type, corresponding more to other languages' implementations (it uses UTF-16 under the hood).
If you want to filter a value of that type, you can either convert it to a listy-String with unpack, or you can simply use the dedicated version of filter provided by the text library.
In standard Haskell, Text values can not be defined as string- or list literals, you'd need to explicitly wrap that like pack ['h','e','l','l','o']. However they can still be defined with a simple string literal, provided that you turn on {-# LANGUAGE OverloadedStrings #-}:
ghci> :m Data.Text
ghci> "hello" :: Text
<interactive>:5:1: error:
• Couldn't match expected type ‘Text’ with actual type ‘[Char]’
• In the expression: "hello" :: Text
In an equation for ‘it’: it = "hello" :: Text
ghci> :set -XOverloadedStrings
ghci> "hello" :: Text
"hello"
With another extension, this also works for the list syntax:
ghci> ['h','e'] :: Text
<interactive>:9:1: error:
• Couldn't match expected type ‘Text’ with actual type ‘[Char]’
• In the expression: ['h', 'e'] :: Text
In an equation for ‘it’: it = ['h', 'e'] :: Text
ghci> :set -XOverloadedLists
ghci> ['h','e'] :: Text
"he"
CodePudding user response:
In Haskell, the square brackets mean a list, as they also do in Python. Haskell also uses white space syntax.
You can tell what type a String is in Haskell by using :t in the ghci REPL.
:t "String" -- "String" :: [Char]
So a string, in double quotes, really is a list of characters.
How about a list of strings?
:t ["airplane","boat","car"] -- ["airplane","boat","car"] :: [[Char]]
So a list of strings is a list of lists of characters.
As for filtering, if I apply a filter to a string, it behaves exactly as a filter on a list of characters:
:m Data.Char
filter isUpper "String" -- returns "S"
