zoukankan      html  css  js  c++  java
  • [osg]osgDB的加载机制,使用3DS插件做参考(转,整理现有osgDB资料)

    参考:http://blog.sina.com.cn/s/blog_7cdaf8b60102uzu3.html

    http://blog.csdn.net/wang15061955806/article/details/51011466

       osgDB机制概述

      osgDB库允许用户程序加载、使用和写入3D数据库,它采用插件管理的架构,可以支持大量常见的2D图形和3D图形文件格式。osgDB负责维护插件的信息注册表,并负责检查将要被载入的OSG插件接口的合法性。由于大型3D地形数据通常是多段数据块的组合体。因此,应用程序从文件中读取各部分数据库信息时,需要在不干扰当前渲染的前提下以后台线程的方式进行,osgDB::DatabaseParger提供了这样的功能。

      插件的不足之处在于,其设计过程中只能遵循固定的格式和工作模式进行编程,它的一切行为都无法超出主系统所提供的公共接口规范。

      osg插件是一组动态链接库,其中实现了osgDB头文件ReaderWriter定义的接口。OSG不可能查找并加载所有的插件以获取它们支持的文件格式,这样,在程序启动时将会是一个很大的开销。因此,OSG使用职责链(Chain of Responsibility)的设计模式,以加载尽量少的插件。当用户程序尝试使用osgDB读取或写入文件时,OSG将按照如下步骤来查找合适的插件:

      1.OSG搜索已注册的插件列表,查找支持文件格式的插件。开始时已注册插件列表仅包含了Registry类构造函数中注册的插件。如果OSG找到了可以支持此文件格式的插件,并成功执行了I/O操作,那么它将返回相应的数据。

      2.如果没有发现可以支持此格式的已注册插件,或者I/0操作失败,那么OSG将根据前面所述的文件命名规则创建插件文件的名称,并尝试读取相应的插件库。如果读取成功,OSG将添加此插件到已注册插件列表中。

      3.OSG将重复步骤(1),如果文件I/O的操作再次失败,OSG将返回失败信息。

        

      用3DS插件做说明:

      先就3ds格式模型做一个简要说明,在加载3ds模型时,会根据模型格式在电脑的所有环境变量中找到对应的dll文件(osgdb_3ds.dll),并试着读取osgdb_3ds.dll,加载并成功执行osgdb_3ds.dll的宏定义,就会成功解析文件。

        程序中代码的实现顺序如下图所示:

                             osgDB插件机制
    查找读取插件流程

        

    函数执行顺序:
        osg::Node *node=osgDB::readNodeFile(某种类型的文件路径),这个语句是osg读取节点数据常用的函数,下面对这个函数进行跟踪,简单介绍下数据的处理过程,方面自己对osgDB的插件进行简单的修改,供自己进行简单的修改;
        osg::Node *readNodeFile(const std::string &filename)  //函数所在位置osgDB/ReadFile,该函数属于文件作用域,调用文件中的另一个函数

    1 {
    2      //函数所在位置osgDB/ReadFile,该函数只是在原来的基础上,调用了osgDB/Registry的文件中的一个Registry单例,获取的是option,默认为空
    3      return readNodeFile(filename,osgDB::Registry::instance()->getOptions());//osgDB/ReadFile ①
    4    }

    //①的具体实现,该函数的作用是调用osgDB/Registry的单例,进行数据的解析,参数options默认为空.

    1 Node* osgDB::readNodeFile(const std::string& filename,const Options* options)
    2 {
    3     ReaderWriter::ReadResult rr = Registry::instance()->readNode(filename,options);②
    4 }

    //②的具体实现,由于options参数为空,并且_readFileCallback为空,所以函数直接执行③

    1     ReaderWriter::ReadResult readNode(const std::string& fileName,const Options* options, bool buildKdTreeIfRequired=true)
    2         {
    3              ReaderWriter::ReadResult result;
    4             
    5              result = readNodeImplementation(fileName,options);③
    6         }

    //③的具体实现,直接执行函数④

    1 ReaderWriter::ReadResult Registry::readNodeImplementation(const std::string& fileName,const Options* options)
    2 {
    3     return readImplementation(ReadNodeFuncto(fileName,options),Options::CACHE_NODES);④                      
    4 }

    //④的具体实现,由于readFunctor的options为空,所以函数在中间没有做实际的数据处理,直接执行⑤

    1 ReaderWriter::ReadResult Registry::readImplementation(const ReadFunctor& readFunctor,Options::CacheHintOptions cacheHint)
    2 {
    3     std::string file(readFunctor._filename);
    4    ReaderWriter::ReadResult rr = read(readFunctor);⑤
    5    return rr;
    6 }

    //⑤的具体实现,这里是主要的实现,主要的思想是,先从_rwList查找是否可以解析当前类型的数据,如果能解析,直接返回解析结果

    //如果不能解析,根据当前的文件名称,创建一个跟当前名称相关的动态库的名称,然后使用loadLibrary加载该动态库,加载成功后

    _rwList这里面存储的对象的数目会增加,这个是个有诀窍的地方,在这个工程中没有发现有对_rwList做增加操作的代码,那么_rwList是怎么增加的

    主要的关键点是,对于每一个dll中都有个宏调用,该宏是如下,这个宏生成了一个静态的对象osgDB::RegistryReaderWriterProxy,这个对象的作用就是

    增加_rwList的对象数目,这是注册机制的关键。然后使用新增加的readerWrite完成对数据文件的读取

     1 /////////////#define REGISTER_OSGPLUGIN(ext, classname) 
     2 ////////////extern "C" void osgdb_##ext(void) {} 
     3 ///////////  static osgDB::RegisterReaderWriterProxy g_proxy_##classname;
     4 ReaderWriter::ReadResult Registry::read(const ReadFunctor& readFunctor)
     5 {
     6   
     7     // first attempt to load the file from existing ReaderWriter's
     8    //看是否有可用的ReaderWriter,对当前的数据进行解析,如果解析成功,就返回结果
     9     AvailableReaderWriterIterator itr(_rwList, _pluginMutex);
    10     for(;itr.valid();++itr)
    11     {
    12         ReaderWriter::ReadResult rr = readFunctor.doRead(*itr);
    13         if (readFunctor.isValid(rr)) return rr;
    14     }
    15     //根据文件名称创建新的动态库名称,然后加载动态库,加载动态库,增加_rwList的对象个数 
    16     std::string libraryName = createLibraryNameForFile(readFunctor._filename);
    17     if (loadLibrary(libraryName)!=NOT_LOADED)
    18     {
    19         //重新遍历一下,使用新的ReaderWriter进行数据的解析
    20         for(;itr.valid();++itr)
    21         {
    22             ReaderWriter::ReadResult rr = readFunctor.doRead(*itr);
    23             if (readFunctor.isValid(rr)) return rr;
    24             else results.push_back(rr);
    25         }
    26     }
    27 }
  • 相关阅读:
    vs与linux的交叉编译环境搭建
    layui框架部分功能介绍
    谷歌添加百度翻译提示Google已将百度翻译标记为恶意程序并阻止安装,怎么办
    七,JOBC数据库编程
    mysql数据库
    六,IO系统
    五,图形界面编程
    四,集合框架
    三,反射类
    二,常用类
  • 原文地址:https://www.cnblogs.com/lyggqm/p/6385284.html
Copyright © 2011-2022 走看看