OpenCASCADE开发指南<四>:OCC 数据类型和句柄

news2025/1/16 10:57:01

  一个软件首先要规定能处理的数据类型, 其次要实现三项最基本的功能——引用管理、内存管理和异常管理。在 OCC 中,这三项功能分别对应基础类中的句柄、内存管理器和异常类。
在这里插入图片描述

1 数据类型

  在基本概念篇里,已经介绍了 OCC 数据类型的分类,即依据处理方式的不同,分为值处理类型和句柄处理类型两类。在详细描述这两种类型之前,先描述 OCC 数据的基本类型。

1. 1 基本类型

  所有基本类型都是用 CDL 预定义的,并且只能通过值处理。依据不同的出自, 或者依据各自的存储性能, 它们可分为耐存的和非耐存的。 耐存的基本类型由 Standard_Storable 类派生, 能应用于持久对象的实现(要么包含于持久对象方法声明的实体中,要么作为持久对象内部结构的一部分)。表 3.1 给出 OCC 提供的所有基本类型和相应的 C++基本类型。

在这里插入图片描述
下面分别论述表中的类型。

(1) Standard_Integer(整型)。它是由 32 位二进制数表示的基本类型,包括正数、 负数和零。 Integer 类型与 C++ int 类型一样。 因此, 可以对 Integer 类型进行+、 -、 *、 /四种代数运算,也可以对其进行<、 <=、 ==、 !=、 >=、 >六种关系运算。

(2) Standard_Real(实型)。 它表示具有确定精度和确定范围的实数的基本类型。 Real 类型与 C++ 中 double(双精度)类型一样。因此, +、 -、 *、 /四种代数运算、 -取反运算和<、 <=、 ==、!=、 >=、 >六种关系运算同样适用于Real 类型。

(3) Standard_ShortReal(短实型)。 它表示具有确定精度和确定范围的实数。ShortReal 类型与 C++中 float 类型一样。 因此, +、 -、 *、 /四种代数运算、 -取反运算和<、 <=、 ==、 !=、 >=、 >六种关系运算同样适用于 ShortReal 类型。

(4) Standard_Boolean(布尔类型)。 它是描述逻辑值的基本类型。 它有两种值: false 和 true。 Boolean 类型与 C++中 unsigned int 类型一样。因此,与、或、异或、非四种代数运算和==、 !=两种关系运算同样适用于 Boolean 类型。

(5) Standard_Character(字符类型)。 它是用来表示 ASCII 字符集的一种基本类型。它能被赋予的值有 128 个,对应 128 个 ASCII 字符。 Character 类型与C++ 中 char 类型一样。 因此, <、 <=、 ==、 !=、 >=、 >六种关系运算同样适用于 Character 类型(如: A<B)。

(6) Standard_ExtCharacter(扩展字符类型)。它是用来表示 Unicode 字符集的一种基本类型。 由它表示的字符得用 16 位二进制数进行编码。 ExtCharacter类型与 C++ 中 short 类型一样。 因此, <、 <=、 ==、 !=、 >=、 >六种关系运算同样适用于ExtCharacter 类型(如: A<B)。

(7) Standard_CString(C 串类型)。 它用来表示文字串。 一个文字串就是由双引号括起来的一个 ASCII 字符序列。CString 类型与 C++中 char*类型是一样的。

(8) Standard_Address(地址类型)。 它用来表示一个通用指针。 Address 类型与 C++中 void*类型一样。

(9) Standard_ExtString(扩展串类型)。 它用来表示由 Unicode 字符序列构成的文字串。 ExtString 类型与 C++中 short*类型一样。

1. 2 值处理类型

  值处理类型可分三大类:
(1)基本类型;
(2)枚举类型;
(3)由这样一些类(既不是由 Standard_Persistent 类派生,也不是由Standard_Transient 类派生,无论是直接派生还是间接派生)定义的类型。

  值处理类型的表现形式比句柄处理类型的表现形式更直接。 因此, 对值处理类型的操作也会更快。 但是值处理类型对象不能单独存于文件中。 图 3.1 表示了对一个值处理类型对象的处理过程。

  需要注意的是: 那些能被数据模式识别(包括基本类型和从 Storable 类继承过来的类型)的值处理类型可以作为持久对象的部分结构而存储在持久对象内部。这是值处理类型对象能够存进文件的唯一方式。
在这里插入图片描述

1. 3 句柄处理类型

  句柄处理类型可以分为两大类:

