How do I manipulate parse trees?

前端 未结 3 1579
再見小時候
再見小時候 2021-01-31 11:49

I\'ve been playing around with natural language parse trees and manipulating them in various ways. I\'ve been using Stanford\'s Tregex and Tsurgeon tools but the code is a mess

相关标签:
3条回答
  • 2021-01-31 11:56

    This is a typical case of using Lisp. You would need a function that maps another function over the tree.

    Here is a procedural matching example using Common Lisp. There are matchers in Lisp that work over list structures, which could be used instead. Using a list matcher would simplify the example (see my other answer for an example using a pattern matcher).

    The code:

    (defun node-children (node)
      (rest node))
    
    (defun node-name (node)
      (second node))
    
    (defun node-type (node)
      (first node))
    
    
    (defun treemap (tree matcher transformer)
      (cond ((null tree) nil)
            ((consp tree)
             (if (funcall matcher tree)
                 (funcall transformer tree)
               (cons (node-type tree)
                     (mapcar (lambda (child)
                               (treemap child matcher transformer))
                             (node-children tree)))))
            (t tree))))
    

    The example:

    (defvar *tree*
      '(ROOT
        (S
         (NP
          (NP (NNP Bank))
          (PP (IN of)
              (NP (NNP America))))
         (VP (VBD used)
             (S
              (VP (TO to)
                  (VP (VB be)
                      (VP (VBN called)
                          (NP
                           (NP (NNP Bank))
                           (PP (IN of)
                               (NP (NNP Italy))))))))))))
    
    
    
    (defun example ()
      (pprint
       (treemap *tree*
                (lambda (node)
                  (and (= (length (node-children node)) 2)
                       (eq (node-type (first (node-children node))) 'np)
                       (some (lambda (node)
                               (eq (node-name node) 'bank))
                             (children (first (node-children node))))
                       (eq (first (second (node-children node))) 'pp)))
                (lambda (node)
                  (list (node-type node)
                        (append (first (node-children node))
                                (node-children (second (node-children node)))))))))
    

    Running the example:

    CL-USER 75 > (example)
    
    (ROOT
     (S
      (NP
       (NP (NNP BANK) (IN OF) (NP (NNP AMERICA))))
      (VP
       (VBD USED)
       (S
        (VP
         (TO TO)
         (VP
          (VB BE)
          (VP
           (VBN CALLED)
           (NP
            (NP
             (NNP BANK)
             (IN OF)
             (NP (NNP ITALY)))))))))))
    
    0 讨论(0)
  • 2021-01-31 12:18

    Beauty is in the eye of the beholder. But you never say how the code of Tregex or Tsurgeon is a mess. It sounds more like you can't deal with Java or greater abstraction and so you're looking for something concrete written in Python.

    There's nothing wrong with hand-writing tree matching and transformation functions. Indeed, we used to do that all the time. But after the first couple of hundred, it seemed like there had to be a better way, and hence we moved to using the domain-specific languages of Tregex and Tsurgeon. This is generally seen as a laudable programming style. See in Wikipedia. They're well-specified languages with an exact syntax specification, etc. Here is your example using them.

    Tree t = Tree.valueOf("(ROOT (S (NP (NP (NNP Bank)) (PP (IN of) (NP (NNP America)))) (VP (VBD used) (S (VP (TO to) (VP (VB be) (VP (VBN called) (NP (NP (NNP Bank)) (PP (IN of) (NP (NNP Italy)))))))))))");
    TregexPattern pat = TregexPattern.compile("NP <1 (NP << Bank) <2 PP=remove");
    TsurgeonPattern surgery = Tsurgeon.parseOperation("excise remove remove");
    Tsurgeon.processPattern(pat, surgery, t).pennPrint();
    

    Note that the Java code is actually shorter than the Lisp code, precisely because of use of the domain-specific language. It's hard to see how this could be simpler: specify pattern, specify operation, apply.

    But if you'd prefer to be hand-writing methods that match patterns on trees and change them into other trees in Python, then you're most welcome to go off and do that.

    0 讨论(0)
  • 2021-01-31 12:21

    Here is a second version in Common Lisp. This time I'm using a pattern matcher.

    I'm using a function that matches a pattern against Lisp data. PMATCH:MATCH is an enhanced version of a pattern matcher found in the book Winston/Horn, Lisp, 3rd Edition. There are similar pattern matching functions available.

    The data is as in my other answer.

    The tree mapping function is changed to use the pattern matcher. The PMATCH:MATCH function returns T or an assoc list of bindings if the match is successful. It returns NIL if the match is not successful. The PMATCH:INSTANTIATE-PATTERN takes a pattern and a set of bindings. It returns a new list structure, where the pattern variables are replaced with the bindings.

    (defun treemapp (tree pattern transformer)
      (cond ((null tree) nil)
            ((consp tree)
             (let ((bindings (pmatch:match pattern tree)))
               (if bindings
                   (pmatch:instantiate-pattern transformer bindings)
                 (cons (node-type tree)
                       (mapcar (lambda (child)
                                 (treemapp child pattern transformer))
                               (node-children tree))))))
            (t tree)))
    

    The example uses now patterns.

    The pattern is a list structure. #?symbol matches a single item and creates a binding for symbol. #$symbol matches a list of items and creates a binding for symbol.

    The transformer is a pattern that will be instantiated with the bindings.

    (defun example1 ()
      (pprint (treemapp *tree*
                        '(NP (NP (#?type bank)) (PP #$children))
                        '(NP (NP (#?type bank) #$children)))))
    

    Running this code returns the same result as in my other answer.

    0 讨论(0)
提交回复
热议问题