块表记录是包裹实体对象的最后一层包装了,接下来让我们继续利用MgdDbg工具查看上一小节创建的块定义内的对象有哪些。

操作步骤如下:选择块表记录TestBlock,在右侧列表中找到“Entities within block”,选中该行并单击鼠标左键,弹出Snoop Objects对话框,列表中显示出了创建块定义时选择的两个对象:圆和直线。(下图中BED、BEE分别为实体对象的句柄(Handle))。

图 4‑10 TestBlock块表记录中的实体对象

如果刚才创建块时,您选择了删除所选对象,此时模型空间应该是不存在实体对象的,如果您选择*Model_Space块表记录,可以发现Entities within block为常规字体,没有加粗,点击时不会弹出对话框。

如果你选择的是保留所选对象,此时模型空间是存在一个圆和一条直线的,*Model_Space块表记录中的实体对象与TestBlock块表记录中的实体对象一致。(注意截图中选中的块表记录与上图不同)

图 4‑11 *Model_Space块表记录中的实体对象

如果你选择的是转换为块,此时模型空间中有一个块参照,*Model_Space块表记录中的实体对象为块参照。

图 4‑12 *Model_Space块表记录中的实体对象

再次回想之前我们利用命令Block创建块的过程:

操作

块表变化

块表记录变化

模型空间创建实体

无变化

*Model_Space内增加实体

创建块定义

增加块表记录TestBlock

TestBlock内增加实体

创建块时的不同选项

保留

无变化

不变

转换为块

无变化

*Model_Space中实体被删除,增加块参照

删除

无变化

*Model_Space中实体被删除

注:1.表中的实体指的是用于创建块的实体,例如之前示例中的圆和直线。

2.如果用命令Bedit来创建块,上述块表及块表记录的变化会有多不同。

接下来就让我看一下如何采用代码块表、块表记录及块表记录内部实体的查询。继续4.1.2 节的项目,在MyCommands类中添加以下代码:

 [CommandMethod("MyGroup", "ListEnts", CommandFlags.Modal)]

 public void ListEnts()

 {

     Document doc = Application.DocumentManager.MdiActiveDocument;   //活动文档

     Editor ed = doc.Editor;                                    //获取编辑器

     Database db = doc.Database;                           //获取数据库

     //Database db = HostApplicationServices.WorkingDatabase;               //另一种方法

     ObjectId blockTblId = db.BlockTableId;                          //块表Id

     using (Transaction tr = db.TransactionManager.StartTransaction())

     {

         BlockTable bt = blockTblId.GetObject(                               //块表

             OpenMode.ForRead) as BlockTable;

         foreach (ObjectId btrId in bt)                              //遍历块表记录

         {

             BlockTableRecord btr = btrId.GetObject(

                 OpenMode.ForRead) as BlockTableRecord;

             ed.WriteMessage("\n块表记录:{0}", btr.Name);    //输出块表记录名称

             foreach (ObjectId entId in btr)                                //遍历实体

             {

                 DBObject obj = entId.GetObject(OpenMode.ForRead);

                 ed.WriteMessage("\n实体类型为:{0}\t句柄:{1}",

                     obj.GetType().Name, obj.Handle);          //类型名称及句柄

             }

         }

         tr.Commit();

     }

 }

这段代码输出了每条块表记录的名称,并遍历块表记录的实体,输出实体类型与句柄。代码第6行通过活动文档获取数据库,第7行(注释掉的一行)采用另一种方法获取了当前正在操作的数据库(注意:这种方法获取的数据库并不一定是活动文档的数据库,您需要查阅更多资料);代码第8行通过数据库获得块表的Id;代码第11、12行打开块表;代码第13行针对块表中的每一个ObjectId进行循环操作,这个Id是块表记录的Id;代码第15、16行代开块表记录;代码第13行针对块表记录中的每一个ObjectId进行循环操作,这个Id是数据库对象(实体)的Id。阅读这段代码,您应该把重点放在理解对象之间层级关系上,至于如何打开对象、获取对象属性等操作并不是重点。

