ArcGIS Engine空间查询功能的实现(QueryFilterClass+SpatialFilterClass)
地图中包含大量的信息,为了快速地了解所需信息,必须借助为空间数据专门编写的空间查询功能。
空间查询主要有两种类型:
基于属性的查询,也称为属性查询。
基于空间位置的查询,也称为空间查询。
查询类的基本思路(适用于属性查询以及空间查询)

1、属性查询
基于属性的查询,即属性查询,是通过对要素的属性信息设定查询条件来查询、定位空间位置。
QueryFilterClass是专门用于属性查询的属性查询过滤器。
被称为过滤器是因为拥有WhereClause属性——能够设置查询条件的Sql语句。
注意:
使用WhereClause应该注意以下几点:
(1)大小写——字段值不区分大小写。
(2)通配符的使用——“?”代表单个字符,“*”代表一组字符。
(3)不支持Orderby关键字,可以通过ITablesort接口完成。
2、空间查询
基于空间位置查询,即空间查询,是根据要素之间的空间位置关系进行的查询,查询结果包括了被查询对象的空间信息以及属性信息。
SpatialFilterClass对象类被称为空间查询过滤器。它是属性过滤器的子类,它拥有的ISpatialFilter也继承了IQueryFilter接口。
因此,空间查询过滤器既能进行空间查询也能进行属性查询。
一般地,要素之间的空间关系,即查询范围与被查询对象的空间关系主要有以下几种:
相交(Intersect)、叠加(Overlap)、穿越(Cross)、在内部(Within)和包含(Con-tain)。
空间关系(SpatialRel)属性成员不需要用户自定义,系统以枚举常量的方式(即ESRISpatialRelEnum)向用户提供了多种空间关系。
(1)ESRISpatialRellntersects:空间相交
(2)ESRISpatialRelTouches:空间相接
(3)ESRISpatialRelOverlaps:覆盖
(4)ESRISpatialRelCrosses:穿越
(5)ESRISpatialRelWithin:在内部
(6)ESRISpatialRelContains:包含
I.数据类型转换(底层模块)
/// <summary>
/// 转换数据类型
/// </summary>
/// <param name="type"></param>
/// <returns></returns>
private static string ParseFieldType(esriFieldType FieldType)
{
switch (FieldType)
{
case esriFieldType.esriFieldTypeInteger:
return "System.Int32";
case esriFieldType.esriFieldTypeOID:
return "System.Int32";
case esriFieldType.esriFieldTypeDouble:
return "System.Double";
case esriFieldType.esriFieldTypeDate:
return "System.DateTime";
default:
return "System.String";
}
}
II.属性查询实现模块
/// <summary>
/// 核心属性查询函数
/// </summary>
/// <param name="pFtClass"></param>
/// <param name="pWhereClause"></param>
/// <returns></returns>
public static DataTable Search(IFeatureClass pFtClass, string pWhereClause)
{
//定义过滤器对象
IQueryFilter pQueryFilter = new QueryFilter();
//设置sql查询语句
pQueryFilter.WhereClause = pWhereClause;
//设置游标
//调用.Search方法;false表示游标到达最后一条要素以后不回收
//IFeatureCursor Search(IQueryFilter filter, bool Recycling);
IFeatureCursor pFtCursor = pFtClass.Search(pQueryFilter, false);
//声明一个pFt要素并将查询结果中的第一条Feature赋值给它
IFeature pFt = pFtCursor.NextFeature();
//实例化一个DataTable内存表对象, 用以存储从要素中读取出来的各属性字段值
DataTable DT = new DataTable();
for (int i = ; i < pFtCursor.Fields.FieldCount; i++)
{
//构建表结构:字段名和数据类型
DataColumn dc = new DataColumn(pFtCursor.Fields.get_Field(i).Name,
System.Type.GetType(ParseFieldType(pFtCursor.Fields.get_Field(i).Type)));
//字段生成完成后添加到DT的列中
DT.Columns.Add(dc);
}
//当pFt不为空, 遍历查询属性值放到DataTable中显示
while (pFt != null)
{
//遍历查询结果逐个写入到DT中
DataRow dr = DT.NewRow();
//以DT的表结构新建行对象
for (int i = ; i < pFt.Fields.FieldCount; i++)
{
dr[i] = pFt.get_Value(i);
}
//完成某一行的字段值录入后向DT中添加此行对象
DT.Rows.Add(dr);
//指向下一个要素
pFt = pFtCursor.NextFeature();
}
return DT;//返回DataTable对象
}
III.空间查询实现模块
/// <summary>
/// 核心空间查询函数
/// </summary>
/// <param name="pFtClass">查询要素类</param>
/// <param name="pWhereClause">SQL语句</param>
/// <param name="pGeometry">空间查询范围</param>
/// <param name="pSpRel">空间关系</param>
/// <returns></returns>
private DataTable SpatialSearch(IFeatureClass pFtClass, string pWhereClause, IGeometry pGeometry, esriSpatialRelEnum pSpRel)
{
//定义空间查询过滤器对象
ISpatialFilter pSpatialFilter = new SpatialFilterClass();
//设置sql查询语句
pSpatialFilter.WhereClause = pWhereClause;
//设置查询范围
pSpatialFilter.Geometry = pGeometry;
//给定范围与查询对象的空间关系
pSpatialFilter.SpatialRel = pSpRel; //查询结果以游标的形式返回(下面与属性查询一样)
IFeatureCursor pFtCursor = pFtClass.Search(pSpatialFilter, false);
IFeature pFt = pFtCursor.NextFeature();
DataTable DT = new DataTable();
for (int i = ; i < pFtCursor.Fields.FieldCount; i++)
{
DataColumn dc = new DataColumn(pFtCursor.Fields.get_Field(i).Name,
System.Type.GetType(ParseFieldType((pFtCursor.Fields.get_Field(i).Type))));
DT.Columns.Add(dc);
}
while (pFt != null)
{
DataRow dr = DT.NewRow();
for (int i = ; i < pFt.Fields.FieldCount; i++)
{
dr[i] = pFt.get_Value(i);
}
DT.Rows.Add(dr);
pFt = pFtCursor.NextFeature();
}
return DT;
}
IV.调用方法、显示
1、属性查询

