基于C#的ArcEngine二次开发34:使用ConstructUnion方法进行多要素合并(下)

简介: 基于C#的ArcEngine二次开发34:使用ConstructUnion方法进行多要素合并(下)

3 根据几何合并

通过几何条件判断的方式连接断开的线要素的ArcGIS 插件完成,在这里把思路和代码和大家分享下:

3.1 程序思路和实现过程

1.首先读取shp线文件,将各条线要素遍历,存储在List,这里之所以不存在List和List中的原因是后两者会丢失要素的属性信息;


2.为了简化合并算法的逻辑和复杂性,采用分治思想。将线要素分为需要进行合并和不需要进行合并(独立的线,ToPoint或FromPoint是“节点”的线和ToPoint,FromPoint均是“节点”线)的两大类,分别存储在两个集合中;


3.现在只需对需要进行合并的的线要素集合进行合并操作。首先从集合中选取一条Polyline,然后在集合剩下的元素中寻找与其ToPoint或FromPoint相同的线要素,然后使用ITopologicalOperator2.Union()方法进行合并,然后寻找和合并后新的Polyline的ToPoint或FromPoint相同的线要素继续合并,没合并一次,就将合并入的Polyline从集合中移除,直到剩下的Polyline不可和当前合并的Polyline合并为止,这样就得到了第一条合并好的Polyline。接着从集合剩下的线要素再找一条Polyline开始新一轮合并操作,直到List为空,整个合并操作结束;


4.得到合并好的List mergeResultLineList后,需要为其添加属性(NAME等字段),通过IFeatureBuffer接口写入shape和Fields,然后Flush到文件中,结束。

3.2 程序实例和结果

prk5jtgggn43i_166e073e4e4441ab84214568d5f3d5c1.png

prk5jtgggn43i_0fbce72c450d4f7bb108ea751e49b48e.png

prk5jtgggn43i_da3c535dc3a0488ba1830751ca8bda0f.png

3.3 代码

