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. END OF FILE(文件结束)

下面是对DXF的基本结构举一实例进行说明:

0 0 后接SECTION

SECTION 表明这是一个段的开始

2 2 后接的是段名

HEADER 说明该段是HEADER 段(标题段)

9

$ACADVER 文件是由AUTOCAD 产生的

1

AC1008

9 9 后接 $UCSORG

$UCSORG 用户坐标系原点在世界坐标系中的坐标

10 10 对应 X

0.0 X 的值

20 20 对应 Y

0.0 Y 的值

30 30 对应 Z

0.0 Z 的值

9

$UCSXDIR 这是一段不太相关的部分,略去

10

1.0

... ....

9 9 后接 $EXTMIN

$EXTMIN 说明三维实体模型在世界坐标系中的最小值

10 10 对应 X

-163.925293 X 的值

20 20 对应 Y

-18.5415860.0 Y 的值

30 30 对应 Z

78.350945 Z 的值

9 9 后接 $EXTMAN

$EXTMAX 说明三维实体模型在世界坐标系中的最大值

10 10 对应 X

202.492279 X 的值

20 20 对应 Y

112.634300 Y 的值

30 30 对应 Z

169.945602 Z 的值

0 0 后接 ENDSEC

ENDSEC 说明这一段结束了

0 0 后接SECTION

SECTION 表明这是一个段的开始

2 2 后接的是段名

TABLES 说明该段是TABLES 段(表段)

... ... ... ... 该段对我们不太相关,此处略去不进行说明

0 0 后接 ENDSEC

ENDSEC 说明这一段结束了

0 0 后接SECTION

SECTION 表明这是一个段的开始

2 2 后接的是段名

ENTITIES 说明该段是ENTITIES 段(实体段)这是我

0 们要详细说明的段,该段包含了所有实体的

POLYLINE 点的坐标和组成面的点序。0后接POLYLINE

8 表明以下数据是对于一个新的实体;

OBJECT01 8后接的字符串是这个实体的名称

66

1

70 从66 1 到70 64

64 说明该实体是由许多小平面组成的

71

38 71 38说明该实体共有38 个点

72

72 72 72 说明该实体由72 个三角形构成

0 0 VERTEX

VERTEX 表明后面紧跟着的是实体的数据

8

OBJECT01

10 对应X 坐标

-163.925293 X 的值

20 对应Y 坐标

-17.772665 Y 的值

30 对应Z 坐标

128.929947 Z 的值

70 70 192

192 表明上面的数据信息是点的坐标

0 每一个从0 VERTEX 到70 192 之间

VERTEX 的一小段是点的坐标

... ... ...

70

192

0

VERTEX

8

OBJECT01

10

0

20

0

30

0 当70 后跟128 时,表明该实体的每个点的坐标数据已经记录

70 完了,下面紧跟着的是记录这些点是以什么样的方式组合成各

128 个三角形。

71 71、72、73 后面跟着的值表明某一个三角形是第二个、第

2 一个、第四个点构成的,点的顺序是按照记入DXF 文件的顺

72 序。当某一值为负数时,则表明该点到下一点的线不要画出,

1 如果要画三维实体的线型图,就必须使用这一特性,否则线条

73 将会出现紊乱。

-4

0

VERTEX

... ... ... ...

0 0 后接SEQEND 表明该实体的数据已经全部记录完了

SEQEND

8

OBJECT01

0

POLYLINE 0 后接POLYLINE 表明以下又是一个新的实体

... ... ... ...

0

ENDSEC 0 后接ENDSEC 表明这是该段的结尾

0

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用来存储点的坐标

typedef struct {

int a,b,c;

}SEQUENCE; 结构SEQUENCE用来存储实体的面的组成

typedef struct {

char obName[20]; 定义结构myVertex来存储实体的名字,点的坐标以及面的组成,

CObArray Vertex; 其中,点的坐标和面的组成是由聚合类CObArray定义的对象来

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: 我们还需要建立一个由CObject类派生的CVertex类。在CVertex类

CVertex(VERTEX& ver);

中有一个VERTEX结构的变量:m_vertex,信息实际上是存储在这个变量中的,函数CVertex(VERTEX&

ver)把VERTEX结构的变量

VERTEX m_vertex; 存入CObArray对象中。

};

class CSequence : public CObject

{ 这也是一个由CObject类派生的类,作用和刚才CVertex类一样,

protected: 只不过Csequence类是用来存储实体中面的组成(点序)的。

CSequence();

DECLARE_DYNCREATE(CSequence)

virtual ~CSequence();

public:

CSequence(SEQUENCE& sequ);

SEQUENCE m_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,目的在于根据模型中实体的多少来给指针分配合适的内存,使之成为结构数组。

定义一个函数,用于确定模型中有多少个实体,函数的返回值就是实体的个数。

int CJupiterView::getObjectNumber()

{

char str1[10],str2[10];

char name[]="theFirst";

int 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)

{

fscanf(fp,"%s/n",str2); 打开DXF文件,data.dxf

fscanf(fp,"%s/n",str2) ;这个函数是根据实体的名字来判断实体的个数的

if(strcmp(name,str2) != 0) 所以函数只读取实体的名字,一旦出现新的实体名字,

{实体数就加一。

strcpy(name,str2);

num++;

}

}

}

fclose(fp);

return num;

}

以下是读取实体点的坐标以及点序的程序代码,在这个程序中,读取了模型中点的坐标的最大值与最小值、实体的名字、点的坐标,以及点序。

void CJupiterView::OnFileInput()

