dxf格式说明.docx
《dxf格式说明.docx》由会员分享,可在线阅读,更多相关《dxf格式说明.docx(19页珍藏版)》请在冰豆网上搜索。
dxf格式说明
建议你看DXF底格式介绍
读取DXF格式文件
OpenGL是美国SGI公司最新推出的一套开放式的三维图形软件接口,适用于广泛的计算机环境,从个人计算机到工作站,OpenGL都能实现高性能的三维图形功能。
OpenGL本身不仅提供对简单图元的操作和控制,还提供了许多函数用于复杂物体的建模。
但是,我们通常喜欢使用AutoCAD和3DS及3Dmax等工具来建立模型,并且我们已经有了很多这样的模型,那么我们如何才能资源共享,避免重复劳动呢?
利用CAD图形标准数据交换格式—DXF格式,我们就能很容易地实现资源共享,而不需要重复建模。
DXF文件的结构很清楚,具体如下:
1.标题段(HEADER)
有关图形的一般信息都可以DXF文件的这一节找到,每一个参数具有一个变量名和一个相关值。
2.表段
这一段包含的指定项的定义,它包括:
a、
线形表(LTYPE)
b、
层表(LYER)
c、
字体表(STYLE)
d、
视图表(VIEW)
e、
用户坐标系统表(UCS)
f、
视窗配置表(VPORT)
g、
标注字体表(DIMSTYLE)
h、
申请符号表(APPID)
3.块段(BLOCKS)
这一段含有块定义实体,这些实体描述了图形种组成每个块的实体。
4.实体段(ENTITIES)
这一段含有实体,包括任何块的调用。
5.ENDOFFILE(文件结束)
下面是对DXF的基本结构举一实例进行说明:
00后接SECTION
SECTION表明这是一个段的开始
22后接的是段名
HEADER说明该段是HEADER段(标题段)
9
$ACADVER文件是由AUTOCAD产生的
1
AC1008
99后接$UCSORG
$UCSORG用户坐标系原点在世界坐标系中的坐标
1010对应X
0.0X的值
2020对应Y
0.0Y的值
3030对应Z
0.0Z的值
9
$UCSXDIR这是一段不太相关的部分,略去
10
1.0
.......
99后接$EXTMIN
$EXTMIN说明三维实体模型在世界坐标系中的最小值
1010对应X
-163.925293X的值
2020对应Y
-18.5415860.0Y的值
3030对应Z
78.350945Z的值
99后接$EXTMAN
$EXTMAX说明三维实体模型在世界坐标系中的最大值
1010对应X
202.492279X的值
2020对应Y
112.634300Y的值
3030对应Z
169.945602Z的值
00后接ENDSEC
ENDSEC说明这一段结束了
00后接SECTION
SECTION表明这是一个段的开始
22后接的是段名
TABLES说明该段是TABLES段(表段)
............该段对我们不太相关,此处略去不进行说明
00后接ENDSEC
ENDSEC说明这一段结束了
00后接SECTION
SECTION表明这是一个段的开始
22后接的是段名
ENTITIES说明该段是ENTITIES段(实体段)这是我
0们要详细说明的段,该段包含了所有实体的
POLYLINE点的坐标和组成面的点序。
0后接POLYLINE
8表明以下数据是对于一个新的实体;
OBJECT018后接的字符串是这个实体的名称
66
1
70从661到7064
64说明该实体是由许多小平面组成的
71
387138说明该实体共有38个点
72
727272说明该实体由72个三角形构成
00VERTEX
VERTEX表明后面紧跟着的是实体的数据
8
OBJECT01
10对应X坐标
-163.925293X的值
20对应Y坐标
-17.772665Y的值
30对应Z坐标
128.929947Z的值
7070192
192表明上面的数据信息是点的坐标
0每一个从0VERTEX到70192之间
VERTEX的一小段是点的坐标
.........
70
192
0
VERTEX
8
OBJECT01
10
0
20
0
30
0当70后跟128时,表明该实体的每个点的坐标数据已经记录
70完了,下面紧跟着的是记录这些点是以什么样的方式组合成各
128个三角形。
7171、72、73后面跟着的值表明某一个三角形是第二个、第
2一个、第四个点构成的,点的顺序是按照记入DXF文件的顺
72序。
当某一值为负数时,则表明该点到下一点的线不要画出,
1如果要画三维实体的线型图,就必须使用这一特性,否则线条
73将会出现紊乱。
-4
0
VERTEX
............
00后接SEQEND表明该实体的数据已经全部记录完了
SEQEND
8
OBJECT01
0
POLYLINE0后接POLYLINE表明以下又是一个新的实体
............
0
ENDSEC0后接ENDSEC表明这是该段的结尾
0
EOF0后接EOF表明这个DXF文件结束了
在DXF文件中,我们最关心的是如何得到模型上各个点的坐标,并且用这些点连成许多个三用形,构成面,进而绘制出整个模型。
在DXF文件的结构中,我们已经看到,DXF文件先叙述实体上各个点的坐标,然后叙述实体上有多少个面,每个面由哪些点构成。
这样,我们至少需要2个数组来存储一个实体的信息,一个用于存储点的坐标,一个用于存储点序,我们可以把这2个数组放到一个结构中,如果模型中实体的数目不止一个是,我们就用这个结构来定义一个数组。
在本文中,我们使用VisualC++6.0来写一个读取DXF文件的小程序。
在实际应用中,模型中实体的数目以及实体中点和面的数目都是不定的,为了有效地利用内存,我们选择MFC类库中的聚合类CobArray类所创建的对象vertex,sequence来存储和管理实体的点坐标和点序。
CObArray类是一个用来存放数组类的聚合类,它能根据要存进来的数组(或结构)多少自动进行自身大小的高速,而且这个类本身具有的成员函数使得我们对它的对象的操作更加方便、快捷,用它编的程序也易于读懂。
三维实体模型的模型信息中的一部分信息可以在标题段中读出,通过读取变量名为$UCSORG的三个变量,可以得到三维实体在世界坐标系中自身所定义的用户坐标系原点的三维坐标。
通过读取$EXTMAX,$EXTMIN可以获知三维实体在世界坐标系中的范围,而其它部分的信息只有读完了全部DXF文件后才可以通过计算确定。
对于三维实体模型的全部点坐标、点序,可以在实体段中按照前面介绍的DXF文件基本结构读出。
现在我们开始写这个程序。
先建立一个头文件HEAD.H定义如下的结构:
VERTEX,SEQUENCE和类CVertex,Csequence。
typedefstruct{
floatx,y,z;
}VERTEX;结构VERTEX用来存储点的坐标
typedefstruct{
inta,b,c;
}SEQUENCE;结构SEQUENCE用来存储实体的面的组成
typedefstruct{
charobName[20];定义结构myVertex来存储实体的名字,点的坐标以及面的组成,
CObArrayVertex;其中,点的坐标和面的组成是由聚合类CObArray定义的对象来
CObArraySequence;在存储的,我们可以把VERTEX结构和SEQUENCE结构加入到
}myVertex;这两个对象中保存
classCVertex:
publicCObject
{因为CObArray类的对象中只能加入由CObject派生的对象,所以
protected:
我们还需要建立一个由CObject类派生的CVertex类。
在CVertex类
CVertex();中有一个VERTEX结构的变量:
m_vertex,信息实际上是存储在这
DECLARE_DYNCREATE(CVertex)个变量中的。
virtual~CVertex();
//Attributes
public:
我们还需要建立一个由CObject类派生的CVertex类。
在CVertex类
CVertex(VERTEX&ver);中有一个VERTEX结构的变量:
m_vertex,信息实际上是存储在这个变量中的,函数CVertex(VERTEX&ver)把VERTEX结构的变量
VERTEXm_vertex;存入CObArray对象中。
};
classCSequence:
publicCObject
{这也是一个由CObject类派生的类,作用和刚才CVertex类一样,
protected:
只不过Csequence类是用来存储实体中面的组成(点序)的。
CSequence();
DECLARE_DYNCREATE(CSequence)
virtual~CSequence();
public:
CSequence(SEQUENCE&sequ);
SEQUENCEm_sequence;
};
声明好结构与类后,我们还需要建立一个.CPP文件,来定义几个函数。
IMPLEMENT_DYNCREATE(CVertex,CObject)
CVertex:
:
CVertex()
{
}
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()
{
}
CSequence:
:
CSequence(SEQUENCE&sequ)
{
m_sequence=sequ;
}
然后用结构myVertex(如前所定义)定义一个指针*myData,目的在于根据模型中实体的多少来给指针分配合适的内存,使之成为结构数组。
定义一个函数,用于确定模型中有多少个实体,函数的返回值就是实体的个数。
intCJupiterView:
:
getObjectNumber()
{
charstr1[10],str2[10];
charname[]="theFirst";
intnum;
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)
{
fscanf(fp,"%s\n",str2);打开DXF文件,data.dxf
fscanf(fp,"%s\n",str2);这个函数是根据实体的名字来判断实体的个数的
if(strcmp(name,str2)!
=0)所以函数只读取实体的名字,一旦出现新的实体名字,
{实体数就加一。
strcpy(name,str2);
num++;
}
}
}
fclose(fp);
returnnum;
}
以下是读取实体点的坐标以及点序的程序代码,在这个程序中,读取了模型中点的坐标的最大值与最小值、实体的名字、点的坐标,以及点序。
voidCJupiterView:
:
OnFileInput()
{
//TODO:
Addyourcommandhandlercodehere
FILE*fp,*fp2;
inti,k,j;
floattempX,tempY,tempZ;
floatxMin,yMin,zMin,xMax,yMax,zMax,Max;
intlab;
charstr1[20],str2[20],str[20],HT;
charmyName[20];
intmyNumber;
VERTEXtempVertex;
SEQUENCEtempSequence;
typedefstruct{
floatx,y,z,max;
}MAX;
MAXmax;
HT=9;
objectNumber=getObjectNumber();
myData=newmyVertex[objectNumber];
fp=fopen(FileName,"r");
i=0;
j=0;
k=0;
myNumber=-1;
strcpy(myName,"ObjectName");
while(!
feof(fp)&&!
ferror(fp))
{
fscanf(fp,"%s\n",str);
if(strcmp(str,"$EXTMIN")==0)
{
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&xMin);
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&yMin);
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&zMin);
}
if(strcmp(str,"$EXTMAX")==0)
{
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&xMax);
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&yMax);
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&zMax);
max.x=max(abs(xMax),abs(xMin));
max.y=max(abs(yMax),abs(yMin));
max.z=max(abs(zMax),abs(zMin));
max.max=max(max.x,max.y);
max.max=max(max.max,max.z);
}
if(strcmp(str,"VERTEX")==0)
{
fscanf(fp,"%s\n",str1);
fscanf(fp,"%s\n",str1);
if(strcmp(myName,str1)!
=0)
{
myNumber++;
strcpy(myName,str1);
strcpy((myData+myNumber)->obName,myName);
}
fscanf(fp,"%s\n",str2);
fscanf(fp,"%f\n",&tempX);
fscanf(fp,"%s\n",str2);
fscanf(fp,"%f\n",&tempY);
fscanf(fp,"%s\n",str2);
fscanf(fp,"%f\n",&tempZ);
fscanf(fp,"%d\n",&lab);
fscanf(fp,"%d\n",&lab);
if(lab==192)
{
tempVertex.x=tempX/max.max;
tempVertex.y=tempY/max.max;
tempVertex.z=tempZ/max.max;
(myData+myNumber)->Vertex.Add(newCVertex(tempVertex));
}
if(lab==128)
{
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&tempX);
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&tempY);
fscanf(fp,"%s\n",str1);
fscanf(fp,"%f\n",&tempZ);
tempSequence.a=abs(tempX);
tempSequence.b=abs(tempY);
tempSequence.c=abs(tempZ);
(myData+myNumber)->Sequence.Add(newCSequence(tempSequence));
}
}
}
fclose(fp);
}
解析DXF图形文件格式
作者:
lywhust提交日期:
2006-9-1512:
02:
00
一、DXF文件格式分析
DXF文件由标题段、表段、块段、实体段和文件结束段5部分组成,其内容如下。
☆标题段(HEADER)标题段记录AutoCAD系统的所有标题变量的当前值或当前状态。
标题变量记录了AutoCAD系统的当前工作环境,如SNAP捕捉当前状态、栅格间距式样、当前图层层名及线型、颜色等。
☆表段(TABLES)表段共包含4个表,每个表又包含可变数目的表项。
这些表在文件中出现的顺序是线型表(LTYPE)、图层表(LAYER)、字样表(STYLE)、视图表(VIEW)。
☆块段(BLOCK)块段记录了所用块的块名,当前图层层名、块的种类、块的插入基点及组成该块的所有成员。
块的种类分为图形块、带有属性的块和无名块三种。
无名块包括用HATCH命令生成的剖面线和用DIM命令所完成的尺寸标准。
☆实体段(ENTITIES)实体段记录了每个实体的名称、所在图层及其名字、线型、颜色等。
☆文件结束段(EOFOFFILE)DXF文件的结束标志。
一个DXF文件由若干个组构成,每个组占两行,第一行为组的代码,第二行为组值。
组代码相当于数据类型的代码,它由CAD图形系统所规定,而组值为具体的数值,二者结合起来表示一个数据的含义和值。
例如,代码10代表一个点的X坐标,占一行,而其第二行4.5425则是点X坐标的具体数值,二者结合表示一点,其X坐标值为4.5425。
(1)组代码和组值的类型组代码为一个非负的不超过三位的整数,而组值由组代码的类型决定。
例如:
代码0~9组值类型为字符型。
代码10~59组值类型为实型。
代码60~79组值类型为整型。
代码999表示解释行。
(2)组代码的含义每个组代码均有规定的含义,有些代码含义是固定的,而有些组代码则因应用场合不同而有多个含义,应具体分析。
另外,一些代码是备用的,目前版本尚未用到,现将他们的含义举例介绍如下。
0:
表示一个事物的开始,如一个块、表、图层、实体等。
1:
字符型数据的值,如TEXT的字符串、文件名、属性值等。
2:
一个事物的名字,如段、表、块、线型、视图等的名字。
3~5:
字符型数据的值,如文件名、线型说明等。
6:
线型名(固定类型)。
8:
图层名(固定类型)。
关于实体的坐标与相应的组代码10~18、20~28、30~38的用法应根据实体所用到点的数量,按组代码个位的0、1、2、……的顺序使用。
例如LINE的起点组代码为10、20、30,而11、21、31为其终点。
其他类似。
DXF文件的结构相当复杂,完整读取DXF文件也是一项异常繁琐的工程。
在实际应用中,为了提取图形的实体信息,可以省略DXF文件中的许多项,只要获取其中的层表、块段和实体段,就可以完成相应几何图形的描述。
在层表中说明每一层的颜色、线型,在块段中说明块所在的层、属性及其在图形中的位置,在实体段中说明直线的起点、终点及圆的圆心、半径等几何信息和各实体所在的层。
根据实体所在的层,在层表中搜索每一层的颜色、线型并将其添加到实体对象中。
在利用CAD进行绘图时,需将变量参数DIMASO设为ON,以保证块段中定义的尺寸块为一个整体,并可被正确无误地获取。
根据尺寸类型名、尺寸定义的起点、终点坐标与实体类型名、实体空间坐标是否匹配,对实体对象进行尺寸附加。
在绘图时,如对自定义的粗糙度、形位公差等块不进行“EXPLODE”炸开操作,也可通过增加相应读取函数提取其属性。
下面举例说明实体段在DXF文件中的格式。
圆弧(ARC)
0(开始)ARC(圆弧)
8(层的组码)
W1(层名为“W1”)
10(圆弧圆心的X坐标组码)
7.0(圆弧圆心的X坐标组值)
20(圆弧圆心的Y坐标组码)
9.0(圆弧圆心的Y坐标组值)
30(圆弧圆心的Z坐标组码)
0.0(圆弧圆心的Z坐标组值)
40(圆弧的半径组码)
25.0(圆弧的半径值)
50(圆弧的起始角度组码)
90.0(圆弧的起始角度值)
51(圆弧的终止角度组码)
180.0(圆弧的终止角度组值)
若实体的线型用BYLAYER,颜色为256(即BYLAYER),基面高度和厚度为零,其格式可简化。
对于其他图形实体的格式描述(LINE、CIRCLE、TEXT等),这里就不再一一详述了。
二、几何图形实体信息的提取
生成DXF文件必须首先进入图形编辑环境,在命令(Command)提示下键入DXFOUT,回答所需的一个文件名后,即把内部压缩格式的图形数据转换为ASCⅡ的数据并写入相应DXF文件(以DXF为扩展名)中,供应用程序提取并处理实体数据。
三、几何图形描述的数据结构
通过对CAD系统的DXF图形交换文件进行分析,可以得到描述零件的全部几何图形元素。
为简化问题,设定组成零件内外轮廓的图形元素只包括点、直线和圆弧,并分别用下列数据结构表示。
点:
classPOINT{floatx;floaty;floatz;}虚基类:
classSHAPE{//由虚基类派生出直线、圆弧,便于链表操作public:
[replyview]
virtualvoidshowEntity()=0;//表示此函数在派生类中必须定义……//在此定义其他虚函数}直线:
classLINE:
publicSHAPE,publicPOINT{public:
//直线类由两个基类SHAPE和POINT派生而来POINTstart_point;//直线起点POINTend_point;//直线终点……//在此定义其他内部变量和函数}圆弧:
classARC:
publicSHAPE,publicPOINT{public:
//圆弧类由两个基类SHAPE和POIN