Parsing 如何解析函数参数块?

Parsing 如何解析函数参数块?,parsing,rebol,Parsing,Rebol,我想在给定的Rebol函数中重新组织arguments块,以便更容易理解函数所需的参数。Rebol函数中的arguments块是Rebol中可延展数据结构的一个很好的例子: adjoin: func [ "Adjoins" series [series!] "Series to adjoin" joinee /local other ][...] 但是我需要一些更可预测的东西来理解这个元数据。如何将其转换为更符合要求的格式?例如: [ ; should

我想在给定的Rebol函数中重新组织arguments块,以便更容易理解函数所需的参数。Rebol函数中的arguments块是Rebol中可延展数据结构的一个很好的例子:

adjoin: func [
    "Adjoins"
    series [series!] "Series to adjoin"
    joinee
    /local other
][...]
但是我需要一些更可预测的东西来理解这个元数据。如何将其转换为更符合要求的格式?例如:

[
    ; should include about value whether the about string is there or not
    about none none "Adjoins"
    argument series [series!] "Series to Adjoin"
    argument joinee none none
    option local none none
    argument other none none
]

任何关于转换方法或表示参数内容的最佳方式的想法都会非常有用。

这里是一个完整的rebol脚本,它应该可以帮助您:-)

请注意,变量不必以点开头,解析规则也不需要用=符号包围。这是一种在规则范围内将每件事情的任务分开的快速方法。这样,识别哪个词做什么就容易多了,这在您开始构建更大的规则时尤为重要

rebol [
    title: "func spec extractor"
]



code: {adjoin: func [
    "Adjoins"
    series [series!] "Series to adjoin"
    joinee
    /local other
][...]

append: func [
    {Appends a value to the tail of a series and returns the series head.}
    series [series! port!]
    value
    /only "Appends a block value as a block"
][ ... ]
}



code: load code

;----
; setting to a temp variable, prevents .param-str from being erased 
; if the rule doesn't match at the point the rule is used (it may be optional)
;----
=param-str=: [set .tmp string! (.param-str: .tmp)] 
=param-types=: [set .tmp into [some [word!]] (.param-types: .tmp)] 

=param=: [
    (.param-types: .tmp: .param-str: none )
    set .param-name word!
    opt =param-str= 
    opt =param-types= 
    opt =param-str=
    ( 
        append/only .param-blk .tmp: reduce [ .param-name .param-str .param-types ]
    )
]

=refinements=: [
    (.ref-str: none)
    set .refinement refinement! 
    opt [ set .ref-str string! ]
    (
        append .param-blk .refinement
        append .param-blk .ref-str
    )    
    any =param=
]

=func-rule=: [
    ; set/reset variables
    (
        func-def: context [name: none doc-str: none args: [] refinements: [] code: none]
        .tmp: .func-name: .doc-str: .param-str: none
    )

    set .func-name set-word!  
    'func into [
        opt [ set .doc-str string! ]
        ( func-def/args:  .param-blk: copy [] )
        any =param=
        ( func-def/refinements:  .param-blk: copy [] )
        any =refinements=
        here:        
    ]
    set .func-body block!
    (
        func-def/name:    .func-name
        func-def/doc-str: .doc-str
        func-def/code:    .func-body
    )
]      

funcs: []
parse code [ some [ =func-rule=  ( append funcs func-def) | skip ]]

probe funcs
下面是它打印出来的内容:

[make object! [
        name: adjoin:
        doc-str: "Adjoins"
        args: [[
                series "Series to adjoin" [series!]
            ] [
                joinee none none
            ]]
        refinements: [
            /local none [other none none]
        ]
        code: [...]
    ] make object! [
        name: append:
        doc-str: {Appends a value to the tail of a series and returns the series head.}
        args: [[
                series none [series! port!]
            ] [
                value none none
            ]]
        refinements: [
            /only "Appends a block value as a block"
        ]
        code: [...]
    ]]