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 ? : ));
odbf.Open(dbfPath, FileMode.Open);
读取记录数、字段数、长度
var header=odbf.Header;
int dbfRecordCount=Convert.ToInt32( header.RecordCount);
//header.ColumnCount字段数
for (int i = ; i < header.ColumnCount; i++)
{
this.dataGridView2.Rows.Add();
//字段名
this.dataGridView2[, i].Value = header[i].Name;
//字段类型
this.dataGridView2[, i].Value = header[i].ColumnType.ToString();
//字段长度与小数位
if (header[i].DecimalCount!=)
{
this.dataGridView2[, i].Value = header[i].Length.ToString() + "," + header[i].DecimalCount;
}
else
this.dataGridView2[, 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();
int i = ;
while (orec != null)
{
DataRow[] dataRows = dt.Select(linkFieldNameExcel + "=" + "'" + orec[prms.linkFieldIndexShp].ToString() + "'");
if (dataRows.Count() > )
{
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();
}