您的位置:首页 > 编程语言 > Python开发

GDAL 安装支持Python和C#出现的问题

2015-04-10 11:28 141 查看
本文是复制网友的经验原文地址在下面给出,希望对有同样问题的同学有点帮助

Windows下编译C#可用的GDAL附带netCDF格式支持库的一点拾遗

/article/5018345.html

之前一直在搞GDAL的C++开发,因此对linux下的GDAL编译算是有一点心得,但是从未搞过Windows下的GDAL编译,更没有弄过.NET平台下的GDAL库。

今天一天时间总算是编译出了.NET(x86)可用的GDAL with netCDF库,把遇到的几个问题记下来。

1、Windows下GDAL C/C++库的编译

(1)由于要支持netCDF格式,因此首先下载UNIDATA提供的netcdf的二进制发行包,下载地址:netCDF4.3RC4-NC4-DAP-32.exe

安装解压后,只需要其中的bin、deps、include、lib四个文件夹及其内容,假设放在本地D:\GDAL\compiled\netcdf430 目录下。

(2)下载SWIG-2.0.6(或者1.3.39,最新的2.0.9均可),解压后放置在D:\GDAL\swigwin-2.0.6 目录下。

(3)下载GDAL-1.9.2源码,点击此处下载地址,下载并解压后放置在本地目录,例如 D:\GDAL\gdal-1.9.2 下。

(4)修改nmake.opt文件:



GDAL_HOME = "C:\warmerda\bld"
修改为:
GDAL_HOME = "D:\GDAL\current_bld"

# Uncomment the following to enable NetCDF format.
#NETCDF_PLUGIN = NO
#NETCDF_SETTING=yes
#NETCDF_LIB=C:\Software\netcdf\lib\netcdf.lib
#NETCDF_INC_DIR=C:\Software\netcdf\include
修改为:
# Uncomment the following to enable NetCDF format.
NETCDF_PLUGIN = yes
NETCDF_SETTING=yes
NETCDF_LIB=D:\GDAL\compiled\netcdf430\lib\netcdf.lib
NETCDF_INC_DIR=D:\GDAL\compiled\netcdf430\include

# Set the location of your SWIG installation
!IFNDEF SWIG
SWIG = swig.exe
!ENDIF
修改为:
# Set the location of your SWIG installation
!IFNDEF SWIG
SWIG = D:\GDAL\swigwin-2.0.6\swig.exe
!ENDIF




(5)启动VS2010命令行工具,转到 D:\GDAL\gdal-1.9.2 目录下, 依次运行下面的命令:

nmake -f makefile.vc

nmake -f makefile.vc install
或者
nmake -f makefile.vc devinstall


(6)将netcdf的dll以及依赖的所有dll全部copy到gdal19.dll目录下,运行gdalinfo.exe --formats命令查看支持的文件格式,能够找到下述一行即说明成功。

netCDF (rw+): Network Common Data Format


2、编译C#可用的GDAL库

第一步完成后得到的gdal19.dll是C/C++可用的本地代码DLL,还不能被C#所调用,使用SWIG可以实现对GDAL接口的快速封装和多语言导出。

第一步编译以及安装完成后,转到D:\GDAL\gdal-1.9,2\swig\csharp目录下,依次运行下述命令:

nmake -f makefile.vc interface
nmake -f makefile.vc
nmake -f makefile.vc install


在设置的GDAL_HOME目录下出现内含8个dll的csharp文件夹,将其copy到gdal19.dll相同目录下,C#引用gdal_csharp.dll即可。

在执行上述命令时,可能遇到的问题有以下几个:

(1)接口重定义

osr\OsrPINVOKE.cs(192,10): error CS0111:
类型“OSGeo.OSR.OsrPINVOKE”已定义了一个名为“OsrPINVOKE”的具有相同参数类型的成员
osr\OsrPINVOKE.cs(188,10): (与前一个错误相关的符号位置)
NMAKE : fatal error U1077: “C:\Windows\Microsoft.NET\Framework\v4.0.30319\csc.EXE”: 返回代码“0x1”


解决办法比较简单,只需要进入到..\gdal-1.9.2\swig\csharp\gdal|ogr|osr三个文件夹下,找到GdalPINVOKE.cs、OgrPINVOKE.cs、OsrPINVOKE.cs三个文件大约都是第188~192行,将下述重复的声明注释掉其中一个。

static GdalPINVOKE() {
}
//static GdalPINVOKE() {
//}


(2)SWIG生成的接口成员名称错误,如:



gdal\Band.cs(17,79): error CS0117:
“OSGeo.GDAL.GdalPINVOKE”并不包含“BandUpcast”的定义
gdal\Dataset.cs(17,82): error CS0117:
“OSGeo.GDAL.GdalPINVOKE”并不包含“DatasetUpcast”的定义
gdal\Driver.cs(17,81): error CS0117:
“OSGeo.GDAL.GdalPINVOKE”并不包含“DriverUpcast”的定义
NMAKE : fatal error U1077: “C:\Windows\Microsoft.NET\Framework\v4.0.30319\csc.EXE”: 返回代码“0x1”




上述3个错误比较难定位,只能根据名称去相关类的文件里面搜索,经过反复查找发现SWIG-2.0.6/9均犯了同样的一个错误:在GdalPINVOKE.cs声明为下述三个名称的接口:



[DllImport("gdal_wrap", EntryPoint="CSharp_Driver_SWIGUpcast")]
public static extern IntPtr Driver_SWIGUpcast(IntPtr jarg1);

[DllImport("gdal_wrap", EntryPoint="CSharp_Dataset_SWIGUpcast")]
public static extern IntPtr Dataset_SWIGUpcast(IntPtr jarg1);