(1) 由 Persistent 类的派生类定义的类型。 这些类型可以被长久地存在文件中。
(2)由 Transient 类的派生类定义的类型。

  图 3.2 表示了对一个句柄处理类型对象的处理过程。
在这里插入图片描述

2 句柄

2.1 句柄的定义

  OCC 的引用管理采用的是一种句柄机制。这种机制的基本元素是句柄。在 OCC 中,句柄是通过类实现的。句柄含有多个接口成员,其中一个包含一个引用。一般情况下,仅需要使用它的引用。正因为这样,习惯将句柄比作C++指针。 与 C++指针一样, 几个句柄可以引用同一个对象; 一个句柄也可以引用多个对象,但是每次只能引用一个;在句柄访问对象前,句柄必须被声明。

2.2 句柄处理类的组织

  一般情况下, 真正需要的是句柄引用的对象而非引用本身。 在此, 有必要介绍一下句柄处理类的组织。句柄处理类要么是持久的,要么是短暂的。如果由Standard_Transient 类派生,则是短暂的;如果由 Standard _Persistent 类派生, 则
是持久的。 不论短暂还是持久, 它们的组织情况是一样的。 故, 下面一段文字将仅介绍短暂句柄处理类及其相关句柄的组织情况。

  Standard_Transient 类是 OCC 中所有句柄处理类的一个根类(另一个根类是Standard_Persistent 类)。它提供了一个引用计数器(被其所有后裔类继承)。该计数器被 Handle()类(也就是所谓的句柄) 使用, 用于计算指向对象实例的句柄
数。对于每一个继承(直接或间接)自 Transient 类的类, CDL 提取器都创建了相应的 Handle()类(句柄)。该 Handle()类(句柄)的参数名字和由“Handle_”作前缀修饰的名字一样。 OCC 专门提供了一个宏预处理函数 Handle()。 它能够将一个 Handle()类(句柄)的参数名字提取出来作为指定短暂类的名字。

这里提三个注意事项:

(1) Transient 类和 Persistent 类不完全是通过句柄处理的,它们也可以通过值处理;
(2)持久对象不能含有非耐存句柄(那些引用非持久对象的句柄);
(3) 使用句柄的目的是共享对象(对于所有局部操作, 建议使用值处理类)。

2.3 句柄的使用

  句柄通过它引用的对象被特征化。在对一个短暂对象进行任何操作之前, 必须对句柄进行声明。比如, Point 和 Line 是来自 Geom 包的两个短暂类,声明得像例 3.1 这样写。

例 3.1:

Handle(Geom_Point) p1, p2;
Handle(Geom_Line) aLine;

  对一个句柄进行声明,只是创建了一个空句柄,该句柄不指向任何对象。 要初始化句柄, 要么得创建一个新的对象, 要么得将其它句柄值赋予该句柄(假定两种句柄类型是兼容的)。

2.4 句柄的类型管理

  首先介绍句柄的通用管理。OCC 能以通用方式对数据类型进行描述。这样,可以在程序运行时才去严格核对给定对象的类型。这与 C++ RTTI(运行时类型信息机制)类似。对于每一种由 Standard_Transient 类派生的类类型, CDL 提取器都创建了相应的代码段,用于对 Standard_Type 类进行实例化。 通常 Standard _Type 类(也称类型描述器)持有类型信息:类型名和其祖先类型列表。

  类型的实例(实际上是指向该类型的句柄)由虚函数 DynamicType()(该虚函数在 Standard_Transient 类的派生类中)返回。检查给定对象是否具有给定类型或给定类型的后裔类型,需调用另一个虚函数 IsKind()。

  为给定类类型寻找相关的类型描述器, 得用相关的宏 STANDARD_TYPE()。其中的宏参数就是给定类的名字。接着介绍句柄的类型一致原则。句柄声明中的对象类型是对象的静态类型, 它能被编译器识别。

  句柄能够引用静态类型的子类对象。 因此, 对象的动态类型(也称对象的实际类型) 可以是静态类型的后裔类型。这就是句柄的类型一致原则。

  考虑到持久类 CartesianPoint 是 Point 类的一个子类, 所以, 类型一致原则可以用例 3.2 表示。
例 3.2:

Handle (Geom_Point) p1;
Handle (Geom_CartesianPoint) p2;
p2 = new Geom_CartesianPoint;
p1 = p2; //可以,类型是兼容的。