代码第9行中涉及到的事务(Transaction)、第11、12行涉及到的由ObjectId获取Object后续章节中详细讲述。

再次回想一下刚才的操作过程:获取文档、获取数据库、从数据库中找到块表、块表内部是块表记录、块表记录内是实体。一层又一层的包装箱,如果您能清楚的知道刚才拆了几层包装箱,说明您对数据库之间的层级关系也就明白了。

编译、加载项目,打开之前的dwg文件,输入命令ListEnts,看到的结果与下图类似;如果您是在其他图形文件,例如用_AutoCAD Civil 3D (Metric) NCS.dwt样板文件创建的图形,这里列出的块表记录会很多。

图 4‑13 代码运行结果

(注:前后几个截图中实体句柄不同是因为我并不是利用一个图形文件完成的测试)

您可能早就想到这样一个问题:如何直接获取模型空间或某个图纸空间的呢?

在MyCommands类中添加以下代码:

 [CommandMethod("MyGroup", "ListBlkRcd", CommandFlags.Modal)]

 public void ListBlkRcd()

 {

     string[] btrNames = new string[]{

             "*Model_Space","*model_space","*Paper_Space",

             "*Paper_Space0", "*Paper_Space1", "*Paper_Space2",

             "Line","TestBlock","Something",

             BlockTableRecord.ModelSpace,BlockTableRecord.PaperSpace};

     Document doc = Application.DocumentManager.MdiActiveDocument;   //活动文档

     Editor ed = doc.Editor;                                          //获取编辑器

     Database db = doc.Database;                                    //获取数据库

     //Database db = HostApplicationServices.WorkingDatabase;

     ObjectId blockTblId = db.BlockTableId;                                //块表Id

     using (Transaction tr = db.TransactionManager.StartTransaction())

     {

         BlockTable bt = blockTblId.GetObject(                             //块表

             OpenMode.ForRead) as BlockTable;

         foreach (string btrName in btrNames)                   //遍历每一名称

         {

             if (bt.Has(btrName))                                    //判断是否存在

             {

                 BlockTableRecord btr = bt[btrName].GetObject(

                     OpenMode.ForRead) as BlockTableRecord;

                 ed.WriteMessage("\n块表记录{0}找到了!\t句柄:{1}",

                     btr.Name, btr.Handle);                         //输出名称及句柄

             }

             else

             {

                 ed.WriteMessage("\n块表记录{0}不存在!", btrName);

             }

         }

         tr.Commit();

     }

 }

因为目前没有涉及到输入操作,因此代码第4-8行用字符串数组存储了若干块表记录名称,注意其大小写,代码第8行中的两个“字符串”是类BlockTableRecord中的两个静态属性,用于返回模型空间及图纸空间的名称。代码第18行对每一块表记录名称进行循环操作;代码第20-30行根据是否存在相应的块表记录输出不同的信息。

代码第22行bt[btrName]用于获取与名称btrName对应的块表记录的ObjectId,是此段代码的核心。也就是说,我们可以通过名称来获取块表记录的Id,之后就可以进行其他操作。

经过测试,名称字符串的大小写并不敏感,"*Model_Space"、"*model_space"都能顺利获取到模型空间。

图 4‑14 块表记录查询结果

图 4‑15

了解AutoCAD对象层次结构 —— 6 ——块表记录的更多相关文章

  1. 了解AutoCAD对象层次结构 —— 5 —— 块表

    为了清楚的了解块表的组成内容,让我们利用MgdDbg工具查看一下块表中的块表记录.在开始页面,以无样板模式新建一个.dwg文件(图 4‑7(1)),这样的话,默认的块表记录只有3条(图 4‑7(2)) ...

  2. 了解AutoCAD对象层次结构 —— 4 —— 符号表

    上一小节我们看到了符号表包含了一系列的表(共9个),这些表数量是固定的,用户不能增加新的表,也不能删除现有的表. 符号表名称 符号表功能 Block Table 块表 存储图形数据库中定义的块.此表中 ...

  3. 了解AutoCAD对象层次结构 —— 2 ——文档

    再次想象另外一个场景:启动AutoCAD程序后,您新建了两个.dwg文件,也就是说创建了两个文档(Document)对象.将窗口进行层叠,您看到的窗口应该与下图类似: 图 4‑3 如何访问这些文档呢? ...

  4. 了解AutoCAD对象层次结构 —— 1 ——应用程序

    想象这样一个场景:当您开始一天的工作,坐到电脑前面,用鼠标双击桌面上的AutoCAD Civil 3D图标,这时一个AutoCAD Civil 3D应用程序将运行起来.打开Windows任务管理器,我 ...

  5. 了解AutoCAD对象层次结构 —— 3 ——数据库

    数据库的结构是什么样的?对象是如何存储在数据库中的?这些问题我们需要搞明白.在此我们可以借助工具ArxDbg或MgdDbg来查看数据库结构及其内容.下图就是利用MgdDbg工具查看到的内容,我们可以看 ...

  6. Civil 3D 二次开发 创建AutoCAD对象—— 01 —— 创建直线

    在方法CreateLine内完成以下代码: 01 public void CreateLine() 02 { 03 PromptPointOptions ppo = new PromptPointOp ...

  7. 获得指定数据库中指定块表中所有实体的id

    该函数也使用外部指定图纸中的数据库中的块 Int getIdsByDwgBlkName(AcDbDatabase *pDwg, CString strBlkName, AcDbObjectIdArra ...

  8. {django模型层(二)多表操作}一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询、分组查询、F查询和Q查询

    Django基础五之django模型层(二)多表操作 本节目录 一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询.分组查询.F查询和Q查询 六 xxx 七 ...

  9. MyBatis 用户表记录数查询

    搭建MyBatis开发环境,实现用户表记录数查询 1.在MyEclipse中创建工程,导入MyBatis的jar包

随机推荐

  1. python--Numpy and Pandas 笔记01

    博客地址:http://www.cnblogs.com/yudanqu/ 1 import numpy as np import pandas as pd from pandas import Ser ...

  2. 朱晔和你聊Spring系列S1E2:SpringBoot并不神秘

    朱晔和你聊Spring系列S1E2:SpringBoot并不神秘 [编辑器丢失了所有代码的高亮,建议查看PDF格式文档] 文本我们会一步一步做一个例子来看看SpringBoot的自动配置是如何实现的, ...

  3. *args **kwargs

    -----------耐得住寂寞,守得住芳华. # # -------------------------------[day10作业及默写]----------------------------- ...

  4. python实现一个二分查找

    二分查找: 二分查找也称折半查找(Binary Search),它是一种效率较高的查找方法.但是,折半查找要求线性表必须采用顺序存储结构,而且表中元素按关键字有序排列 查找过程: 首先,假设表中元素是 ...

  5. 批量采集世纪佳缘会员图片及winhttp异步采集效率

    原始出处:http://www.cnblogs.com/Charltsing/p/winhttpasyn.html 最近老有人问能不能绕过世纪佳缘的会员验证来采集图片,我测试了一下,发现是可以的. 同 ...

  6. Integer的NPE问题

  7. (poj 2253) Frogger 最短路上的最大路段

    题目链接:http://poj.org/problem?id=2253 Description Freddy Frog is sitting on a stone in the middle of a ...

  8. Jenkins [Error] at org.codehaus.cargo.container.tomcat.internal.AbstractTomcatManagerDeployer.redeploy(AbstractTomcatManagerDeployer.java:192)

    Deploying /root/.jenkins/workspace/zgg-crm-pre/target/crm.war to container Tomcat 7.x Remote with co ...

  9. 上古神器之Vim编辑器

    在Linux操作环境下进行文本的编辑少不了编辑器vi ,vim,nona... 一. 修改颜色方案 有时候,使用vim打开一个文件,竟然是蓝色的,辨识度相当的差,这个时候,我们可以调整 一下颜色的搭配 ...

  10. 深入解读Promise对象

    promise对象初印象: promise对象是异步编程的一种解决方案,传统的方法有回调函数和事件,promise对象是一个容器,保存着未来才会结束的事件的结果 promise对象有两个特点: 1.p ...