2012-06-23 155 views
0

尝试从下面的DoxyDocs1.pm文件的详细 - > doc - > params - > parameters - > name中提取数据;还包括提取一些数据的脚本示例,但无法提取参数 - >参数 - >名称。取消评论部分的评论以查看数据。从perl中的多个哈希和数组数据结构中提取数据

#!/bin/perl 
use Data::Dumper; 
use warnings; 
require "DoxyDocs1.pm"; 
print "API Content Analyzer\n"; 
&GenerateClassInfo($doxydocs->{classes}); 
sub GenerateClassInfo() 
{ 
    my ($classes) = @_; 
    foreach my $class (@$classes) { 
    print "\nClass name is: $class->{name}\n"; 
    foreach my $pubmeth (@{$class->{public_methods}->{members}}) { 
     print "\n" if exists $pubmeth->{name}; 
     print "\tpublic methods include: $pubmeth->{name}\n"; 
     my ($key, $pmcontent) = each (@{$pubmeth->{detailed}->{doc}}); 
     print "\t\tdescription: $pmcontent->{content}\n" if exists $pmcontent->{content}; 
     # foreach my $pmp (@{$pubmeth->{detailed}->{doc}}) { 
     # print "\t\t"; 
     # print Dumper($pmp); 
     # } 
     print "\t\tkind: $pubmeth->{kind}\n"; 
     print "\t\ttype: $pubmeth->{type}\n" if exists $pubmeth->{type}; 
    } 
    foreach my $privmeth (@{$class->{private_methods}->{members}}) { 
     print "\n" if exists $privmeth->{name}; 
     print "\tprivate methods include: $privmeth->{name}\n"; 
     my ($key, $pmcontent) = each (@{$privmeth->{detailed}->{doc}}); 
     print "\t\tdescription: $pmcontent->{content}\n" if exists $pmcontent->{content}; 
     # foreach my $info (@{$privmeth->{detailed}->{doc}}) { 
     # print "\t\t"; 
     # print Dumper($info); 
     # } 
     print "\t\tkind: $privmeth->{kind}\n"; 
     print "\t\ttype: $privmeth->{type}\n" if exists $privmeth->{type}; 
    }  
    } 
} 

例DoxyDocs1.pm文件

$doxydocs= 
{ 
    classes => [ 
    { 
    name => 'Panoply::Composite', 
    public_methods => { 
    members => [ 
     { 
     kind => 'function', 
     name => 'addChild', 
     virtualness => 'non_virtual', 
     protection => 'public', 
     static => 'no', 
     brief => {}, 
     detailed => { 
      doc => [ 
      { 
       type => 'text', 
       content => 'Add a child to the container ' 
      }, 
      params => [ 
       { 
       parameters => [ 
        { 
        name => 'child' 
        } 
       ], 
       doc => [ 
        { 
        type => 'text', 
        content => 'is the child element to add' 
        } 
       ] 
       } 
      ] 
      ] 
     }, 
     type => 'void', 
     const => 'no', 
     volatile => 'no', 
     parameters => [ 
      { 
      declaration_name => 'child', 
      type => 'Ptr' 
      } 
     ] 
     }, 
     { 
     kind => 'function', 
     name => 'operator<', 
     virtualness => 'non_virtual', 
     protection => 'public', 
     static => 'no', 
     brief => {}, 
     detailed => { 
      doc => [ 
      { 
       type => 'text', 
       content => 'Less than operator' 
      }, 
      { 
       type => 'parbreak' 
      }, 
      params => [ 
       { 
       parameters => [ 
        { 
        name => 'rval' 
        } 
       ], 
       doc => [ 
        { 
        type => 'text', 
        content => 'The ' 
        }, 
        { 
        type => 'url', 
        link => 'classPanoply_1_1Package', 
        content => 'Package' 
        }, 
        { 
        type => 'text', 
        content => ' against which we are comparing this one. ' 
        } 
       ] 
       } 
      ], 
      { 
       return => [ 
       { 
        type => 'text', 
        content => 'true if this.packageID < rval.packageID, false otherwise.' 
       } 
       ] 
      } 
      ] 
     }, 
     type => 'bool', 
     const => 'yes', 
     volatile => 'no', 
     parameters => [ 
      { 
      declaration_name => 'rval', 
      type => 'const Composite &' 
      } 
     ] 
     }, 
    ] 
    }, 
    private_methods => { 
    members => [ 
     { 
     kind => 'function', 
     name => 'addChild', 
     virtualness => 'virtual', 
     protection => 'private', 
     static => 'no', 
     brief => {}, 
     detailed => { 
      doc => [ 
      { 
       type => 'text', 
       content => 'Add a child to the container ' 
      }, 
      params => [ 
       { 
       parameters => [ 
        { 
        name => 'child' 
        } 
       ], 
       doc => [ 
        { 
        type => 'text', 
        content => 'is the child element to add ' 
        } 
       ] 
       }, 
       { 
       parameters => [ 
        { 
        name => 'parent' 
        } 
       ], 
       doc => [ 
        { 
        type => 'parbreak' 
        }, 
        { 
        type => 'text', 
        content => 'is this own parent, except in weak pointer format to avoid a memory leak' 
        } 
       ] 
       } 
      ] 
      ] 
     }, 
     type => 'virtual void', 
     const => 'no', 
     volatile => 'no', 
     parameters => [ 
      { 
      declaration_name => 'child', 
      type => 'Ptr' 
      }, 
      { 
      declaration_name => 'parent', 
      type => 'Ptr' 
      } 
     ] 
     }, 
    ] 
    }, 
    } 
    ] 
}; 
1; 

回答

1

你说你想

detailed -> doc -> params -> parameters -> name 

但缺少很多索引。你想要哪个docparamparameters

detailed -> doc -> ??? -> params -> ??? -> parameters -> ??? -> name 

的语法是:

$member->{detailed}->{doc}->[$i]->{params}->[$j]->{parameters}->[$k]->{name} 

或简称:

$member->{detailed}{doc}[$i]{params}[$j]{parameters}[$k]{name} 

如果你想遍历每个文档,则params,参数,你可以使用:

my $docs = $member->{detailed}{doc}; 
for my $doc (@$docs) { 
    my $params = $doc->{params}; 
    for my $param (@$params) { 
     my $parameters = $param->{parameters}; 
     for my $parameter (@$parameters) { 
     ... 
     } 
    } 
} 

(为什么doc是单数,params是d参数是多个?为什么参数有参数???)

+0

感谢您给我更多的洞察这个结构,虽然{params}不是一个哈希引用,并且似乎导致获取参数名称“child”的问题。 DoxyDocs文件是从Doxygen自动生成的,我无法控制其结构,可能是该应用程序中的一个错误。 –

+0

这是对一个数组的引用,我对它进行了处理。 – ikegami

+0

糟糕,我的意思是说,Data Dumper报告以doc开头的结构,doc是一个散列键,引用一个匿名数组作为其值。该数组包含3个元素:对匿名散列的引用;字符串“params”;以及对匿名数组的引用。我可以通过以下方法访问参数child:$ pubmeth - > {detailed} - > {doc} - > [2] - > [0] - > {parameters} - > [0] - > {name};可能有一个我不知道的更好的方法。 –