[DllImport("gdal_wrap", EntryPoint="CSharp_Band_SWIGUpcast")]
public static extern IntPtr Band_SWIGUpcast(IntPtr jarg1);




在Driver、Dataset、Band类中调用时竟然搞错了名字,写成了DriverUpcast、DatasetUpcast、BandUpcast,分别修改为下列名称即可。



public Driver(IntPtr cPtr, bool cMemoryOwn, object parent) : base(GdalPINVOKE.Driver_SWIGUpcast(cPtr), cMemoryOwn, parent) {
swigCPtr = new HandleRef(this, cPtr);

public Dataset(IntPtr cPtr, bool cMemoryOwn, object parent) : base(GdalPINVOKE.Dataset_SWIGUpcast(cPtr), cMemoryOwn, parent) {
swigCPtr = new HandleRef(this, cPtr);

public Band(IntPtr cPtr, bool cMemoryOwn, object parent) : base(GdalPINVOKE.Band_SWIGUpcast(cPtr), cMemoryOwn, parent) {
swigCPtr = new HandleRef(this, cPtr);




(3)安全透明代码无法调用本机C++代码的问题

System.MethodAccessException”类型的未经处理的异常出现在 gdal_csharp.dll 中。
其他信息: 安全透明方法“OSGeo.GDAL.Gdal.AllRegister()”尝试通过方法“OSGeo.GDAL.GdalPINVOKE.AllRegister()”调用本机代码失败。方法必须是安全关键的或安全可靠关键的,才能调用本机代码。






往往在执行到第一句:Gdal.AllRegister();时就会报出上述错误。

上述错误应该是由.NET平台的安全机制所导致,swig在自动封装GDAL的.NET库时,默认采用下述安全描述(在D:\GDAL\gdal-1.9.2\swig\csharp\AssemblyInfo.cs中):



// The AllowPartiallyTrustedCallersAttribute requires the assembly to be signed with a strong name key.
// This attribute is necessary since the control is called by either an intranet or Internet
// Web page that should be running under restricted permissions.
[assembly: AllowPartiallyTrustedCallers]

// Use the .NET Framework 2.0 transparency rules (level 1 transparency) as default
#if (CLR4)
[assembly: SecurityRules(SecurityRuleSet.Level1)]
#endif




要解决该问题,只需要将调用该库的代码变为所要求的安全关键代码或者安全可靠关键代码即可,但是我搞了半天也不清楚该怎么修改,此路没走通。

另外一种解决办法是修改swig生成的C#封装类代码,强制声明为可被安全透明代码调用即可,以D:\GDAL\gdal-1.9.2\swig\csharp\gdal\Gdal.cs类和D:\GDAL\gdal-1.9.2\swig\csharp\gdal\Dataset.cs类为例,在其类声明的开头添加下述两行代码:



namespace OSGeo.GDAL {
using System;
using System.Runtime.InteropServices;
using System.Security;//新加
[SecuritySafeCritical]//新加
public class Gdal {
//...}}






namespace OSGeo.GDAL {
using System;
using System.Runtime.InteropServices;
using System.Security;//新加
[SecuritySafeCritical]//新加
public class Dataset : MajorObject {
//...}}




同理,如果想在C#中调用哪个类,就为哪个类添加上述两行代码即可。

3、C#读取nc文件实验结果



Open netCDF file SUCCESS!
Data X Size:512; Y Size:512
SUBDATASET_1_NAME=NETCDF:"D:\test\nc\copy.nc":floatv
SUBDATASET_1_DESC=[0x5x5x4] floatv (32-bit floating-point)
SUBDATASET_2_NAME=NETCDF:"D:\test\nc\copy.nc":doublev
SUBDATASET_2_DESC=[0x5x5x4] doublev (64-bit floating-point)
SUBDATASET_3_NAME=NETCDF:"D:\test\nc\copy.nc":chv
SUBDATASET_3_DESC=[0x80] chv (8-bit character)




做人当然要厚道,编译后的GDAL库的下载地址:http://download.csdn.net/detail/yeahgis/5241135

于2013-04-10 00:53

/article/7742813.html

在运行python setup.py 时,报错信息如下:

unable to find vcvarsall.bat



通过一个网友的对此问题的进行更深入的了解,特转贴过来:

经过对C:/Python32/Lib/distutils目录下.py文件的仔细翻阅,发现“unable to find vcvarsall.bat”这句话出在msvc9compiler.py中

再研究这个文件发现问题所在了:

它是在注册表中寻找到vs的目录,再去在vs目录下找vcvarsall.bat。

但是问题就出在python32里面默认是vc9.0也就是vs2008所以它在注册表里面找不到键值返回None

productdir = Reg.get_value(r"%s/Setup/VC" % vsbase,

"productdir")

vsbase值为:Software/Microsoft/VisualStudio/9.0

具体修改代码如下:

msvc9compiler.py中

def find_vcvarsall(version) 用于查找vcvarsall.bat version就是版本号,实际上是9.0

vc_env = query_vcvarsall(VERSION, plat_spec)

VERSION = get_build_version() 因为python32自己是由vc9.0构建的所以VERSION是9.0

修改MSVCCompiler函数:

vc_env = query_vcvarsall(VERSION, plat_spec)

为:

vc_env = query_vcvarsall(8.0, plat_spec)

不过还有这句话:

if VERSION < 8.0:

raise DistutilsPlatformError("VC %0.1f is not supported by this module" % VERSION)

看来,python3.2中的distutils只支持vc8.0以上,即vs2005以上的编译器。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: