VTK包括两个基础的子系统:一个已编译的C ++类库和一个“解释”包装层,“解释”包装层让你可以使用Java,Tcl和Python这些解释型语言操作已编译的C++类库。如图3-1。

vtk_android使用 vtk user guide_架构


这种架构的优点是,你可以在编译的C ++语言中构建高效(在CPU和内存中使用)的算法,并保留解释型语言的快速代码开发功能(避免编译/链接循环,简单但功能强大的工具,以及访问GUI工具的接口)。

本节将从高层(high-level)来介绍VTK中的两个主要组件(概念):可视化管线和渲染引擎。可视化管线用于获取或创建数据、处理数据,并将结果写入文件或传递给渲染引擎显示出来。渲染引擎负责创建数据的可视化表示。

1 底层对象模型

VTK对象模型都是超类vtkObject的派生类,几乎所有VTK类都派生自此类,但在某些特殊情况下也会派生自其超类vtkObjectBase。所有VTK类的实例都必须使用其对象的New()方法创建,并且必须使用对象的Delete()方法销毁。VTK对象无法在堆内存上创建,因为其构造函数是受保护的(protected)方法。VTK使用公共超类以及统一的创建和销毁对象的方法,这使得VTK能够提供几个基本的面向对象操作。

1.1 引用计数

VTK对象显式地保存了自身的被引用个数。当一个对象通过静态方法New()被创建时它的引用计数就会被初始化为1,因为必然会有一个原始指针指向这个新的对象:

vtkObjectBase* obj = vtkExampleClass::New();

当其它对于该对象的引用被创建或销毁时,它的引用计数会通过Register()或UnRegister()方法增加或减少。这一般会通过对象的“set”一类方法自动处理:

otherObject->SetExample(obj);

现在它的引用计数就变成了2,因为原始的指针和另外一个对象中存储的指针都指向了它。当不再需要原始指针时,可通过Delete()删除该引用:

obj->Delete();

这时候这个指针就不再拥有对象的引用,也就不能再通过这个指针去访问该对象了。但为了保证有效管理对对象的引用,每次调用New()方法创建对象并使用完成后,都要调用Delete()方法释放内存。

上述手动释放内存的方法较为繁琐,可以通过使用类模板vtkSmartPointer<>提供的智能指针来管理对象,比如上述例子改写为:

vtkSmartPointer<vtkObjectBase> obj = vtkSmartPointer<vtkExampleClass>::New();
otherObject->SetExample(obj);

这样,当智能指针变量超出作用于且不再使用时即可自动释放,不再需要手动调用Delete()方法来释放内存。

1.2 运行时类型信息

VTK中的所有对象都可以通过GetClassName()方法来获取类名标识符,该方法返回一个字符串:

const char* type = obj->GetClassName();

通过IsA()方法可以判断对象的类型,是否为一个类或其子类的实例:

if(obj->IsA("vtkExampleClass")) { ... }

可以通过静态方法SafeDownCast()将父类指针转为子类指针(向下转型)。以下代码只有当对象example是obj的子类实例时才能转型成功,否则返回空指针:

vtkExampleClass* example = vtkExampleClass::SafeDownCast(obj)

1.3 显示对象状态(详细信息)

可以通过对象的Print()方法输出对象的状态(详细信息):

obj->Print(cout);

2 渲染引擎

构成VTK渲染引擎的类主要负责接收可视化管线的输出数据并将其渲染显示到窗口中。以下是相关的一些超类。

2.1 vtkProp

渲染场景中数据的可视化描述是由vtkProp的子类负责的。vtkProp的子类用于确定对象在场景中的位置、大小和方向等信息。其常用的其子类包括:

  • vtkActor:表示几何数据(Geometry Data);
  • vtkVolume:表示体数据(Volumetric Data);
  • vtkAnnotation:表示标注数据(Annotation Data)。

此外,vtkProp还有两个对象:

  • mapper对象:存放数据和渲染信息;
  • property对象:负责控制颜色、不透明度等参数,一般会由对象自动创建。

VTK中定义了大量用于控制特定数据的细化的prop(超过50个)。如vtkImageActor、vtkPieChartActor等

2.2 vtkAbstractMapper

用于存放对输入数据(渲染对象)的引用及相关渲染功能。如常见的mapper:

  • vtkPolyDataMapper:用于渲染多边形几何数据
  • vtkFixedPointVolumeRayCastMapper:用于vtkImageData数据的体渲染;
  • 等等。

2.3 vtkProperty和vtkVolumeProperty

采用单独的属性对象来存放控制渲染效果的参数,这样的好处是可以实现渲染参数的共享。

  • vtkProperty:可用于存储颜色、不透明度、材质的环境光(ambient)系数、散射光(diffuse)系数和反射光(specular)系数等参数;
  • vtkVolumeProperty:用于存储对象的体绘制参数,如颜色和不透明度传递函数等。

2.4 vtkCamera

存储了场景的摄像机参数,包括相机的位置、焦点、向上方向、投影方式(平行投影or透视投影)、视场角及远近裁减平面等。一般会由vtkRenderer自动创建。