例子中 ,编译器将 p1 看做是指向 Point 类的句柄 ,尽管 p1 实际指向CartesianPoint 类型对象。

  最后介绍句柄的直接类型转换。依据类型一致原则,我们总可以将低层句柄向上赋值给高层句柄。但是, 反过来则不行。因此,我们需要一种直接类型转换机制。如果一个句柄所指对象的实际类型是抛掷者(抛掷句柄的对象)的后裔类型,那么, 该句柄可以直接向其子类型转换。 这就是句柄的直接类型转换(见例 3.3)。

3.3Handle (Geom_Point) p1;
Handle (Geom_CartesianPoint) p2, p3;
p2 = new Geom_CartesianPoint;
p1 = p2; //可以,标准的赋值。
p3 = Handle (Geom_CartesianPoint) : : DownCast (p1);
// 可以, p1 的实际类型是 CartesianPoint 句柄,尽管它的静态类型是 Point 句柄。

  如果直接转换与句柄所指对象的实际类型不兼容, 那么被抛掷的句柄会被清空, 并且不会产生任何异常。 因此, 如果需要一些能在静态类型的子类型中实现的可靠服务,如例 3.4 编写程序:
例 3.4:

void MyFunction (const Handle(A) & a)
{
Handle (B) b = Handle (B):: Downcast(a);
if (! b.IsNull())
{
//如果 B 类由 A 类派生,我们就可以使用“b”。
}
else
{
// 类型不兼容。
}
}

  向下抛掷尤其被用于处理由不同类型对象组成的集合容器, 但是有一个限制条件——这些对象必须继承自同一个根类。 例如, 有一个由多个短暂对象构成的SequenceOfTransient 序列,同时还有两个继承自 Standard_T- ransient 类的类, 那
么例 3.5 的构造语句是有效的。

例 3.5:

Handle (A) a;
Handle (B) b;
Handle (Standard_Transient) t;
SequenceOfTransient s;
a = new A;
s.Append (a);
b = new B;
s.Append (b);
t = s.Value (1);
//这里我们不能这样写:
a = t;
//这是错误的。
// 因此我们向下抛掷:
a = Handle (A) :: Downcast (t)
if (! a.IsNull())
{
//类型兼容的话,我们就可以使用 “a”。
}
else
{
// 类型不兼容。
}

2.5 用句柄创建对象

  要创建一个由句柄处理的对象,得对句柄进行声明,并用 C++ new 操作符初始化句柄。 这样的声明和初始化必须被构造函数的调用紧跟着, 如例 3.6 所示。

例 3.6:

Handle (Geom_CartesianPoint) p;
p = new Geom_CartesianPoint (0, 0, 0);

  与 C++指针不同, 句柄并不支持 delete 函数。 当句柄所指对象不再被使用时,该对象会被自动析构。

2.6 通过句柄调用对象方法

  对于指向持久对象或短暂对象的句柄,可以像使用 C++指针那样使用它。 通过句柄,可以调用对象方法。调用对象方法有两种方式: 一是使用操作符“->”; 二是使用函数调用语句。但是当需要调用方法来测试或者修改句柄状态时,则必须用操作符“.”。

例 3.7 说明了怎样获取点的坐标。

例 3.7:

Handle (Geom_CartesianPoint) centre;
Standard_Real x, y, z;
if (centre.IsNull())
{
centre = new PGeom_CartesianPoint (0, 0, 0);
}
centre->Coord(x, y, z);

例 3.8 说明了怎样获取一个笛卡尔点的类型。

例 3.8:

Handle(Standard_Transient) p = new Geom_CartesianPoint(0.,0.,0.);
if ( p->DynamicType() == STANDARD_TYPE(Geom_CartesianPoint) )
cout << "Type check OK" << endl;
else
cout << "Type check FAILED" << endl;

  需要注意的是: 如果指向对象方法或对象定义域的句柄是空的, 那么将产生NullObject 异常。

2.7 句柄的存储分配

  在删除对象前,必须确保对象没有被引用。为了减少与对象生存管理有关的编程, 在每个句柄处理类里都含有一个删除函数。 句柄能使引用计数管理自动进行, 并且当对象不再被引用时, 自动析构对象。通常,不能对 Standard_Transient 类的子类实例直接调用 delete 函数。

  当对象有一个新句柄时, 引用计数器的值加 1。 当一个句柄被删除, 或被清空, 或被重新赋值而指向另一对象时, 计数器的值减 1。 一旦引用计数器的值为0 时,对象将被自动析构。句柄的分配规则可以通过例 3.9 体现出来。

例 3.9:

