C++ AO读取shapefile的属性值

时间:2021-08-14 15:26:24

  C++ AO读取一个shapefile文件的所有属性值
 
   #include "stdafx.h"
 
   #include "iostream.h"
 
   #include "atlbase.h"
 
   #include "comdef.h" 字串1
 
   #pragma warning(push)
 
   #pragma warning(disable : 4192)
 
   #pragma warning(disable : 4146) 字串6
 
   #import "esriSystem.olb" \
 
   raw_interfaces_only, \
 
   raw_native_types, \
 
   no_namespace, \
 
   named_guids, \ 字串7
 
   exclude("OLE_COLOR", "OLE_HANDLE", "VARTYPE")
 
   #import "esriGeometry.olb" raw_interfaces_only, raw_native_types, no_namespace, named_guids
 
   #import "esriDataSourcesFile.olb" raw_interfaces_only, raw_native_types, no_namespace, named_guids
 
   #import "esriGeoDatabase.olb" raw_interfaces_only, raw_native_types, no_namespace, named_guids
 
   字串2
 
   #pragma warning(pop)
 
   字串5
 
   //CString VariantToString1(VARIANT * va);
 
   字串2
 
   int main(int argc, char* argv[])
 
   {
 
   ::CoInitialize(NULL);
 
   字串1
 
   IWorkspaceFactoryPtr pWSFactory(CLSID_ShapefileWorkspaceFactory);
 
   IWorkspacePtr pWS;
 
   // BSTR path;
 
   // path = ::SysAllocString(L"C:\\test");
 
   HRESULT hr = pWSFactory->OpenFromFile(CComBSTR(L"C:\\test\\bj"),0,&pWS);
 
   if (FAILED(hr) || pWS==NULL)
 
   {
 
   cerr《"Failed to open the destination folder."《endl;
 
   return 1;
 
   } 字串1
 
   IFeatureWorkspacePtr pFWS(pWS);
 
   IFeatureClassPtr pFClass;
 
   pFWS->OpenFeatureClass (CComBSTR(L"bj_est1"),&pFClass);
 
   字串3
 
   IFeatureCursorPtr pFCursor;
 
   pFClass->Search (NULL,VARIANT_TRUE,&pFCursor);
 
   字串7
 
   long lngFieldCount ;
 
   IFieldsPtr pFields;
 
   IFieldPtr pField;
 
   pFCursor->get_Fields (&pFields);
 
   pFields->get_FieldCount (&lngFieldCount); 字串4
 
   IFeaturePtr pFeature;
 
   pFCursor->NextFeature (&pFeature); 字串8
 
   USES_CONVERSION; 字串9
 
   long t;
 
   CComBSTR bsValues;
 
   CComVariant varValue;
 
   esriFieldType esriFldType;
 
   while (pFeature)
 
   {
 
   bsValues = CComBSTR(L" ");
 
   for (t =0; t < lngFieldCount ; t++)
 
   {
 
   pFields->get_Field(t, &pField);
 
   pField->get_Type(&esriFldType);
 
   switch (esriFldType)
 
   {
 
   case (esriFieldTypeGeometry):
 
   bsValues += CComBSTR(L"Shape ");
 
   break;
 
   default:
 
   pFeature->get_Value(t, &varValue);
 
   if (varValue.vt != VT_BSTR)
 
   varValue.ChangeType(VT_BSTR); 字串4
 
   bsValues += varValue.bstrVal;
 
   bsValues += CComBSTR(L" ");
 
   break;
 
   }
 
   } 字串3
 
   // _bstr_t outString(bsValues,FALSE); //方法1:为控制台输出作准备
 
   // cout 《 outString《endl;
 
   //方法2:也可以用OLE2A来进行转化,
 
   //当然需要加上 USES_CONVERSION;
 
   cout 《 OLE2A(bsValues)《 endl;
 
   pFCursor->NextFeature (&pFeature);
 
   } 字串6
 
   pWSFactory=NULL;
 
   pWS=NULL;
 
   pFWS=NULL;
 
   pFClass=NULL;
 
   pFCursor=NULL;
 
   pFields=NULL;
 
   pField=NULL;
 
   字串8
 
   ::CoUninitialize();
 
   字串3
 
   return 0;
 
   } 字串5
 
   需要注意的地方:
 
   1、程序最开始要初始化COM库,即::CoInitialize(NULL);,这个一般可以在程序最开始的地方(确切的说只要在开始使用COM组件之前就行吧),如果是做组件我想放在构造的时候是个不错的选择 www.yztrans.com
 
   2、既然初始化了,最后就要卸载COM库(暂且这么说吧),即::CoUninitialize();.它一般在程序的最后(程序退出之前就行)。当然不要它,似乎也能运行,但还是规范点好。这里有个问题,就是程序中用到的变量置空的问题。如果在卸载COM库之前,不将这些变量的置空(释放掉对象的引用),那么在卸载的时候会得到烦人的错误。我就被这个错误困扰了好一阵子(毕竟在VB里COM库的初始化和卸载不要我来做)。当然在一个子过程中,我想不一定需要置空(过程结束,这些变量也就撤销了),但作为一个好习惯,我想还是自己来做一下这个事免得出错.
 
   3、在Import的时候,这些库有先后顺序。比如这里的esriGeometry.olb要放在前面,否则你会得到一个基类未定义的错误,因为这个基类是在esriGeometry.olb这个库里定义的:(
 
   4、下面的问题全集中在字符串的处理上了。COM里用到了BSTR和VARIANT,目的COM原理的书里都有。这两种数据类型会出现在很多地方,搞清楚这两种数据类型很重要。它们的使用有一些麻烦,为了方便我们使用ATL封装的两个类CComBSTR和CComVariant.程序在最后读取属性的时候可以看到,读取的属性值的类型很多,但我们为了在Console中输出,需要做一些处理。字串4
 
   [a]具体是:如果它是Geometry,那么就直接用"Shape"来表示,如果它是其它的类型就把它转化成BSTR,当然本身是BSTR就不用转化了。(注意里面用到的L"",When working with CComBSTR, use the text mapping L"" to declare constant OLECHAR strings. )
 
   [b]对BSTR本身来说无法直接链接两个BSTR,要连接需要转换(如用_bstr_t类)。但CComBSTR重载了+=,真是幸福的事,呵呵。
 
   [c]最后一步在console中输出,cout不支持BSTR,无疑又要用到转换了。方法有两个:第一是用_bstr_t类,将BSTR转换成_bstr_t类型,然后输出;第二个是使用ATL提供的宏,OLE2A,这里需要在程序中添加USES_CONVERSION; 程序中只要添加一次就行了托福答案 www.lefeng123.com