2.5 vtkLight

用于计算渲染场景中的光照参数,如光源的方向、颜色及强度等。以及其相对于相机的运动等接口(随相机运动or固定位置)。一般会由vtkRenderer自动创建。

2.6 vtkRenderer

vtkRenderer负责场景的渲染过程,上述对象都会被集中在vtkRenderer对象中。一个渲染窗口(vtkRenderWindow)可以有多个vtkRenderer对象,并可以渲染在窗口的不同区域(可相互覆盖)。

2.7 vtkRenderWindow

负责连接操作系统与VTK的渲染管线。包含一个或多个vtkRenderer对象。

2.8 vtkRenderWindowInteractor

负责监听鼠标、键盘和定时器事件,并通过VTK中的命令/观察者模式做出响应。

2.9 vtkTransform

用于描述对象在三维空间中的变换,本质上是一个4x4的齐次矩阵。

2.10 vtkLookupTable,vtkColorTransferFunction和vtkPiecewiseFunction

用于标量数据可视化过程中,标量数据到颜色和不透明度的映射。

  • vtkLookupTable::用于几何数据渲染;
  • vtkColorTransferFunction和vtkPiecewiseFunction:用于数据体绘制。

2.11 一个简单的例子(利用上述对象来渲染场景)

vtkCylinderSource *cylinder = vtkCylinderSource::New();
vtkPolyDataMapper *cylinderMapper = vtkPolyDataMapper::New();
cylinderMapper->SetInputConnection(cylinder->GetOutputPort());
vtkActor *cylinderActor = vtkActor::New();
cylinderActor->SetMapper(cylinderMapper);
vtkRenderer *ren1 = vtkRenderer::New();
ren1->AddActor(cylinderActor);
vtkRenderWindow *renWin = vtkRenderWindow::New();
renWin->AddRenderer(ren1);
vtkRenderWindowInteractor *iren = vtkRenderWindowInteractor::New();
iren->SetRenderWindow(renWin);
renWin->Render()
iren->Start();

3 渲染管线

VTK可视化管线负责读取或生成数据,主要涉及两种基本数据类型:

  • vtkDataObject:一般类型的数据。有规则结构的数据用vtkDataSet表示,图3-2显示了VTK中支持的vtkDataSet类型(子类)。包含了几何结构(顶点)和拓扑结构(顶点之间的连接关系),另外还有属性数据(标量或向量数据),如图3-3。
  • vtkAlgorithm:又被称为滤波器/过滤器(Filter),用于处理输入数据并产生新的数据对象。图3-4描述了一个可视化管线。

    图3-4和图3-5说明了可视化管线中的一些概念。通过读取(Reader)或创建数据对象产生源数据并传入Filter进行处理,Filter可接收一个或多个数据并产生一个或多个数据输出。Mapper接收传入的数据并将其转换为可被渲染引擎绘制的数据,特殊的,Writer可以看作是将数据写入文件或者数据流的Mapper。

    VTK中可视化管线的构建有几个关键点:
  • VTK中通过Filter的SetInputData-GetOutput或SetInputConnection-GetOutputPort方法进行连接。如:
    aFilter->SetInputData(anotherFilter->GetOutput());
    aFilter->SetInputConnection(anotherFilter->GetOutputPort());
  • VTK中采用“惰性求值方案”(Lazy Evaluation Scheme),即只有在需要数据的时候才进行计算,这是通过每个对象内部记录的修改时间(Modification Time)实现的。
  • 管线的连接中,要求anotherFilter的输出对象与aFilter的输入对象类型相互兼容(如父子关系)。

3.1 管线的执行

如前面所说,VTK中采用Lazy Evaluation方案,因此考虑如下例子:

vtkSphereSource *sphere = vtkSphereSource::New();
sphere->SetRadius(10);
std::cout<<sphere->GetNumberOfPoints()<<std::endl;

上述代码中GetNumberOfPoints()函数的返回值为“0”,但但我们在输出前调用其Update()方法后即可输出正确的点个数:

vtkSphereSource *sphere = vtkSphereSource::New();
sphere->SetRadius(10);
sphere->Update();
std::cout<<sphere->GetNumberOfPoints()<<std::endl;

但通常在一个已经构建好的渲染管线中,当位于管线末端的对象Actor接收到(来自Renderer)的Render()渲染请求后,会将该请求沿着管线向上游回溯(请求数据),最终,整条管线被执行。如图3-6所示:

vtk_android使用 vtk user guide_系统概述_02

3.2 图像处理

VTK支持大量的图像处理和体绘制功能,包括二维图像和三维图像(volume),它们都可以被看作vtkImageData。

4 VTK中的设计模式

VTK中采用命令/观察者模式来设计实现回调函数(Callback,又称为用户方法/UserMethod/回调函数)。VTK中几乎所有类均可通过SetObserver()方法创建回调函数。观察者(Observer)通过监听对象中相应的激活事件,当事件被激活时执行相应的命令(Command)并调用回调函数(Callback)。