向ListBox中添加图层字段:
/// <summary>
/// 获取ComboBox被选定的索引号
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void cboSelectLayer_SelectedIndexChanged(object sender, EventArgs e)
{
AddFields();
} /// <summary>
/// 向ListBox中添加图层字段
/// </summary>
private void AddFields()
{
//清空ListBox
lbShow.Items.Clear();
//将pWorkspace强转成要素工作空间
IFeatureWorkspace pFtWorkspace = pWorkspace as IFeatureWorkspace;
//通过要素工作空间打开cboSelectLayer选择的图层, 并放在要素类中
IFeatureClass pFtClass = pFtWorkspace.OpenFeatureClass(cboSelectLayer.Text);
//打开游标, null为查询全部, false表示游标到达最后一条要素以后不回收
IFeatureCursor pFtCursor = pFtClass.Search(null, false);
//使用for循环逐一添加图层字段
for (int i = ; i < pFtCursor.Fields.FieldCount; i++)
{
IField pField = pFtCursor.Fields.get_Field(i); //获取字段的下标
lbShow.Items.Add(pField.Name.ToString());//将字段名加到ListBox中
}
}
获取唯一值:
/// <summary>
/// 获取唯一值
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btnGetValue_Click(object sender, EventArgs e)
{
lbValue.Items.Clear();
//将pWorkspace强转成要素工作空间
IFeatureWorkspace pFtWorkspace = pWorkspace as IFeatureWorkspace;
//通过要素工作空间打开cboSelectLayer选择的图层, 并放在要素类中
IFeatureClass pFtClass = pFtWorkspace.OpenFeatureClass(cboSelectLayer.Text);
//打开游标, null为查询全部, false表示游标到达最后一条要素以后不回收
IFeatureCursor pFtCursor = pFtClass.Search(null, false);
//声明一个pFt要素并将查询结果中的第一条Feature赋值给它
IFeature pFt = pFtCursor.NextFeature();
//声明index作为下标使用
int index = ;
//将index赋值为找到选中字段下标
index = pFtCursor.FindField(lbShow.Text);
//当要素不为空时
while(pFt!=null)
{
//如果lbValue中包含要素下标则游标到下一个要素, 跳过本次循环
if(lbValue.Items.Contains(pFt.get_Value(index)))
{
pFt = pFtCursor.NextFeature();
continue;
}
//将游标的的要素加到lbValue中
lbValue.Items.Add(pFt.get_Value(index));
//转到下一个要素
pFt = pFtCursor.NextFeature();
} }
查询按钮:
private void btnSelect_Click(object sender, EventArgs e)
{
//将工作空间强转成要素工作空间
IFeatureWorkspace pFtWorkspace = pWorkspace as IFeatureWorkspace;
//通过要素工作空间打开cboSelectLayer选择的图层, 并放在要素类中
IFeatureClass pFtClass = pFtWorkspace.OpenFeatureClass(cboSelectLayer.Text);
try
{
//调用核心查询方法, 返回类型为DataTable
Global.myDGV1.DataSource = Search(pFtClass, txtSql.Text.Trim());
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
}
}
2、空间查询

