[Haskell-beginners] How to write Read instance
Baa
aquagnu at gmail.com
Thu Nov 2 16:45:37 UTC 2017
David, your experience and intuition were absolutely right: solution is
in the right Show and Read pair. So, this seems to work in all cases:
import qualified Text.ParserCombinators.ReadP as P
import Text.ParserCombinators.ReadPrec (lift)
import qualified Data.Char as C
-- |Type of Git hashes
newtype GitHash = GitHash { ghValue :: T.Text } deriving (Eq, Ord, Generic)
instance Show GitHash where
showsPrec p a = showParen (p > 10) (showString $ T.unpack $ ghValue a)
hexDigits :: P.ReadP String
hexDigits = P.munch1 C.isHexDigit
instance Read GitHash where
readPrec = parens $ do
lift P.skipSpaces
s::String <- lift hexDigits
return $ GitHash $ T.pack s
Thanks a lot again!!
> My own toy example works. I don't know how yours differs.
>
> On Thu, Nov 2, 2017 at 10:51 AM, Baa <aquagnu at gmail.com> wrote:
>
> > Hello, David!
> >
> > `Show` instance is simple:
> >
> > instance Show Hex where
> > show = T.unpack . ghValue
> >
> > so
> >
> > > show (Hex "1234ab")
> > 1234ab
> > > read "1234ab"::Hex
> > 1234ab
> > > read "Just 1234ab"::Maybe Hex -- fails like wrapped in the
> > > record!!
> >
> > Yes, I'm ignoring precedence usually too. And I return rest of the
> > string. Playing with `UTCTime`'s readsPrec shows me that behaviour
> > looks the same: UTCTime's reader returns the same rest. So, may be
> > trick is in the:
> >
> > 1. Precedence ?!
> > 2. readListPrec = readListPrecDefault
> > readList = readListDefault ?
> >
> > But I get error: readListPrec is not visible method of class
> > Read... 3. readPrec ? I implemented readsPrec which is old-style,
> > but am I right that old-style and new-style are absolutely the same
> > from reading point of view and old-style can replace new-style and
> > vice versa?
> >
> > PS. I don't import any special modules.
> >
> >
> > > The most common way is to just auto derive Read. I'm not sure
> > > that that ever really fails. Are you sure the problem isn't with
> > > the Show instance of the type? People commonly write invalid
> > > Show instances to make them look pretty and they shouldn't. read
> > > and show are supposed to be inverses of each other because when
> > > they aren't, problems like this occur.
> > >
> > > The simple way to do a Read instance is to implement the reads
> > > function for it. The confusion comes from its type. readsPrec ::
> > > Int -> ReadS a. ReadS is defined as String -> [(a, String)],
> > > where a is the parsed result and String is the rest of the string
> > > that is being parsed, , which may look confusing, and the Int is
> > > precedence, which can usually be ignored. It could have been Int
> > > -> String -> Maybe (a, String), but Read predates Maybe. So
> > > instead it returns a list and if it fails to parse, it returns []
> > > instead of Nothing. So.
> > >
> > > data MyFoo = MyFoo
> > >
> > > instance Read MyFoo where
> > > -- readsPrec :: Int -> String -> [(MyFoo, String)]
> > > readsPrec _ = readFoo
> > >
> > > readFoo :: String -> [(MyFoo, String)]
> > > readFoo str = case splitAt 5 str of
> > > ("MyFoo", rest) -> [(MyFoo, rest)]
> > > otherwise -> []
> > >
> > > If you need something more complex, there are functions to do it
> > > in base that perform lexing and parsing. I have never used them
> > > but you can go ahead and read some of the instances such as
> > > Ordering at
> > > https://hackage.haskell.org/package/base-4.10.0.0/docs/
> > src/GHC.Read.html#line-398
> > > to try and learn how it might work for you.
> > >
> > > But honestly I think you should look at fixing your Show instance
> > > first, if possible.
> > >
> > >
> > > On Thu, Nov 2, 2017 at 9:41 AM, Baa <aquagnu at gmail.com> wrote:
> > >
> > > > Hello all!
> > > >
> > > > I found some errors in the reading of previously shown big and
> > > > complex record. Reason is: some of fields are reading wrongly.
> > > > So, is there any useful documents how to write `Read` instance
> > > > correctly (because I can't find good tutorial in the Web and
> > > > often hit errors with `Read` instance)? May be
> > > > tutorial/examples/any good info...
> > > >
> > > >
> > > > ===
> > > > Best regards, Paul
> > > > _______________________________________________
> > > > Beginners mailing list
> > > > Beginners at haskell.org
> > > > http://mail.haskell.org/cgi-bin/mailman/listinfo/beginners
> > > >
> >
> > _______________________________________________
> > Beginners mailing list
> > Beginners at haskell.org
> > http://mail.haskell.org/cgi-bin/mailman/listinfo/beginners
> >
More information about the Beginners
mailing list