public class MergeDisconnectLine : ESRI.ArcGIS.Desktop.AddIns.Button
    {
        public int CountPercent { get; set; }
        IMap map = null;
        private List<IFeature> DisconnPolylineList = new List<IFeature>();
        private List<IFeature> firstRowFeatureList = new List<IFeature>();
        public MergeDisconnectLine()
        {
            IMxDocument mxDoc = ArcMap.Application.Document as IMxDocument;
            map = mxDoc.FocusMap;
        }
        protected override void OnClick()
        {
            //
            //  TODO: Sample code showing how to access button host
            //
            ArcMap.Application.CurrentTool = null;
            //计算程序耗时
            DateTime beforDT = System.DateTime.Now;
            //Application.EnableVisualStyles();
            //Application.SetCompatibleTextRenderingDefault(false);
            ProgressBar pBar = new ProgressBar();
            //Application.Run(new ProgressBar());           
            List<IFeature> allPolylineList = getAllPolyline();
            List<IFeature> noNeedUnionLineList = getWithoutNeedUnionLineList(allPolylineList);
            List<IFeature> toUnionLineList = GetToUnionLineList(allPolylineList, noNeedUnionLineList);
            List<IFeature> unionLineList = MergeLineListOperate(toUnionLineList);
            AddField();
            WriteNoNeedUnionLineToFile(noNeedUnionLineList);
            WriteUnionLineToFile(unionLineList, firstRowFeatureList);
            DateTime afterDT = System.DateTime.Now;
            TimeSpan ts = afterDT.Subtract(beforDT);
            int minutes = ts.Minutes;
            int seconds = ts.Seconds%60;
            //pBar.Hide();
            MessageBox.Show("线要素合并结束,运行程序共耗时约:" + minutes + "分"+ seconds + "秒");
        }
        //获取shp文件中所有的Polyline(IFeature)对象
        public List<IFeature> getAllPolyline()
        {
            IFeatureLayer featureLayer = map.get_Layer(1) as IFeatureLayer;
            IFeatureClass featureClass = featureLayer.FeatureClass;
            IQueryFilter queryFilter = new QueryFilter();
            queryFilter.WhereClause = "";
            IFeatureCursor pFeatCursor = featureClass.Search(queryFilter, false);
            IFeature pFeature = pFeatCursor.NextFeature();
            while (pFeature != null)
            {
                if (featureClass.ShapeType == esriGeometryType.esriGeometryPolyline)
                {
                    DisconnPolylineList.Add(pFeature);
                }
                pFeature = pFeatCursor.NextFeature();                 
            }
            return DisconnPolylineList;
        }
        //获取需要进行合并的线(是noNeedUnionLineList的补集)
        public List<IFeature> GetToUnionLineList(List<IFeature> allPolylineList, List<IFeature> noNeedUnionLineList)
        {
            List<IFeature> toUnionLineList = allPolylineList;
            foreach (IFeature featureLine in noNeedUnionLineList)
            {
                toUnionLineList.Remove(featureLine);
            }
            return toUnionLineList;
        }
        //获取不需要进行合并的线(独立线,一端是节点的线,和两端都是节点的线)       
        public List<IFeature> getWithoutNeedUnionLineList(List<IFeature> allPolylineList)
        {
            List<IFeature> noNeedUnionLineList = new List<IFeature>();
            foreach (IFeature featureLine in allPolylineList)
            {
                int count = featureLine.Fields.FieldCount;
                List<IFeature> allPolylineListCopy = allPolylineList;
                IGeometry geoLine = featureLine.Shape;
                IPolyline lineFirst = geoLine as IPolyline;
                IPoint startPt1 = lineFirst.FromPoint;
                IPoint endPt1 = lineFirst.ToPoint;
                int fromFlag = 0;
                int toFlag = 0;
                for (int i = 0; i < allPolylineListCopy.Count; i++)
                {
                    IFeature line2 = allPolylineListCopy[i];
                    IGeometry geoLine2 = line2.Shape;
                    IPolyline lineSecond = geoLine2 as IPolyline;
                    IPoint startPt2 = lineSecond.FromPoint;
                    IPoint endPt2 = lineSecond.ToPoint;
                    //FromPoint相同的点
                    if ((startPt1.X == startPt2.X && startPt1.Y == startPt2.Y) ||
                        (startPt1.X == endPt2.X && startPt1.Y == endPt2.Y))
                    {
                        fromFlag++;
                    }
                    //ToPoint相同的点
                    if ((endPt1.X == endPt2.X && endPt1.Y == endPt2.Y) ||
                    (endPt1.X == startPt2.X && endPt1.Y == startPt2.Y))
                    {
                        toFlag++;
                    }
                }
                if (fromFlag > 2 || toFlag > 2 || (fromFlag == 1 && toFlag == 1))
                {
                    noNeedUnionLineList.Add(featureLine);
                    //noNeedUnionLineFileds.Add(featureLine.Fields);
                }
            }
            return noNeedUnionLineList;
        }
        //将需要进行合并的线要素(没有节点)集合进行合并,结果为多条线       
        public List<IFeature> MergeLineListOperate(List<IFeature> toUnionLineList)
        {
            List<IFeature> mergeResultLineList = new List<IFeature>();
            int CountPercent = 0;
            while (toUnionLineList.Count > 0)
            {
                CountPercent++;
                //初始化当前合并的线要素
                IFeature unionCurrentLine = toUnionLineList[0];
                //将第一个要素的属性字段值作为最终合并线要素的值
                firstRowFeatureList.Add(unionCurrentLine);
                List<IFeature> currentMergeLineList = new List<IFeature>();
                int count2 = 0;
                do
                {
                    count2++;
                    IFeature featureFirst = unionCurrentLine;
                    IGeometry geoLineFirst = featureFirst.Shape;
                    IPolyline lineFirst = geoLineFirst as IPolyline;
                    IPoint startPt1 = lineFirst.FromPoint;
                    IPoint endPt1 = lineFirst.ToPoint;
                    toUnionLineList.Remove(featureFirst);
                    currentMergeLineList.Clear();
                    currentMergeLineList.Add(featureFirst); 
                    List<IFeature> allPolylineListTemp1 = new List<IFeature>();
                    List<IFeature> allPolylineListTemp2 = new List<IFeature>();
                    int bStart1 = 0;
                    int bStart2 = 0;
                    for (int j = 0; j < toUnionLineList.Count; j++)
                    {
                        IFeature featureSecond = toUnionLineList[j];
                        IGeometry geoLineSecond = featureSecond.Shape;
                        IPolyline lineSecond = geoLineSecond as IPolyline;
                        IPoint startPt2 = lineSecond.FromPoint;
                        IPoint endPt2 = lineSecond.ToPoint;
                        if ((startPt1.X == startPt2.X && startPt1.Y == startPt2.Y) ||
                            (startPt1.X == endPt2.X && startPt1.Y == endPt2.Y))
                        {
                            bStart1++;
                            if (bStart1 > 0)
                            {
                                allPolylineListTemp1.Add(featureSecond);
                                currentMergeLineList.AddRange(allPolylineListTemp1);
                                toUnionLineList.Remove(featureSecond);
                            }
                        }
                        if ((endPt1.X == endPt2.X && endPt1.Y == endPt2.Y) ||
                            (endPt1.X == startPt2.X && endPt1.Y == startPt2.Y))
                        {
                            bStart2++;
                            if (bStart2 > 0)
                            {
                                allPolylineListTemp2.Add(featureSecond);
                                currentMergeLineList.AddRange(allPolylineListTemp2);
                                toUnionLineList.Remove(featureSecond);
                            }
                        }
                    }
                    if (currentMergeLineList.Count > 1)
                    {
                        unionCurrentLine = UnionCurrentLineList(currentMergeLineList);
                    }
                    else
                    {
                        int ii = 0;
                    }
                } while (currentMergeLineList.Count > 1);
                mergeResultLineList.Add(unionCurrentLine);
            }
            return mergeResultLineList;
        }
        //为待写入图层添加Name和GB字段
        public void AddField()
        {
            IFeatureLayer featureLayer2 = map.get_Layer(0) as IFeatureLayer;
            IFeatureClass featureClass2 = featureLayer2.FeatureClass;
            IClass pClass = featureClass2 as IClass;
            IField fld1 = new Field();
            IField fld2 = new Field();
            IFieldEdit2 fld_NAME = fld1 as IFieldEdit2;
            IFieldEdit2 fld_GB = fld2 as IFieldEdit2;
            fld_NAME.Type_2 = esriFieldType.esriFieldTypeString;
            fld_NAME.Name_2 = "NAME";
            fld_GB.Type_2 = esriFieldType.esriFieldTypeString;
            fld_GB.Name_2 = "GB";
            pClass.AddField(fld_GB);
            pClass.AddField(fld_NAME);
        }
        public void WriteNoNeedUnionLineToFile(List<IFeature> pLineList)
        {           
            foreach (IFeature featureLine in pLineList)
            {
                IFeatureLayer featureLayer2 = map.get_Layer(0) as IFeatureLayer;
                IFeatureClass featureClass2 = featureLayer2.FeatureClass;          
                IFeatureBuffer featureBuffer = featureClass2.CreateFeatureBuffer();
                IFeatureCursor featureCursor;
                featureCursor = featureClass2.Insert(true);               
                IGeometry pGeometry = featureLine.Shape;
                featureBuffer.Shape = pGeometry;
                int NAME_Index = featureLine.Fields.FindField("NAME");
                int GB_Index = featureLine.Fields.FindField("GB");
                string name = featureLine.get_Value(NAME_Index).ToString();
                string gb = featureLine.get_Value(GB_Index).ToString();
                int fieldindex1 = featureBuffer.Fields.FindField("NAME");
                int fieldindex2 = featureBuffer.Fields.FindField("GB");
                if (fieldindex1 >= 0)
                {
                    featureBuffer.set_Value(fieldindex1, name);
                }
                if (fieldindex2 >= 0)
                {
                    featureBuffer.set_Value(fieldindex2, gb);
                }
                featureCursor.InsertFeature(featureBuffer);
                featureCursor.Flush();
                System.Runtime.InteropServices.Marshal.ReleaseComObject(featureCursor);               
            }           
        }
        public void WriteUnionLineToFile(List<IFeature> mergeResultLineList, List<IFeature> firstFeatureList)
        {
            int index = 0;
            foreach (IFeature featureLine in mergeResultLineList)
            {
                IFeatureLayer featureLayer2 = map.get_Layer(0) as IFeatureLayer;
                IFeatureClass featureClass2 = featureLayer2.FeatureClass;
                IFeatureBuffer featureBuffer = featureClass2.CreateFeatureBuffer();
                IFeatureCursor featureCursor;
                featureCursor = featureClass2.Insert(true);
                IGeometry pGeometry = featureLine.Shape;
                featureBuffer.Shape = pGeometry;
                int NAME_Index = firstFeatureList[index].Fields.FindField("NAME");
                int GB_Index = firstFeatureList[index].Fields.FindField("GB");
                string name = firstFeatureList[index].get_Value(NAME_Index).ToString();
                string gb = firstFeatureList[index].get_Value(GB_Index).ToString();
                int fieldindex1 = featureBuffer.Fields.FindField("NAME");
                int fieldindex2 = featureBuffer.Fields.FindField("GB");
                if (fieldindex1 >= 0)
                {
                    featureBuffer.set_Value(fieldindex1, name);
                }
                if (fieldindex2 >= 0)
                {
                    featureBuffer.set_Value(fieldindex2, gb);
                }
                featureCursor.InsertFeature(featureBuffer);
                featureCursor.Flush();
                System.Runtime.InteropServices.Marshal.ReleaseComObject(featureCursor);
                index++;
            }
        }
        //将传入的List<IPolylne>中的多条线要素进行合并为一条线要素
        public IFeature UnionCurrentLineList(List<IFeature> currentMergeLineList)
        {
            IFeatureLayer featureLayer = map.get_Layer(1) as IFeatureLayer;
            IFeatureClass featureClass = featureLayer.FeatureClass;
            ITopologicalOperator2 pTopologicalOperator;
            IFeature pFeatureTemp = currentMergeLineList[0];
            IGeometry pGeometry = pFeatureTemp.Shape;
            int i = 1;
            while (i < currentMergeLineList.Count)
            {
                pTopologicalOperator = pGeometry as ITopologicalOperator2;
                //ITopologicalOperator的操作是bug很多的,先强制的检查下面三个步骤,再进行操作
                //modifiy in 2016/03/20 12:47
                pTopologicalOperator.IsKnownSimple_2 = false;
                pTopologicalOperator.Simplify();
                pGeometry.SnapToSpatialReference();
                pGeometry = currentMergeLineList[i].Shape;
                pGeometry = pTopologicalOperator.Union(pGeometry);
                i++;
            }
            IFeature unionLine = featureClass.CreateFeature();
            unionLine.Shape = pGeometry;
            IDataset pDataset = featureClass as IDataset;
            pDataset.Workspace.ExecuteSQL("delete from " + featureClass.AliasName + " where SHAPE_Length = 0" );
            return unionLine;
        }
        protected override void OnUpdate()
        {
            Enabled = ArcMap.Application != null;
        }
    }
