本文来探究下OSG中的各种数组!

一、根据我平日用的较多的,(比如设置顶点数组,法线数组),先看osg::Vec3。
osg::Vec3定义在<osg/Vec3>头文件中,只有一句:
namespace osg {
  typedef Vec3fVec3;
}
仅是Vec3f的别名。
二、osg::Vec3f
顾名思义,是有三个float元素的容器。下面详细探究其内部。
1. 为float定义个别名。
typedef float value_type;
这种方式经常出现(VTK也用此方式),所以说这是个好习惯?
2. 定义个枚举,标识元素个数。
enum { num_components = 3 };
为什么使用枚举而不使用int常量甚至是short呢?
3. 定义内部元素数组。
value_type _v[3];
4. 构造函数与析构函数。
      Vec3f() { _v[0]=0.0f; _v[1]=0.0f;_v[2]=0.0f;}
      Vec3f(value_type x,value_type y,value_type z) {_v[0]=x; _v[1]=y; _v[2]=z; }
      Vec3f(const Vec2f& v2,value_typezz)
      {
          _v[0] =v2[0];
          _v[1] =v2[1];
          _v[2] =zz;
      }
构造函数的三个版本都很简单。
5. 重载比较操作符。
      inline bool operator == (constVec3f& v) const { return _v[0]==v._v[0]&& _v[1]==v._v[1]&& _v[2]==v._v[2]; }
      inline bool operator != (constVec3f& v) const { return _v[0]!=v._v[0] ||_v[1]!=v._v[1] || _v[2]!=v._v[2]; }
      inline bool operator <  (const Vec3f& v) const
      {
          if(_v[0]<v._v[0]) return true;
          else if(_v[0]>v._v[0]) return false;
          else if(_v[1]<v._v[1]) return true;
          else if(_v[1]>v._v[1]) return false;
          elsereturn (_v[2]<v._v[2]);
      }
这里重载了三个操作符:==, !=, <,关键在于,如何比较两个向量的大小。
6. 重载中括号[]操作符,以实现下标操作。
      inline value_type& operator [](int i) { return _v[i]; }
      inline value_type operator [] (int i) const {return _v[i]; }
7. 重载*操作符,实现点乘、标量乘法。
      inline value_type operator * (constVec3f& rhs) const
      {
          return_v[0]*rhs._v[0]+_v[1]*rhs._v[1]+_v[2]*rhs._v[2];
      }
      inline const Vec3f operator * (value_type rhs)const
      {
          returnVec3f(_v[0]*rhs, _v[1]*rhs, _v[2]*rhs);
      }
两个向量的点乘结果是一个值。
8. 重载^操作符,实现叉乘。
inline const Vec3f operator ^ (const Vec3f&rhs) const
      {
          returnVec3f(_v[1]*rhs._v[2]-_v[2]*rhs._v[1],
                        _v[2]*rhs._v[0]-_v[0]*rhs._v[2] ,
                        _v[0]*rhs._v[1]-_v[1]*rhs._v[0]);
      }
