当前位置:编程语言->C++语言 ->osg::NodeVisitor遍历的秘密

原创版权标志osg::NodeVisitor遍历的秘密

作者:ooklasd  发表时间:2018/8/10 9:13:06  阅读:
[摘要] 扩展自己的NodeVisitor,使得可以遍历自己的类
使用支付宝扫码领红包,余额宝付款才可以使用红包哦!不要忘记哈。每天扫一次,天天赚红包!!可以将二维码保存到手机,每天直接扫码领红包啦!!

    NodeVisitor遍历的秘密

    ——扩展自己的nodeVisitor

    By 郭顺铭

    我想实现想NodeVisitor一样的遍历器,方便地遍历自己的类,从而降低访问类的难度,看到源码才知道里面的奥秘

     

    我们先看NodeVisitor的两个重要函数

    void traverse(Node& node)

    void apply()

     

    先解析NodeVisitor::traverse()

    NodeVisitor::traverse遍历某个节点,他的作用是转发调用,里面实际上是调用node里面traverse(NodeVisitor&) 函数。继承Node可以实现自己的traverse动作,例如LOD节点和Switch节点只遍历显示(active)的节点,他们本身有自己的遍历逻辑,所以NodeVisitor::traverse只是转发而已。

    下一步是node::traverse里面调用node::apply(NodeVisitor&);

     

    第二个apply()

    Apply 实际上是在node节点的成员函数转发回去NodeVisitor,里面的实现是

    Group::apply(NodeVisitor&){

    XXX //一些事

    NodeVisitor::apply(*this);

    XXX //一些事

    }

     

    也就是如果我是Group,那么我调用的是NodeVisitor::apply(Group &);这个函数。NodeVisitor里面各种apply函数就是这样被调用的。那么我们看代码看到继承node的各个类唯独只有group显式实现了这样的apply。那么其他类是怎么调用的呢?但是我们并没有在Transform里面明显地找到这样的调用apply(*this),那个这个实现是放在哪里呢?

     

    apply(*this)在哪里实现?

    答案是META_Node宏定义

    一开始我认为这个宏定只是osg里面用到的,用户并不需要用到。后来在NodeVisitor::apply(Transform&);中设置断点才发现META_Node里面有这样的实现

    virtualvoidaccept(osg::NodeVisitor& nv) { if(nv.validNodeMask(*this))

    { nv.pushOntoNodePath(this); nv.apply(*this); nv.popFromNodePath(); } } \

     

    知道这个以后我们就可以做的业务类Visitor,并且继承这个visitor实现一些遍历,这样就省了很多动态转换语句,并且可以知道我可以重载那些apply函数,遍历那些类了。

     

    遇到问题

    虚基类不能直接使用META_Node 所以我做了自己的一个特殊版本

     

    //虚类没有clone函数

    #defineMETA_VNode(library,name) \

            virtualboolisSameKindAs(constosg::Object* obj) const{ returndynamic_cast<constname*>(obj)!=NULL; } \

            virtualconstchar* className() const{ return#name; } \

            virtualconstchar* libraryName() const{ return#library; } \

            virtualvoidaccept(osg::NodeVisitor& nv) { if(nv.validNodeMask(*this)) { nv.pushOntoNodePath(this); nv.apply(*this); nv.popFromNodePath(); } }

     

    //返回的是name*,省去dynamic_cast<name*>()这一步

    #defineMETA_CNode(library,name) \

            virtualname* cloneType() const{ returnnewname(); } \

            //virtualname* clone(constosg::CopyOp& copyop) const{ returnnewname(*this,copyop); }\

             META_VNode(library,name)

     

     

微信扫码关注公众号CPP技术网,微信号cpp_coder,关注我们的公众号,阅读更多精彩内容!每天还可以领取大红包哦!!!每天还可以领取大红包哦!!!每天还可以领取大红包哦!!!
文章来源:C++技术网原创文章版权为网站和作者共同所有,会员文章禁止转载。非会员文章转载做好本文超链接即表示授权转载。通过文章下面的分享按钮可以自由分享所有文章。

返回顶部

在线提问
问题标题:
问题描述:(简陋的描述会导致问题被最后回答、没有针对性回答甚至无法解答。请确保问题描述的足够清楚。)