博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
愿你走出半生,归来仍是Java Parser
阅读量:6850 次
发布时间:2019-06-26

本文共 2929 字,大约阅读时间需要 9 分钟。

几天前,我的一个朋友给了我一个Haskell问题

Hey, MK,假设我有个BNF,并且我在Haskell中有个这个BNF的parser。
现在,我想给这个BNF改一行,有没有办法不用动这个BNF parser的代码(因为是其他人写的),而是对这parser进行扩展呢?

这问题挺有趣的,也不算难。

这问题说是extensibility problem,其实有两个地方需要扩展。

0:Parser需要用open recursion之类的方法扩展

1:Parse出来的ADT也需要可扩展性

后半个需求见多了,Final Tagless,DTALC,Tree that grow,Recursion scheme style fix。。。于是放下不表,我们来处理前一个。

前半个。。Haskell's Overlooked Object System就搞过,当然他们有点heavy weight,打算随手弄一个超级轻量级的:5行就够了,多一行是小莎莎。

Ready?

1 data Object x = MkObject (x -> x)

1。Inheritance is not subtyping式的Object=recursive type。为了简易性(反正也不需要多高的扩展性)就不model真。recursive type,而只有recursive dependency。

1 use :: Object x -> x2 use (MkObject x) = let res = x res in res

2。3。最典型的tying the knot。其实就是fix了。

我们想想,这个x是什么variant的呢?covariant还是contravariant?

1 inherit :: (a -> b) -> (b -> a) -> Object a -> Object b2 inherit ab ba (MkObject aa) = MkObject (ab . aa . ba)

既然是invariant,那fmap contramap都用不上,但invariant依然能有map:两边一起传进来就行了。4。5。

这就是一个prototype based oo system了。

接下来讲怎么用哈:

1 test :: Object (Int, Int)2 test = MkObject $ \self -> (2, fst self + fst self)

这弄了个两个field的object,第零个field初始值为2(可能因为继承被override),第一个field为第零个field的值*2(不一定是3,如果任何field被override这个值都能改)。use test应该是(2, 4)。

1 inheritTest :: Object ((Int, Int), Int)2 inheritTest = inherit (\(l, r) -> ((l + 1, r + 2), r + 1)) fst test

这里继承了上面的Object,override了l(l + 1是super + 1),r被override到super + 2,加了个新的field,值是r+1。use inheritTest应该是((3, 8), 7)。记着传进来的参数不是self而是super就很好理解了。

好,open recursion搞好了,剩下的就是标准的final tagless了,体力活,没啥意思

1 class AST repr where 2   lit :: Int -> repr 3   plus :: repr -> repr -> repr 4  5 class Var repr where 6   var :: String -> repr 7  8 type WholeParser repr = Parser repr 9 type LitParser repr = Parser repr10 type PlusParser repr = Parser repr11 12 intP :: Parser Int13 intP = read <$> many1 digit14 15 stringP :: Parser String16 stringP = many1 letter17 18 type OriginalParser repr = ((LitParser repr, PlusParser repr), WholeParser repr)19 originalParser :: AST repr => Object (OriginalParser repr)20 originalParser = MkObject $ \(~(_, p)) -> let21   litP = lit <$> intP22   plusP = between (char '(') (char ')') (do {l <- p; spaces; char '+'; spaces; r <- p; return $ plus l r})23   wholeP = litP <|> plusP in24   ((litP, plusP), wholeP)25 26 type VarParser repr = Parser repr27 extendedParser :: (AST repr, Var repr) => Object (VarParser repr, OriginalParser repr)28 extendedParser = inherit extend snd originalParser29   where30     extend ~((litP, plusP), wholeP) = let31       varP = var <$> stringP in32       (varP, ((litP, plusP), varP <|> wholeP))33 34 instance AST String where35   lit = show36   plus x y = "(" ++ x ++ " " ++ "+" ++ " " ++ y ++ ")"37 38 instance Var String where39   var x = x

大功告成。

代码在

 

Q:封装呢?

A:Abstract Type is Existential Type

 

Q:这是prototype based的,class怎么办?

A:A Theory Of Object里面讲过怎么用prototype来做class

 

Q:多继承呢?

A:给定Object a,Object b,可以组合出Object (a, b),要菱形继承自己手动再inherit一下就好

 

Q:Subtyping?

A:Typeclass。

 

如果大家感兴趣,请评论下,我可以再写个blog把这些功能补完。

转载地址:http://oprul.baihongyu.com/

你可能感兴趣的文章
C++ 使用STL string 实现的split,trim,replace-修订
查看>>
2011年7月10个非常棒的jQuery插件
查看>>
.NET简谈事务、分布式事务处理
查看>>
我是如何推理出王珞丹住址的zz
查看>>
C#泛型列表List<T>基本用法总结
查看>>
《UNIX环境高级编程》单个源码编译方法
查看>>
追涨必须具备的四个条件
查看>>
最大存款方式
查看>>
GridView删除时激发了未处理的事件“RowDeleting"
查看>>
ZOJ 3213 Beautiful Meadow
查看>>
什么是聚合根
查看>>
机器学习&数据挖掘笔记_17(PGM练习一:贝叶斯网络基本操作)
查看>>
图像旋转
查看>>
css两列等高布局
查看>>
PHP适合做大型网站吗?
查看>>
lua入门之二:c/c++ 调用lua及多个函数返回值的获取
查看>>
C使用FILE指针文件操作
查看>>
cobbler pxe-menu
查看>>
openssl 非对称加密 RSA 加密解密以及签名验证签名
查看>>
MyBatis Generator生成DAO——序列化
查看>>