...
{
Handle (TColStd_HSequenceOfInteger) H1 = new TColStd_HSequenceOfInteger;
// H1 有一个引用,对应的内存空间是 48 字节。
{
Handle (TColStd_HSequenceOfInteger) H2;
H2 = H1;
// H1 有两个引用。
if (argc == 3)
{
Handle (TColStd_HSequenceOfInteger) H3;
H3 = H1;
// H1 有三个引用。
...
}
// H1 有两个引用。
}
// H1 有一个引用。
}
// H1 没有引用。
// TColStd_HSequenceOfInteger 对象被析构。

  如果两个或多个对象通过句柄(作为定义域)互相引用,就会出现循环。 在这种情况下,对象不会被自动析构。

  以图表(graph)为例,它的元素对象得知道它们所属的图表对象,也就是说,元素得有一个对图表对象的引用。如果元素和图表都是通过句柄处理的, 并且都以句柄作为自己的定义域, 则将出现循环。 当最后一个句柄被删除时, 图标对象不会被删除。这是因为图表里面还有许多指向该图表的句柄——这些句柄作为图表的数据结构(元素)被存储。

  有两种方式可以避免循环的出现:
  (1)用 C++指针代替每一个引用(比如由元素对图表的引用)。
  (2)当图表对象需要被删除时,将整套句柄(如元素中指向图表的句柄)清空。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1513587.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

(2021 AAAI) Self_MM

《Learning Modality-Specific Representations with Self-Supervised Multi-Task Learning for Multimodal Sentiment Analysis》 Abstract 表征学习是多模态学习中一项重要而富有挑战性的任务。有效的模态表示应该包含两部分特征:一致性和差异性。由于多模态标注的统一性,…

Java集合中经典的 5种设计模式,打死也要记住啊!