两个向量的叉乘是垂直于这两个向量的向量(向量叉乘的方法,我忘了这个算法
9. 重载*=操作符,点乘后再赋值。
      inline Vec3f& operator *=(value_type rhs)
      {
          _v[0]*=rhs;
          _v[1]*=rhs;
          _v[2]*=rhs;
          return*this;
      }
为什么要返回*this?
考虑如下:
Vec3f v1,v2,v3;
v1 = v2 *= v3;
如果没有返回*this,此语句是不合法的。但是,谁会写v1 = v2 *= v3这种可读性巨差的语句呢?
10. 重载/操作符,实现标量除法:
      inline const Vec3f operator / (value_type rhs)const
      {
          returnVec3f(_v[0]/rhs, _v[1]/rhs, _v[2]/rhs);
      }
并没有检测rhs的值,留给运行时么?
11. 重载/=操作符,实现除完后赋值。
      inline Vec3f& operator /=(value_type rhs)
      {
          _v[0]/=rhs;
          _v[1]/=rhs;
          _v[2]/=rhs;
          return*this;
      }
12. 重载+、+=、-、-=操作符。
      inline const Vec3f operator + (constVec3f& rhs) const
      {
          returnVec3f(_v[0]+rhs._v[0], _v[1]+rhs._v[1], _v[2]+rhs._v[2]);
      }
      inline Vec3f& operator += (constVec3f& rhs)
      {
          _v[0] +=rhs._v[0];
          _v[1] +=rhs._v[1];
          _v[2] +=rhs._v[2];
          return*this;
      }
      inline const Vec3f operator - (constVec3f& rhs) const
      {
          returnVec3f(_v[0]-rhs._v[0], _v[1]-rhs._v[1], _v[2]-rhs._v[2]);
      }
      inline Vec3f& operator -= (constVec3f& rhs)
      {
          _v[0]-=rhs._v[0];
          _v[1]-=rhs._v[1];
          _v[2]-=rhs._v[2];
          return*this;
      }
13. 负号-操作符。
      inline const Vec3f operator - () const
      {
          returnVec3f (-_v[0], -_v[1], -_v[2]);
      }
14. 得到指向内部数组的指针。
      inline value_type* ptr() { return _v; }
      inline const value_type* ptr() const { return_v; }
提供了两个版本,版本2不允许修改内部元素。
15. 提供了两个版本的访问元素的方法。
      inline value_type& x() { return_v[0]; }
      inline value_type& y() { return_v[1]; }
      inline value_type& z() { return_v[2]; }
      inline value_type x() const { return _v[0];}
      inline value_type y() const { return _v[1];}
      inline value_type z() const { return _v[2];}
版本1返回的是引用,故而是可以修改内部值的,版本2则仅仅是读。
16. 得到向量的长度。
      inline value_type length2() const
      {
          return_v[0]*_v[0] + _v[1]*_v[1] + _v[2]*_v[2];
      }
17. 向量的标准化。
      inline value_type normalize()
      {
          value_typenorm = Vec3f::length();
          if(norm>0.0)
          {
              value_type inv =1.0f/norm;
              _v[0] *= inv;
              _v[1] *= inv;
              _v[2] *= inv;
          }               
          return(norm );
      }
为什么在实现时不直接除以norm,而要乘以1.0f/norm。并return了nrom,有什么意义吗?
18. set值的方法。
      inline void set( value_type x, value_type y,value_type z)
      {
          _v[0]=x;_v[1]=y; _v[2]=z;
      }
      inline void set( const Vec3f&rhs)
      {
          _v[0]=rhs._v[0]; _v[1]=rhs._v[1]; _v[2]=rhs._v[2];
      }
19.另外在<osg/Vec3f>中提供了两个非成员函数:
inline Vec3f componentMultiply(const Vec3f&lhs, const Vec3f& rhs)
{
  returnVec3f(lhs[0]*rhs[0], lhs[1]*rhs[1], lhs[2]*rhs[2]);
}
inline Vec3f componentDivide(const Vec3f& lhs,const Vec3f& rhs)
{
  returnVec3f(lhs[0]/rhs[0], lhs[1]/rhs[1], lhs[2]/rhs[2]);
}

[osg]源码分析:osg::Vec3, osg::Vec3f相关推荐

  1. [前沿技术] AMD FSR 1.0源码分析(二)

    FSR技术分析 前文:[前沿技术] AMD FSR 1.0源码分析(一) 2. EASU源码分析 2.3 FsrEasuF分析 1️⃣首先,就参数而言,主要是: void FsrEasuF(out A ...

  2. threejs 源码解析_ThreeJS 物理材质shader源码分析(顶点着色器)

    ThreeJS 物理材质shader源码分析(顶点着色器) Threejs将shader代码分为ShaderLib和ShaderChunk两部分,ShaderLib通过组合ShaderChunk的代码 ...

  3. 【Golang源码分析】Go Web常用程序包gorilla/mux的使用与源码简析

    目录[阅读时间:约10分钟] 一.概述 二.对比: gorilla/mux与net/http DefaultServeMux 三.简单使用 四.源码简析 1.NewRouter函数 2.HandleF ...

  4. SpringBoot-web开发(四): SpringMVC的拓展、接管(源码分析)

    [SpringBoot-web系列]前文: SpringBoot-web开发(一): 静态资源的导入(源码分析) SpringBoot-web开发(二): 页面和图标定制(源码分析) SpringBo ...

  5. SpringBoot-web开发(二): 页面和图标定制(源码分析)

    [SpringBoot-web系列]前文: SpringBoot-web开发(一): 静态资源的导入(源码分析) 目录 一.首页 1. 源码分析 2. 访问首页测试 二.动态页面 1. 动态资源目录t ...

  6. SpringBoot-web开发(一): 静态资源的导入(源码分析)

    目录 方式一:通过WebJars 1. 什么是webjars? 2. webjars的使用 3. webjars结构 4. 解析源码 5. 测试访问 方式二:放入静态资源目录 1. 源码分析 2. 测 ...

  7. Yolov3Yolov4网络结构与源码分析

    Yolov3&Yolov4网络结构与源码分析 从2018年Yolov3年提出的两年后,在原作者声名放弃更新Yolo算法后,俄罗斯的Alexey大神扛起了Yolov4的大旗. 文章目录 论文汇总 ...

  8. ViewGroup的Touch事件分发(源码分析)

    Android中Touch事件的分发又分为View和ViewGroup的事件分发,View的touch事件分发相对比较简单,可参考 View的Touch事件分发(一.初步了解) View的Touch事 ...

  9. View的Touch事件分发(二.源码分析)

    Android中Touch事件的分发又分为View和ViewGroup的事件分发,先来看简单的View的touch事件分发. 主要分析View的dispatchTouchEvent()方法和onTou ...

最新文章

  1. HTML中单选框的设置,和提交按钮之间的组合
  2. 选择排序算法(基于Java实现)
  3. numpy 100题
  4. iPhone5帮助了谁?
  5. 系统相机裁剪比例_拍不出好照片,你缺的不是好手机而是相机设置的秘笈
  6. 腾讯视频怎么上传自己的视频?
  7. Android第十四期 - 可扩展选项卡
  8. Mybatis Plus 的BaseMapper 和 Model有什么用
  9. PHP06 流程控制
  10. profibus 主站软件_PROFIBUS网络的DP主站与DP从站
  11. 三级数据库知识点总结
  12. HTML干净的网址导航
  13. 二十一世纪大学英语读写教程(第二册)学习笔记(原文)——9 - Get Ready for Some Wild Weather(准备应对厄尔尼诺)
  14. 读懂React原理之调和与Fiber
  15. cad添加自己线性_CAD2014怎么自定义线型? cad设置线型的方法
  16. layui table 渲染动态列及列数据
  17. 米斯齐超声波传感器显示测量距离(oled)内附Arduino代码
  18. rocketmq初学者入门
  19. 做一个功能比较齐全的小程序商城选择好的系统很重要
  20. CS224d: Deep Learning for NLP Lecture1 听课记录

热门文章

  1. 限制input输入字符数(中文2个字符,英文1个字符)
  2. 我的HTML学习(二)----html的基本分类与字符集的学习
  3. 计算机等级二级等保要求
  4. mac 重装 mysql
  5. Java String的intern方法
  6. 华科智标_停车场定位导航反向寻车系统
  7. 几行CSS让你的页面立体起来
  8. Linux学习笔记(六)——文件打包与解压缩
  9. 微信广告转化归因几个坑 gdt_vid和clickid
  10. 基于Java毕业设计智创员工管理系统源码+系统+mysql+lw文档+部署软件