GIS之所以是一个通用的工具,就是因为它具有各种各样分析和处理数据的能力。Silverlight API中提供了Task,使我们能够轻松完成常见的分析任务。
先来考虑一下吃饺子的场景。要想吃饺子,我们需要先去买菜,买肉,回家后在厨房里洗菜,揉面, 拌馅,包饺子,煮饺子,吃饺子,之后别忘了洗碗;另一种情况就是去饭馆,告诉服务员我要吃3两茴香,3两韭菜的饺子,然后等着饺子端到你面前,开吃,走人。
在ArcGISServer程序开发中,要完成GIS的分析功能其实和吃饺子是一样的。用ADF编程就像在家里吃饺子,除了架设服务器,所有的工作基本上也都得我们自己在服务器端来完成,要处理的地方比较多;而用客户端API编程相当于去外面吃饺子,我们只要把任务交给相应的Task,之后接受结果就行了,不用做饺子。唯一不同的就是在外面吃完饺子别忘了付钱,而用Task完成分析任务则是免费的。这点也体现在使用客户端API中的Task时,是由ArcGISOnline提供给你的,不需要自己购买AGS软件。
现在来看看Silverlight API目前给我们提供了那些Task功能:
Query:能够在已经发布的服务数据中,通过属性条件(可以属性字段中进行关系判断,字符查找等),图形条件(与输入的图形相交、包含、相离等),或者是两者的组合,查询出满足条件的数据并返回。相当于Engine中的SpatialFilter,当然也是QueryFilter。
Find:在地图数据的属性字段中查找包含有关键字参数的数据并返回。
Identity:对鼠标当前点击位置上的数据进行辨识并返回结果,可以对多个图层的数据进行辨识。
Address Locator:输入经纬度,返回地址结果(Geocoding);输入一个地方的地址,返回经纬度结果(Reverse Geocoding)。由于国内地图数据保密工作做的相当好,这个Task暂时用不到。
Geometry Service:可以对输入的地理数据进行如缓冲区,动态投影,面积/周长量算等几何操作。
Geoprocessing:能够完成复杂的GIS任务,类似ToolBox中的工具。
抽象一下,可以看出,Query完全可以完成Identity和Find的工作,但后两者在特定场合下使用起来比Query要方便的多;Geoprocessing完全可以替代Geometry Service,但是在利用REST API编写的程序中,要尽量使用GeometryService。
再抽象一下,Silverlight API中的这几个Task和JavaScript/FlexAPI中的Task是大同小异的,因为其实它们都是AGS 9.3 REST API中暴露出来的操作资源(OperationResource)见下图:
后面的代码中实际上也是把输入参数封装起来提交到了REST API的特定Endpoint上。要理解好客户端API中的Task,建议熟读AGS的REST SDK。
Task的用法基本上相同,都遵循这几个步骤:初始化Task,设置Task所需参数,提交任务,等待服务器完成任务后,处理返回的结果;进饭馆,想好你要吃什么饺子,告诉服务员,等饺子做好端上来,开始吃。好了,下面我们就通过一个实例(点击这里,查看实例),来学习一下Query和Geometry两个Task的用法。
首先选择工具条中的画线工具,在屏幕上画一条曲线,会根据曲线自动生成一个距离100公里的缓冲区显示在地图上,之后开始查询缓冲区图形经过的州(相交),将结果显示在地图上。可以单击每个州查看详细信息。这里假设你已学习了前几节的内容,只讨论Task用法的部分。
1、利用所画的线生成缓冲区。画线利用的是Draw工具中的Freehand,在这个动作完成后会触发Draw的OnDrawCompleted事件,自然可以在这里开始进行缓冲区的工作,用的是Geometry Service里的Buffer。
初始化Geometry Service。假设已经在Map1中添加了ID为glayerResult的GraphicsLayer,linesymbolred是提前设置好的CartographicLineSymbol:
- private void Draw1_OnDrawComplete(object sender, DrawEventArgs args)
- {
- Draw1.Deactivate();//Freehand动作失效
- //将Freehand画的曲线显示在地图上
- GraphicsLayer glayer = Map1.Layers["glayerResult"] as GraphicsLayer;
- Graphic g = new Graphic();
- g.Symbol = linesymbolred;
- g.Geometry = args.Geometry;
- glayer.Graphics.Add(g);
- //初始化Geometry Service
- GeometryService geometrytask = new GeometryService("http://sampleserver1.arcgisonline.com/ArcGIS/rest/services/Geometry/GeometryServer");
- }
当一个Task完成时会触发Completed事件,失败时也有Failed事件,对这两个事件进行监听:
- geometrytask.BufferCompleted += new EventHandler(geometrytask_BufferCompleted);
- geometrytask.Failed += new EventHandler(geometrytask_Failed);
- BufferParameters bufferparameters = new BufferParameters();
- bufferparameters.Unit = LinearUnit.Kilometer;
- //必须指定下面两个spatialreference,否则buffer结果集为空
- bufferparameters.BufferSpatialReference = new SpatialReference(3395);
- bufferparameters.OutSpatialReference = Map1.SpatialReference;
- bufferparameters.Distances.Add(100);
- bufferparameters.Features.Add(g);
- geometrytask.BufferAsync(bufferparameters);
- private void geometrytask_BufferCompleted(object sender, GraphicsEventArgs args)
- {
- if (args.Results.Count>0)
- {
- GraphicsLayer glayer = Map1.Layers["glayerResult"] as GraphicsLayer;
- Graphic g = new Graphic();
- g.Symbol = fillsymbolBuffer;
- g.Geometry = args.Results[0].Geometry;
- glayer.Graphics.Add(g);
- }
- }
2、利用生成缓冲区的缓冲区进行空间查询。要达到我们的目的,就还需要进行一个Query的Task,那么就可以在这里马不停蹄的开始Query的Task。步骤基本都是一样的,初始化,设置参数,提交结果,处理结果:
- private void geometrytask_BufferCompleted(object sender, GraphicsEventArgs args)
- {
- if (args.Results.Count>0)
- {
- GraphicsLayer glayer = Map1.Layers["glayerResult"] as GraphicsLayer;
- Graphic g = new Graphic();
- g.Symbol = fillsymbolBuffer;
- g.Geometry = args.Results[0].Geometry;
- glayer.Graphics.Add(g);
- //初始化QueryTask
- QueryTask querytask = new QueryTask("http://sampleserver1.arcgisonline.com/ArcGIS/rest/services/Demographics/ESRI_Census_USA/MapServer/5");
- //准备接收结果或者处理失败的通知
- querytask.ExecuteCompleted += new EventHandler(querytask_ExecuteCompleted);
- querytask.Failed += new EventHandler(querytask_Failed);
- //设置Query Task所需的参数
- Query query = new Query();
- query.OutFields.Add("*");//也顺便设置了query.ReturnGeometry=true;
- query.Geometry = g.Geometry;
- query.SpatialRelationship = SpatialRelationship.esriSpatialRelIntersects;
- //向服务器上的对应图层提交任务
- querytask.ExecuteAsync(query);
- Map1.Cursor = System.Windows.Input.Cursors.Wait;
- }
- }
接下来处理QueryTask完成后的结果:
- private void querytask_ExecuteCompleted(object sender, QueryEventArgs args)
- {
- GraphicsLayer graphicslayer = Map1.Layers["glayerResult"] as GraphicsLayer;
- FeatureSet featureset = args.FeatureSet;
- if (featureset != null && featureset.Features.Count > 0)
- {
- graphicslayer.ClearGraphics();
- listboxResults.Items.Clear();
- foreach (Graphic graphic in featureset.Features)
- {
- graphic.Symbol = fillsymbolresult;
- graphicslayer.Graphics.Add(graphic);
- }
- }
- MyMapTip.GraphicsLayer = graphicslayer;
- Map1.Cursor = System.Windows.Input.Cursors.Arrow;
- }
- <esriWidgets:MapTip x:Name="MyMapTip" BorderBrush="#99000000"
- BorderThickness="1" Title="详细信息" VerticalOffset="10"
- HorizontalOffset="10" Background="#DDFFFFFF" />
别忘了万一处理任务失败时的提示:
- private void geometrytask_Failed(object sender, TaskFailedEventArgs args)
- {
- MessageBox.Show("Buffer Error:" + args.Error);
- }
- private void querytask_Failed(object sender, TaskFailedEventArgs args)
- {
- MessageBox.Show("Query failed: " + args.Error);
- Map1.Cursor = System.Windows.Input.Cursors.Arrow;
- GraphicsLayer graphicslayer = Map1.Layers["glayerResult"] as GraphicsLayer;
- graphicslayer.ClearGraphics();
- }
本节内容完毕。上面讲的相对简略,要理解各个Task和参数的用法,还是需要熟悉Silverlight API和前面提到的REST API。另外,Geoprocessing Service实际上是最强大Task,如果有自己的GISServer,完全可以在上面发布自制的Model或者Python脚本,以完成各种GIS分析任务,简单的在线编辑也是可能的。它的用法也万变不离其宗:初始化,设置参数,提交任务,处理结果。不同的是GeoprocessingService有两种提交任务的方法:同步和异步。前者服务器端处理完任务后会立即将结果发送回客户端;后者将任务提交后会得到服务器端返回的一个JobID,即使任务处理完成也不会立即返回,而是需要你拿这个JobID去询问服务器:完成了吗?完成了吗?完成了吗?如果完成,则可以取回相应的结果。
前面说到,虽然去外面吃饺子很方便,但是毕竟那是人家做好的,对于老饕来说还需要自己的口感,自己下厨毕竟能控制整个过程的方方面面,哪怕你想做出饺立方也都是有可能的。同样,ADF编程可以调用服务器端的ArcObjects,让你为所欲为,这点是客户端API无论如何也办不到的。
原文地址:http://bbs.esrichina-bj.cn/ESRI/thread-45302-1-1.html
本文转自温景良(Jason)博客园博客,原文链接:http://www.cnblogs.com/wenjl520/archive/2009/06/02/1494144.html,如需转载请自行联系原作者