集合 一、 迭代器模式(Iterator Pattern)二、 工厂模式(Factory Pattern)三、 装饰器模式(Decorator Pattern)四、 适配器模式(Adapter Pattern)五、 组合模式(Composite Pattern) Java 集合框架中的 List、Set、Map 以及其实现类都使用了多种经典的设计模式 一、 迭代器模式(I…

《汇编语言》- 读书笔记 - 第17章-外传之 DOSBox-X 调用 int 13 读写磁盘

《汇编语言》- 读书笔记 - 第17章-外传之 DOSBox-X 调用 int 13 读写磁盘 总结dosbox-x.conf 不完美读取成功写入成功参考资料 总结 DOSBox 中访问 int 13h 始终没反应。网上查了下有人说是没支持&#xff0c;建议使用 DOSBox-X 经过无数遍尝试后&#xff1a; 环境状态Win11…

云计算 3月13号 (OSI 七层模型:物理层、数据链路层、网络层、传输层、会话层、表示层、应用层)

走进网络 1.认识计算机 1.计算机网络是由计算机和通讯构成的&#xff0c;网络研究的是“通信”。 ------1946 世界上第一台计算机 2.终端&#xff1a;只有输入和输出功能&#xff0c;没有计算和处理功能。 3.数据&#xff1a;一串数字&#xff08;二进制数&#xff09;&#…

【el-dialog】解决同一组件使用俩个el-dialog,内容被遮罩层覆盖的问题

如果需要在一个 Dialog 内部嵌套另一个 Dialog或者同一组件有多个Dialog时&#xff0c;需要使用 append-to-body属性 &#xff0c;只要在第二次弹框上面加上属性

第十四届蓝桥杯省赛真题 Java A 组【原卷】

文章目录 发现宝藏【考生须知】试题 A \mathrm{A} A : 特殊日期试题 B: 与或异或试题 C : \mathrm{C}: C: 平均试题 D: 棋盘试题 E : \mathrm{E}: E: 互质数的个数试题 F: 阶乘的和试题 G: 小蓝的旅行计划试题 H: 太阳试题 I: 高塔试题 J \mathrm{J} J : 反异或 01 串 发现…

MongoDB常见面试题总结(一)MongoDB面试题及答案

1. MongoDB的特点&#xff1a; 你能简要介绍一下MongoDB吗&#xff1f;它与关系型数据库的主要区别是什么&#xff1f; MongoDB是一个开源、面向文档的NoSQL数据库&#xff0c;它采用了BSON&#xff08;Binary JSON&#xff09;格式存储数据。以下是MongoDB与关系型数据库的主…

【开发】微服务整合Sentinel

目录 前言 1W&#xff1a;什么是Sentinel&#xff1f; 2W&#xff1a;为什么使用Sentinel&#xff1f; 3W&#xff1a;如何使用Sentinel&#xff1f; 1. 在pom.xml中导入Sentinel依赖坐标 2. 配置控制台 3. 访问API接口的任意端点 流量控制 1. 簇点链路 2. 快速入门…

【HTML】1px边框与1px分割线

对比图 箭头标注的是处理过的 1px分割线 使用transform的scaleY进行缩小 码 <div class"mini-heriz"></div><br><div style"border: solid 1px black; width: 300px;height: 1px;"></div> <style> .mini-heriz {wi…

Java的变量类型详解

目录 局部变量 实例变量 类变量&#xff08;静态变量&#xff09; 参数变量 实例分析 总结 在Java这门静态类型的编程语言中&#xff0c;如何巧妙地使用变量&#xff0c;就像是掌握了一把精准的雕刻刀&#xff0c;能让你在编码的世界里自由地创造。变量在Java中的应用无处…

2024年了,SEO优化是不是已经穷途末路了呢?(川圣SEO)蜘蛛池

baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; 2024年了&#xff0c;SEO优化是不是已经穷途末路了呢&#xff1f;#蜘蛛池SEO SEO优化并没有穷途末路。虽然随…

pcl弧度角度换算:rad2deg,deg2rad

角度弧度换算公式: 代码及结果在:cmath 中cos sin等常用函数的坑(弧度角度换算)-CSDN博客 pcl也有自带的rad2deg,deg2rad: 头文件 #include<pcl/common/angles.h> 代码如下 #include <iostream> #include<pcl/common/angles.h> int main() {vector<…

Linux编程4.3 网络编程-数据封装

1、数据封装 2、Internet协议&#xff08;IP&#xff09; IP的主要目的是为数据输入/输出网络提供基本算法&#xff0c;为高层协议提供无连接的传送服务。这意味着在IP将数据递交给接收站点以前不在传输站点和接收站点之间建立对话&#xff08;虚拟链路&#xff09;。它只是封…

「哈哥赠书活动 - 50期」-『AI赋能写作:AI大模型高效写作一本通』

⭐️ 赠书 - 《AI赋能写作&#xff1a;AI大模型高效写作一本通》 ⭐️ 内容简介 本书以ChatGPT为科技行业带来的颠覆性革新为起点&#xff0c;深入探讨了人工智能大模型如何为我们的创作提供强大支持。本书旨在帮助创作者更好地理解AI的价值&#xff0c;并充分利用其能力提升写…

ubuntu安装开源汇编调试器NASM

安装 安装很简单&#xff0c;直接在终端输入以下命令即可 sudo apt-get install nasm 安装完成后&#xff0c;如果可以查看到nasm的版本号即可视为安装成功 nasm -version 测试 创建汇编文件 创建一个asm文件 vim hello.asm 文件内容如下 section .datahello: db …

《ElementPlus 与 ElementUI 差异集合》el-button 属性 type=“text“ 被删除

差异 element-ui el-button中&#xff0c;属性 type"text" 定义文字按钮&#xff0c;也是链接按钮&#xff1b;element-plus el-button中&#xff0c;改为新增属性 link 并与其它 type 值配合使用&#xff1b; // element-ui <el-button type"text"&g…

网络流量监控软件AnaTraf:优化性能、排除故障的最佳选择

目录 导言 网络流量监控的重要性 AnaTraf网络万用表的功能与优势 网络故障排除与优化网络性能 结论 导言 在当今数字化时代&#xff0c;计算机网络已经成为企业和组织的核心基础设施。然而&#xff0c;网络流量的管理和监控对于确保网络性能的稳定和优化至关重要。本文将介…

商业模式的定义及其成功的四大特点

商业模式&#xff0c;作为企业运营和发展的核心架构&#xff0c;描述了企业如何创造价值、传递价值和获取价值的基本逻辑和方法。简单来说&#xff0c;商业模式就是企业为了实现其市场定位、满足客户需求、实现盈利目标而采取的一系列经营策略和行动的总和。 一个成功的商业模式…

【Linux】进程控制与进程调度

Linux进程介绍 进程的基本概念 Linux是多用户、多任务的操作系统。在这样的环境中&#xff0c;各种计算机资源的分配和管理都是以进程为单位进行的。 Linux操作系统包括三种不同类型的进程&#xff1a; 1&#xff09;交互进程&#xff1a;一种由Shell启动的进程。交互进程既可…

Java项目:基于springboot实现的OA协同办公系统(源码+数据库+毕业论文)

一、项目简介 本项目是一套基于Springbootvue实现的付费自习室系统 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界面美观、操作简单、…