相关文章
|
5月前
|
开发框架 .NET 程序员
C# 去掉字符串最后一个字符的 4 种方法
在实际业务中,我们经常会遇到在循环中拼接字符串的场景,循环结束之后拼接得到的字符串的最后一个字符往往需要去掉,看看 C# 提供了哪4种方法可以高效去掉字符串的最后一个字符
475 0
|
4月前
|
编译器 C#
C#多态概述:通过继承实现的不同对象调用相同的方法,表现出不同的行为
C#多态概述:通过继承实现的不同对象调用相同的方法,表现出不同的行为
148 65
|
3月前
|
JSON 程序员 C#
使用 C# 比较两个对象是否相等的7个方法总结
比较对象是编程中的一项基本技能,在实际业务中经常碰到,比如在ERP系统中,企业的信息非常重要,每一次更新,都需要比较记录更新前后企业的信息,直接比较通常只能告诉我们它们是否指向同一个内存地址,那我们应该怎么办呢?分享 7 个方法给你!
|
3月前
|
C# UED SEO
C# 异步方法async / await任务超时处理
通过使用 `Task.WhenAny`和 `Task.Delay`方法,您可以在C#中有效地实现异步任务的超时处理机制。这种方法允许您在指定时间内等待任务完成,并在任务超时时采取适当的措施,如抛出异常或执行备用操作。希望本文提供的详细解释和代码示例能帮助您在实际项目中更好地处理异步任务超时问题,提升应用程序的可靠性和用户体验。
121 3
|
4月前
|
存储 C#
【C#】大批量判断文件是否存在的两种方法效率对比
【C#】大批量判断文件是否存在的两种方法效率对比
78 1
|
4月前
|
C#
C#的方法的参数传递
C#的方法的参数传递
44 0
|
4月前
|
数据可视化 程序员 C#
C#中windows应用窗体程序的输入输出方法实例
C#中windows应用窗体程序的输入输出方法实例
78 0
|
3月前
|
C# 开发者
C# 一分钟浅谈:Code Contracts 与契约编程
【10月更文挑战第26天】本文介绍了 C# 中的 Code Contracts,这是一个强大的工具,用于通过契约编程增强代码的健壮性和可维护性。文章从基本概念入手,详细讲解了前置条件、后置条件和对象不变量的使用方法,并通过具体代码示例进行了说明。同时,文章还探讨了常见的问题和易错点,如忘记启用静态检查、过度依赖契约和性能影响,并提供了相应的解决建议。希望读者能通过本文更好地理解和应用 Code Contracts。
60 3
|
2月前
|
存储 安全 编译器
学懂C#编程:属性(Property)的概念定义及使用详解
通过深入理解和使用C#的属性,可以编写更清晰、简洁和高效的代码,为开发高质量的应用程序奠定基础。
116 12
|
3月前
|
设计模式 C# 图形学
Unity 游戏引擎 C# 编程:一分钟浅谈
本文介绍了在 Unity 游戏开发中使用 C# 的基础知识和常见问题。从 `MonoBehavior` 类的基础用法,到变量和属性的管理,再到空引用异常、资源管理和性能优化等常见问题的解决方法。文章还探讨了单例模式、事件系统和数据持久化等高级话题,旨在帮助开发者避免常见错误,提升游戏开发效率。
114 4