如何读取DXF格式文件Word格式.docx
《如何读取DXF格式文件Word格式.docx》由会员分享,可在线阅读,更多相关《如何读取DXF格式文件Word格式.docx(16页珍藏版)》请在冰豆网上搜索。
申请符号表(APPID)
3.
块段(BLOCKS)
这一段含有块定义实体,这些实体描述了图形种组成每个块的实体。
4.
实体段(ENTITIES
这一段含有实体,包括任何块的调用。
5.
END
OF
FILE(文件结束)
下面是对DXF的基本结构举一实例进行说明:
0
0
后接SECTION
SECTION
表明这是一个段的开始
2
2
后接的是段名
HEADER
说明该段是HEADER
段(标题段)
9
$ACADVER
文件是由AUTOCAD
产生的
1
AC1008
9
后接
$UCSORG
$UCSORG
用户坐标系原点在世界坐标系中的坐标
10
10
对应
X
0.0
的值
20
20
Y
30
30
Z
$UCSXDIR
这是一段不太相关的部分,略去
1.0
...
....
$EXTMIN
$EXTMIN
说明三维实体模型在世界坐标系中的最小值
-163.925293
-18.5415860.0
78.350945
$EXTMAN
$EXTMAX
说明三维实体模型在世界坐标系中的最大值
202.492279
112.634300
169.945602
ENDSEC
ENDSEC
说明这一段结束了
TABLES
说明该段是TABLES
段(表段)
...
该段对我们不太相关,此处略去不进行说明
ENTITIES
说明该段是ENTITIES
段(实体段)这是我
们要详细说明的段,该段包含了所有实体的
POLYLINE
点的坐标和组成面的点序。
0后接POLYLINE
8
表明以下数据是对于一个新的实体;
OBJECT01
8后接的字符串是这个实体的名称
66
70
从66
1
到70
64
64
说明该实体是由许多小平面组成的
71
38
71
38说明该实体共有38
个点
72
72
说明该实体由72
个三角形构成
VERTEX
VERTEX
表明后面紧跟着的是实体的数据
对应X
坐标
对应Y
-17.772665
对应Z
128.929947
70
192
192
表明上面的数据信息是点的坐标
每一个从0
之间
的一小段是点的坐标
当70
后跟128
时,表明该实体的每个点的坐标数据已经记录
完了,下面紧跟着的是记录这些点是以什么样的方式组合成各
128
个三角形。
71、72、73
后面跟着的值表明某一个三角形是第二个、第
一个、第四个点构成的,点的顺序是按照记入DXF
文件的顺
序。
当某一值为负数时,则表明该点到下一点的线不要画出,
如果要画三维实体的线型图,就必须使用这一特性,否则线条
73
将会出现紊乱。
-4
后接SEQEND
表明该实体的数据已经全部记录完了
SEQEND
后接POLYLINE
表明以下又是一个新的实体
后接ENDSEC
表明这是该段的结尾
EOF
0后接EOF
表明这个DXF
文件结束了
在DXF文件中,我们最关心的是如何得到模型上各个点的坐标,并且用这些点连成许多个三用形,构成面,进而绘制出整个模型。
在DXF文件的结构中,我们已经看到,DXF文件先叙述实体上各个点的坐标,然后叙述实体上有多少个面,每个面由哪些点构成。
这样,我们至少需要2个数组来存储一个实体的信息,一个用于存储点的坐标,一个用于存储点序,我们可以把这2个数组放到一个结构中,如果模型中实体的数目不止一个是,我们就用这个结构来定义一个数组。
在本文中,我们使用
Visual
C++
6.0
来写一个读取DXF文件的小程序。
在实际应用中,模型中实体的数目以及实体中点和面的数目都是不定的,为了有效地利用内存,我们选择MFC类库中的聚合类CobArray类所创建的对象vertex,
sequence来存储和管理实体的点坐标和点序。
CObArray类是一个用来存放数组类的聚合类,它能根据要存进来的数组(或结构)多少自动进行自身大小的高速,而且这个类本身具有的成员函数使得我们对它的对象的操作更加方便、快捷,用它编的程序也易于读懂。
三维实体模型的模型信息中的一部分信息可以在标题段中读出,通过读取变量名为$UCSORG的三个变量,可以得到三维实体在世界坐标系中自身所定义的用户坐标系原点的三维坐标。
通过读取$EXTMAX,$EXTMIN可以获知三维实体在世界坐标系中的范围,而其它部分的信息只有读完了全部DXF文件后才可以通过计算确定。
对于三维实体模型的全部点坐标、点序,可以在实体段中按照前面介绍的DXF文件基本结构读出。
现在我们开始写这个程序。
先建立一个头文件HEAD.H定义如下的结构:
VERTEX,
SEQUENCE和类CVertex,
Csequence。
typedef
struct
{
float
x,y,z;
}VERTEX;
结构VERTEX用来存储点的坐标
int
a,b,c;
}SEQUENCE;
结构SEQUENCE用来存储实体的面的组成
char
obName[20];
定义结构myVertex来存储实体的名字,点的坐标以及面的组成,
CObArray
Vertex;
其中,点的坐标和面的组成是由聚合类CObArray定义的对象来
Sequence;
在存储的,我们可以把VERTEX结构和SEQUENCE结构加入到
}myVertex;
这两个对象中保存
class
CVertex
:
public
CObject
{
因为CObArray类的对象中只能加入由CObject派生的对象,所以
protected:
我们还需要建立一个由CObject类派生的CVertex类。
在CVertex类
CVertex();
中有一个VERTEX结构的变量:
m_vertex,信息实际上是存储在这
DECLARE_DYNCREATE(CVertex)
个变量中的。
virtual
~CVertex();
//
Attributes
public:
CVertex(VERTEX&
ver);
m_vertex,信息实际上是存储在这个变量中的,函数CVertex(VERTEX&
ver)把VERTEX结构的变量
m_vertex;
存入CObArray对象中。
};
CSequence
这也是一个由CObject类派生的类,作用和刚才CVertex类一样,
只不过Csequence类是用来存储实体中面的组成(点序)的。
CSequence();
DECLARE_DYNCREATE(CSequence)
~CSequence();
CSequence(SEQUENCE&
sequ);
SEQUENCE
m_sequence;
声明好结构与类后,我们还需要建立一个.CPP文件,来定义几个函数。
IMPLEMENT_DYNCREATE(CVertex,CObject)
CVertex:
:
CVertex()
}
~CVertex()
构造函数和销毁函数都是空的
CVertex(VERTEX&
ver)
这个函数的作用是:
把一个VERTEX结构的数据存入变量m_vertex中
m_vertex
=
ver;
它是这个类中最重要的一环。
IMPLEMENT_DYNCREATE(CSequence,CObject)
CSequence:
CSequence()
Csequence类的定义与CVertex类的定义差不多,只是其中的参数
m_sequence的类型和CVertex类中的参数my_vertex的类型不一样
~CSequence()
CSequence(SEQUENCE&
sequ)
m_sequence=sequ;
然后用结构myVertex(如前所定义)定义一个指针*myData,目的在于根据模型中实体的多少来给指针分配合适的内存,使之成为结构数组。
定义一个函数,用于确定模型中有多少个实体,函数的返回值就是实体的个数。
CJupiterView:
getObjectNumber()
str1[10],str2[10];
name[]="
theFirst"
;
num;
num=0;
FILE*
fp;
fp=fopen("
data.dxf"
"
r"
);
打开DXF文件,data.dxf
while(!
feof(fp)
&
&
!
ferror(fp))
这个函数是根据实体的名字来判断实体的个数的
所以函数只读取实体的名字,一旦出现新的实体名字,
fscanf(fp,"
%s\n"
str1);
实体数就加一。
if(strcmp(str1,"
VERTEX"
)==0)
str2);
str2)
;
这个函数是根据实体的名字来判断实体的个数的
if(strcmp(name,str2)
=
0)
{实体数就加一。
strcpy(name,str2);
num++;
fclose(fp);
return
以下是读取实体点的坐标以及点序的程序代码,在这个程序中,读取了模型中点的坐标的最大值与最小值、实体的名字、点的坐标,以及点序。
void
OnFileInput()
TODO:
Add
your
command
handler
code
here
fp,*fp2;
i,k,j;
tempX,tempY,tempZ;
xMin,yMin,zMin,xMax,yMax,zMax,Max;
lab;
str1[20],str2[20],str[20],HT;
myName[20];
myNumber;
tempVertex;
tempSequence;
x,y,z,max;