主视图的OnMouseDown事件:
/// <summary>
/// 主视图的OnMouseDown事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void axMapControl1_OnMouseDown(object sender, ESRI.ArcGIS.Controls.IMapControlEvents2_OnMouseDownEvent e)
{
//当空间查询的状态为真时
if (IsSpatialSearch)
{
//获取精确图层
ILayer pLayer = axMapControl1.get_Layer(Get_Layer("北部湾"));
//将图层强转成要素图层
IFeatureLayer pFtLayer = pLayer as IFeatureLayer;
//将要素图层的图层类强转成要素类
IFeatureClass pFtClass = pFtLayer.FeatureClass as IFeatureClass;
//随着鼠标拖动得到一个矩形框
IEnvelope pEnvelope = axMapControl1.TrackRectangle();
//调用核心空间查询函数(采用空间相交的方法esriSpatialRelIntersects)
dataGridView1.DataSource = SpatialSearch(pFtClass,"",
pEnvelope,esriSpatialRelEnum.esriSpatialRelIntersects);
axMapControl1.ActiveView.PartialRefresh(esriViewDrawPhase.esriViewGeography, null, null);
}
}
获得精确图层名下的index
/// <summary>
/// 获得精确图层名下的index
/// </summary>
/// <param name="LayerName">图层名字</param>
/// <returns></returns>
private int Get_Layer(string LayerName)
{
//遍历主视图的图层
for (int i = ; i < axMapControl1.LayerCount; i++)
{
//如果图层索引对应的名字和用户输入的名字相同则返回索引
if (axMapControl1.get_Layer(i).Name.Equals(LayerName))
{
return i;
}
}
return -;//返回-1
}
用于判断空间查询的状态:
//用于判断空间查询的状态
bool IsSpatialSearch = false;
谢谢观看!本人初学GIS二次开发,如果有不对的地方,请多多包涵!
ArcGIS Engine空间查询功能的实现(QueryFilterClass+SpatialFilterClass)的更多相关文章
- ArcGIS Engine空间分析之拓扑分析的实现
简单介绍: 拓扑学是一门研究几何图形位置关系的科学. GIS所关注的拓扑主要集中在拓扑关系——存在于地理实体间的拓扑关系. 拓扑关系在GIS中起着描述两个地理实体的相对空间位置的重要作用.它是GIS空 ...
- ArcGIS Engine空间分析之缓冲区分析的实现
缓冲分析(BufferAnalysis)的结果是一个面状要素——即缓冲要素,点状要素.线状要素和面状要素,被缓冲分析功能处理过之后,它们的周围产生一个缓冲区域,该区域即新产生的面状要素. 在缓冲方向上 ...
- ArcGIS Engine开发之空间查询
空间查询功能是通过用户选择的空间几何体以及该几何体与当前地图中要素之间的几何关系进行空间查找,从而得到查询结果的操作. 相关类与接口 空间查询相关的类主要是SpatialFilter类,其实现的接口主 ...
- ArcGIS api for JavaScript 3.27 FindTask查询功能
在ArcGIS API中查询功能是经常使用的,常用的三个查询分别是FindTask,QueryTask,IdentifyTask.它们各自都有自己的特点. 查询功能分为属性查询和空间查询 FindTa ...
- ArcGIS Engine断开其他ArcSDE用户连接的解决方案
来自:http://blog.csdn.net/linghe301/article/details/38925481 最近有很多用户咨询在ArcGIS Engine中希望能够实现断开其他客户端连接Ar ...
- openlayers4 入门开发系列之地图空间查询篇(附源码下载)
前言 openlayers4 官网的 api 文档介绍地址 openlayers4 api,里面详细的介绍 openlayers4 各个类的介绍,还有就是在线例子:openlayers4 官网在线例子 ...
- ArcGIS Engine开发之属性查询
属性查询即基于空间数据的属性数据的查询,通过用户提交SQL语言中的where语句定义的查询条件,对属性数据进行搜索,从而得到查询结果的操作. 相关的类与接口 与属性查询功能相关的类主要有QureyFi ...
- ArcGIS Engine开发之旅08--和查询相关的对象和接口
原文:ArcGIS Engine开发之旅08--和查询相关的对象和接口 查询在GIS领域应该是一个很频繁的操作,在GIS中除了具有属性查询(和其他关系型数据库的查询类似),还提供了空间查询.在介绍查询 ...
- ArcGIS Engine开发之旅09--几何对象和空间参考
原文:ArcGIS Engine开发之旅09--几何对象和空间参考 1.Geometry Geometry 是 GIS 中使用最为广泛的对象集之一,用户在创建.删除.编辑和进行地理分析的时候,就是处 ...
随机推荐
- JAVA自学笔记 - 从零开始
文中记录的内容都是博主从自己的学习笔记中总结的. 如果遇到问题,或者有不一样的看法,欢迎提出! 1安装JDK 从Oracle官网下载JDK,我使用的版本是1.7.0.80. 操作系统是win7 64位 ...
- 使用 Eslint & standard 规范前端代码
前言 JavaScript的动态语言类型,给它带来了独特的魅力,产生了风格多样的开发范式,同时也带来了一些问题,从运行时常见的 undefined .null 报错,到代码随意的加减分号.换行.空格, ...
- JavaScript之操作符
计算机被发明的初衷仅仅是为了快速实现一些数学计算,然而经过多年发展,计算机已经不单单能实现快速计算这么简单的工作了,现代计算机不仅能够进行数值的计算,还能进行逻辑计算,还具备存储记忆功能,是能够按照程 ...
- 使用Nexus3搭建Maven私服
1.搭建Maven私服背景 公司还是按捺不住,要搭建一个自己的Maven本地仓库,可以让开发人员down架包,从内网还是快很多. 这样公司的maven本地仓库就是 开发人员自己电脑上的maven仓库 ...
- 记一个复杂组件(Filter)的从设计到开发
此文前端框架使用 rax,全篇代码暂未开源(待开源) 原文链接地址:Nealyang/PersonalBlog 前言 貌似在面试中,你如果设计一个 react/vue 组件,貌似已经是司空见惯的问题了 ...
- 如何把安全证书导入到java中的cacerts证书库
每一步:进入某个https://www.xxx.com开头的网站,把要导入的证书下载, 在该网页上右键 >> 属性 >> 点击"证书" >> 再 ...
- 阿里云安装RocketMQ
说明: 我的阿里云是centos 6.9 jdk 1.8.0_192-b12(安装教程参照:https://www.cnblogs.com/kingsonfu/p/9801556.html) mave ...
- Spring Cloud Gateway 使用
简介 Spring Cloud Gateway是Spring Cloud官方推出的网关框架,网关作为流量入口,在微服务系统中有着十分重要的作用,常用功能包括:鉴权.路由转发.熔断.限流等. Sprin ...
- 前端深入之css篇|你真的了解“权重”吗?
写在前面 权重这个概念,相信对许多进行过前端开发的小伙伴来说肯定并不陌生,有时候一个样式添加不上,我们就会一个 !important 怼上去,一切就好像迎刃而解了.但还有的时候,!important也 ...
- HTML5 原生拖放
前言: HTML5提供专门的拖拽与拖放的API,可以方便的指定某个元素可拖动,可以创建自定义的可拖动元素和放置目标 相关知识点: 1.拖放事件 拖放元素时,将依次触发下列事件 dragstart 按 ...