{

// TODO: Add your command handler code here

FILE* fp,*fp2;

int i,k,j;

float tempX,tempY,tempZ;

float xMin,yMin,zMin,xMax,yMax,zMax,Max;

int lab;

char str1[20],str2[20],str[20],HT;

char myName[20];

int myNumber;

VERTEX tempVertex;

SEQUENCE tempSequence;

typedef struct {

float x,y,z,max;

}MAX;

MAX max;

HT=9;

objectNumber=getObjectNumber();

myData=new myVertex[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(new CVertex(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(new CSequence(tempSequence));

}

}

}

fclose(fp);

}

0

0

java read dxf xdata_dxf结构解析及读取坐标 | 学步园相关推荐

  1. java调用存储过程sqlserver_Java调用SqlServer存储过程怎么实现 | 学步园

    在使用Java开发时,经常会遇到调用SqlServer存储过程的问题.下面学步园小编来讲解下Java调用SqlServer存储过程怎么实现? Java调用SqlServer存储过程怎么实现 1.数据库 ...

  2. java 负数存储结构_负数在java中的存储和读取过程 | 学步园

    问题描述: 将-5存储在文本文件中,再读取出来显示到控制台; 预备知识: 1.在java中使用补码处理数字,而且byte(8)的数字在扩展成int(32)类型的时候,正数填充0,负数填充1; 2.负数 ...

  3. Java实现树状结构解析

    第一次实现了树状结构数据 记录自己的第一次 话不多说 直接上代码 数据库信息 代码实现 实体类 public class ChainTree implements Serializable {priv ...

  4. java 命令行读取_Java:从控制台(console,命令行)读取字符 | 学步园

    /** * 从控制台(console,命令行)读取字符:InputStreamReader(System.in) */ import java.io.IOException; import java. ...

  5. java调用dll 指针参数_java调用c dll,指针参数和结构体参数搞定 | 学步园

    终于有了比较大的成果,不过身体也累完了,昨天又去打的吊瓶,坐成铁的时候差点就晕倒在里面了,不知道什么原因,抵抗力这么差,明天放假,要在家好好休息,今天早上碰到欢欢了,她说给我做好吃的,我愿意吃她做的糖 ...

  6. java tlv解析_java解析TLV格式数据 | 学步园

    TLV:TLV格式数据是指由Tag,Length,Value组成的数据.具体说明如下: tag标签的属性为bit,由16进制表示,占1-2个字节长度.例如,"9F33"为一个占用两 ...

  7. java 嵌套对象转xml_Gson对Java嵌套对象和JSON字符串之间的转换 | 学步园

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,具有良好的跨平台特性.近几年来已经和XML一样成为C/S架构中广泛采用的数据格式.有关JSON的更多知识, ...

  8. java tail -f 后返回_tail -f 的实现 | 学步园

    最原始的想法和实现 最容易想到的就是不断的读取一个文件,如果读取到文件结尾(EOF),那么sleep一下然后再次尝试. 事实上Apache Common IO里就有一个这样的实现 这种方法到优点是简单 ...

  9. java axis2 调用webservice 接口_Axis2 调用Webservice 接口 | 学步园

    调用方法: TranslatorString  输入中文,翻译成 拼音.英文. 参数:wordKey(中文) 现在要做,翻译词:[随便],代码如下: package cn.com.webxml; im ...

  10. java card applet_可多选的javacard applet | 学步园

    可多选的javacard applet,与多个逻辑通道上设置各自不同的applet是有区别的. Java Card 2.2支持逻辑通道(logical channels)的概念,允许最多智能卡中的16 ...

最新文章

  1. 数据库运维平台~开源成熟项目
  2. python怎么用excel-Python使用xlwt模块操作Excel的方法详解
  3. Android Studio 约束布局[ConstraintLayout]
  4. svn checkout 提示“由于连接方在一段时间后没有正确答复或连接的主机没有反应,连接尝试失败。”解决方法...
  5. 《手把手教你》系列基础篇之4-python+ selenium自动化测试-xpath使用(详细教程)
  6. skip-grant-tables:非常有用的mysql启动参数
  7. HDFS 读/写数据流程
  8. FutureTask的get()方法之异常处理
  9. 人力资源管理之项目团队建设
  10. 一篇了解TrustZone
  11. 公司内部分享【富有成效的每日站会】总结
  12. openlayers6 解决调用百度地图之瓦片偏移、坐标偏移、无限拖动裂缝偏移问题
  13. 计算机学win7画图,Windows7电脑基础使用画图程序画一个小鸭
  14. ipa图片解密 php,ipa文件详解 - 如何提取App图片
  15. 献给在这个世界上摇摆不定的朋友们
  16. VS2015打开编译VS2013工程时提示fatal error C1083: 无法打开包括文件: “afxwin.h”: No such file or directory
  17. 思科模拟器:修改根交换机
  18. 3d怎么把两个面拼接在一起_720全景视频接缝拼接教程
  19. 大连理工学php,大连理工大学教务处(大连理工大学国内交换生)
  20. 【数据分析数据源】全国各省市行政区坐标(包含边界坐标点和中心坐标点)

热门文章

  1. 物理地址通过什么协议转换为ip地址
  2. 用python预测小孩的身高_Python 孩子身高预测
  3. Linux目录及常用命令英文全称与中文解释(实用干货)
  4. C/S与P2P的主要区别以及相同点
  5. 程序员裸辞2个月找不到工作,心态爆炸了
  6. win7 计算机休眠,WIN7如何关闭睡眠和休眠方式(真正的)
  7. 地震观测仪器的历史和发展趋势(三)
  8. lucas–kanade_异常检测常用光流法量化对比:Farneback/Horn-Schunck / Lucas–Kanade
  9. 文献阅读——梅州佛教香花的结构、文本与变体
  10. 买菜总结之三--水果篇