zoukankan      html  css  js  c++  java
  • ArcGis dbf读写——挂接Excel到属性表 C#

    ArcMap提供了挂接Excel表格信息到属性表的功能,但是当数据量较大到以万计甚至十万计的时候这个功能就歇菜了,当然,你可以考虑分段挂接。这个挂接功能只是做了一个表关联,属性记录每个字段的信息需要通过“字段计算器”计算过来。

    Excel数据写入ArcGis属性表功能开发实例中,博主见到太多使用UpdateCursor的方式,界面卡翔,效率感人。

    ArcGis Shapefile的属性表信息存放在一个dbf格式(dbaseIV,dbf4,dbase4,下称dbf4)的文件中,那么,有没有一种方式可以通过直接对它的读写实现快速挂接?通过数据库连接的方式可以实现,但是对客户机的环境配置有要求,起码得有OLEDB、ODBC驱动……,这样整不爽。那么,有没有一种方式可以借由第三库或者某种方式去直接解析它呢?博主去分析了这个可能,DBF文件的结构并不复杂,找个比较成熟的轮子来研究解析最好不过,去年,博主在GitHub发现了它——“FastDBF”,地址:https://github.com/SocialExplorer/FastDBF。该库作者是老外,所以不出意外的对于中文环境下dbf文件的读写不友好,有点儿小bug,相关说明可以查找本人博客。

    优点:插件方式开发下不操作ArcObject对象,并且可以使用多线程+委托的方式使挂接在子线程进行,进度传回主线程更新UI,挂接速度快且不影响ArcMap的浏览使用。

    • dbf4文件格式与解析

    本篇不讲,可以自行百度或参考:

    https://www.clicketyclick.dk/databases/xbase/format/dbf.html

    需要提醒的是网上的格式解析说明文章都将dbf4编码规则默认为ANSI(中文操作系统下是gbk,codepage=936,在非英文操作系统下,这些文章写为ASCII并不严谨)去解析,而实际上ArcGis10.2之后版本生成的dbf文件默认使用了utf-8(codepage65001)编码。这就牵涉到了“FastDBF”在中文环境下的bug。

    • FastDBF读写dbf4文件

    打开dbf文件,注意选择字符编码规则

    var odbf = new DbfFile(Encoding.GetEncoding(rdoGBK.Checked ? 936 : 65001));
    odbf.Open(dbfPath, FileMode.Open);

    读取记录数、字段数、长度

    var header=odbf.Header;
    int dbfRecordCount=Convert.ToInt32( header.RecordCount);
    //header.ColumnCount字段数
    for (int i = 0; i < header.ColumnCount; i++)
    {
        this.dataGridView2.Rows.Add();
    //字段名
        this.dataGridView2[0, i].Value = header[i].Name;
    //字段类型
        this.dataGridView2[2, i].Value = header[i].ColumnType.ToString();
    //字段长度与小数位
        if (header[i].DecimalCount!=0)
        {
            this.dataGridView2[3, i].Value = header[i].Length.ToString() + "," + header[i].DecimalCount;
        }
        else
            this.dataGridView2[3, i].Value = header[i].Length.ToString();
    }

    从DataTable中匹配记录挂接,Excel装入DataTable可以使用epplus,不讲。

    var odbf = new DbfFile(Encoding.GetEncoding(prms.encode));
    DbfRecord orec;
    try
    {
        odbf.Open(prms.dbfPath, FileMode.Open);
        orec = odbf.Read(0);
        int i = 1;
        while (orec != null)
        {
            DataRow[] dataRows = dt.Select(linkFieldNameExcel + "=" + "'" + orec[prms.linkFieldIndexShp].ToString() + "'");
            if (dataRows.Count() > 0)
            {
                DataRow dr = dataRows.First();
                foreach (var item in prms.updateFieldPrms)
                {   string content=dr[item.fieldNamesExcel].ToString();
                    int byteCount= Encoding.GetEncoding(prms.encode).GetByteCount(content);
                    if (byteCount<=item.fieldLength)
                    {
                        orec[item.fieldNameDbf] = content;
                    }
                }
                odbf.Update(orec);
            }
            orec = odbf.ReadNext();
            ProgressChanged(this, i);
            i++;
        }
    }
    catch (Exception)
    {
        throw new Exception("dbf挂接过程出错!");
    }
    finally
    {
        dt.Dispose();
        orec = null;
        odbf.Close();
    }
  • 相关阅读:
    Spring Security教程之自定义Spring Security默认的403页面
    Spring Security教程之Spring Security实现访问控制
    Spring Security的HTTP基本验证示例
    Maven3+Struts2.3.1.2整合的Hello World例子
    将Flex嵌入到Jsp页面实例-基于FlexModule插件
    Java I/O之FilenameFilter类列举出指定路径下某个扩展名的文件
    Java I/O之用FilenameFilter实现根据文件扩展名删除文件
    Flex与Java通信之HttpService方式
    Flex与Java通信之RemoteObject方式
    Flex之理解Flash中的事件机制
  • 原文地址:https://www.cnblogs.com/yzhyingcool/p/10657350.html
Copyright © 2011-2022 走看看