基于 Python(gma) 的 克里金(Kriging)法插值的主要过程

  由于克里金插值的复杂性,本文不再对其原理进行介绍。详情可自行百度。

  • 本算法基于 Python 的开源克里金插值包 pykrige。
  • 但本算法已对其进行改造,以使其符合 gma 的整体逻辑。
  • 本算法已不包含任何 pykrige 的原始代码。

原始代码构建

from gma.algorithm.spmis.interpolate import *

class Kriging(IPolate):
    '''克里金法插值'''
	# 继承 gma 的标准插值类 IPolate。本处不再做详细说明。
    def __init__(self, Points, Values, Boundary = None, Resolution = None, 
                 InProjection = 'WGS84', 
                 VariogramModel = 'Linear',
                 VariogramParameters = None,
                 **kwargs):
        
        IPolate.__init__(self, Points, Values, Boundary, Resolution, InProjection)
        
        self.eps = eps
        
        # 初始化半变异函数及参数
        self.VariogramModel, self.VParametersList = GetVariogramParameters(VariogramModel, VariogramParameters)
        self.VariogramFUN = getattr(variogram, self.VariogramModel)
        if self.VParametersList is None:
            self.VParametersList = self._INITVariogramModel(**kwargs)
        
        # 调整输入数据
        if self.GType == 'PROJCS':
            self.Center = (self.Points.min(axis = 0) + self.Points.max(axis = 0)) * 0.5
            self.AnisotropyScaling = AnisotropyScaling
            self.AnisotropyAngle = AnisotropyAngle
            self.DistanceMethod = cdist
        else:
            # 方便后期优化
            self.Center = np.array([0,0])
            self.AnisotropyScaling = 1.0
            self.AnisotropyAngle = 0.0
            self.DistanceMethod = GreatCircleDistance
        
        self.AdjustPoints = AdjustAnisotropy(self.Points, self.Center, 
                                             [self.AnisotropyScaling], 
                                             [self.AnisotropyAngle])
        self.XYs = AdjustAnisotropy(self.XYs, self.Center,
                                    [self.AnisotropyScaling], 
                                    [self.AnisotropyAngle])
        
    def _INITVariogramModel(self, **kwargs):
        '''初始化参数'''
        
        if 'NLags' in kwargs:
            NLags = kwargs['NLags']
            initialize.ValueType(NLags, 'pint')
        else:
            NLags = 6
            
        if 'Weight' in kwargs:
            Weight = ToNumericArray(kwargs['Weight']).flatten().astype(bool)[0]
        else:
            Weight = False

        Lags, SEMI = INITVariogramModel(self.Points, self.Values, NLags, self.GType)
        
        # 为求解自动参数准备
        if self.VariogramModel == "Linear":
            X0 = [np.ptp(SEMI) / np.ptp(Lags), np.min(SEMI)]
            BNDS = ([0.0, 0.0], [np.inf, np.max(SEMI)])
        elif self.VariogramModel == "Power":
            X0 = [np.ptp(SEMI) / np.ptp(Lags), 1.1, np.min(SEMI)]
            BNDS = ([0.0, 0.001, 0.0], [np.inf, 1.999, np.max(SEMI)])
        else:
            X0 = [np.ptp(SEMI), 0.25 * np.max(Lags),  np.min(SEMI)]
            BNDS = ([0.0, 0.0, 0.0], [10.0 * np.max(SEMI), np.max(Lags), np.max(SEMI)])
        
        # 最小二乘法求解默认参数
        def _VariogramResiduals(Params, X, Y, Weight):
            if Weight:
                Weight = 1.0 / (1.0 + np.exp(-2.1972 / (0.1 * np.ptp(X)) * (0.7 * np.ptp(X) + np.min(X) - x))) + 1 
            else:
                Weight = 1
            return (self.VariogramFUN(X, *Params) - Y) * Weight

        RES = least_squares(_VariogramResiduals, X0, bounds = BNDS, loss = "soft_l1",
                            args = (Lags, SEMI, Weight))
                            
        return RES.x
    
    def _GetKrigingMatrix(self):
        """获取克里金矩阵"""
            
        LDs = self.DistanceMethod(self.AdjustPoints, self.AdjustPoints)
        
        A = -self.VariogramFUN(LDs, *self.VParametersList)
        A = np.pad(A, (0, 1), constant_values = 1)
        # 填充主对角线
        np.fill_diagonal(A, 0.0)
 
        return  A
    
    def _UKExec(self, A, LDs, SearchRadius):
        """泛克里金求解"""
        Args = LDs.argsort(axis = 1)[:,:SearchRadius]
        Values = self.Values[Args.T].T
        
        # A 的逆矩阵
        AInv = inv(A)
        B = -self.VariogramFUN(LDs, *self.VParametersList)
        B[np.abs(LDs) <= self.eps] = 0.0
        B = np.pad(B, ((0,0),(0,1)), constant_values = 1)

        X = np.dot(B, AInv)
        
        B = B[np.ogrid[:len(B)], Args.T].T
        X = X[np.ogrid[:len(X)], Args.T].T
        X = X / X.sum(axis = 1, keepdims = True)

        UKResults = np.sum(X * Values, axis = 1), np.sum((X * -B), axis = 1)

        return UKResults
    
    def _OKExec(self, A, LDs, SearchRadius):
        """普通克里金求解"""
        Args = LDs.argsort(axis = 1)[:,:SearchRadius]
        LDs = LDs[np.ogrid[:len(LDs)], Args.T].T

        B = -self.VariogramFUN(LDs, *self.VParametersList)
        B[np.abs(LDs) <= self.eps] = 0.0
        B = np.pad(B, ((0,0),(0,1)), constant_values = 1)
 
        OKResults = np.zeros([2, len(LDs)])

        for i, b in enumerate(B):
            
            BSelector = Args[i] 
            ASelector = np.append(BSelector, len(self.AdjustPoints))
            a = A[ASelector[:, None], ASelector]  
            x = solve(a, b)
            
            OKResults[:, i] = x[:SearchRadius].dot(self.Values[BSelector]), -x.dot(b)

        return OKResults
    

    def Execute(self, SearchRadius = 12, KMethod = 'Ordinary'):
        '''克里金插值'''
        initialize.ValueType(SearchRadius, 'pint')
        SearchRadius = np.min([SearchRadius, len(self.AdjustPoints)])
        
        A = self._GetKrigingMatrix()

        LDs = self.DistanceMethod(self.XYs, self.AdjustPoints)
        
        if KMethod not in ['Universal', 'Ordinary']:
            raise ValueError("Undefined Kriging method. Please select 'Universal' or 'Ordinary'!")
        elif KMethod == 'Universal':
            KResults = self._UKExec(A, LDs, SearchRadius)
        else:
            KResults = self._OKExec(A, LDs, SearchRadius)
            
        NT = namedtuple('Kriging', ['Data', 'SigmaSQ', 'Transform'])
    
        return NT(KResults[0].reshape(self.YLAT, self.XLON),
                  KResults[1].reshape(self.YLAT, self.XLON), self.Transform)

插值应用

在 gma 1.0.13.15 之后的版本可以直接引用。

import gma
import pandas as pd

Data = pd.read_excel("Interpolate.xlsx")
Points = Data.loc[:, ['经度','纬度']].values
Values = Data.loc[:, ['值']].values

# 普通克里金(球面函数模型)插值
KD = Kriging(Points, Values,
             Resolution = 0.05, 
             VariogramModel = 'Spherical', 
             VariogramParameters = None,
             InProjection = 'EPSG:4326').Execute(KMethod = 'Ordinary')
             
# 泛克里金类似,这里不做示例

gma.rasp.WriteRaster(r'.\gma_OKriging.tif',
                     KD.Data,
                     Projection = 'WGS84',
                     Transform = KD.Transform, 
                     DataType = 'Float32')

结果对比

   与 ArcGIS Ordinary Kriging 插值结果(重分类后)对比:

在这里插入图片描述
   与 pykrige 包 Universal Kriging 插值结果(重分类后)对比:
在这里插入图片描述

  • 2
    点赞
  • 43
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 7
    评论
### 回答1: 普通克里金(Ordinary Kriging)是一种地统计学方,用于插值未知空间点的属性值。Python中可以使用各种库和软件包来实现克里金插值。 最常用的Python包是`scipy`和`sklearn`。首先,我们需要导入这两个包: ```python import numpy as np from scipy.linalg import solve from scipy.spatial.distance import cdist from sklearn.preprocessing import normalize ``` 接下来,我们需要定义一些辅助函数,如半方差函数、克里金权重函数和克里金预测函数: ```python def variogram(h, nugget, sill, range): return nugget + sill * (1 - np.exp(-(h / range) ** 2)) def kriging_weights(points, target_point, range): distances = cdist(points, target_point.reshape(1, -1)) r = np.sqrt(np.sum(distances ** 2, axis=1)) weights = variogram(r, 0, 1, range) weights /= np.sum(weights) return weights def kriging_interpolation(points, values, target_point, range): weights = kriging_weights(points, target_point, range) interpolated_value = np.sum(values * weights) return interpolated_value ``` 以上代码定义了半方差函数、克里金权重函数和克里金预测函数,使我们能够计算空间点的插值值。 最后,我们需要提供一些样本点和它们的属性值,以及目标点和范围参数,来进行克里金插值: ```python # 样本点的空间坐标 points = np.array([[0, 0], [1, 1], [2, 2], [3, 3]]) # 样本点的属性值 values = np.array([1, 2, 3, 4]) # 目标点的空间坐标 target_point = np.array([0.5, 0.5]) # 范围参数 range = 1.0 # 进行克里金插值 interpolated_value = kriging_interpolation(points, values, target_point, range) ``` 以上代码中,样本点的坐标和属性值是根据实际情况提供的。范围参数用于调整克里金插值的平滑程度,可以根据具体需求进行调整。 总结来说,实现克里金插值的程序涉及定义半方差函数、克里金权重函数和克里金预测函数,以及提供样本点和目标点的坐标和属性值,并进行插值计算。以上是一个简单的基于Python克里金插值实现示例。 ### 回答2: Python中可以使用普通克里金来实现克里金插值克里金是一种空间插值,通过已知的点数据来估计未知点的数值。 首先,需要导入相关的库,如numpy、scipy等。然后,我们需要准备好已知点的数据,包括位置和数值。 接下来,通过定义克里金插值函数来实现克里金的计算过程。这个函数通常包括以下步骤: 1. 计算半方差函数:根据已知点的位置和数值,通过半方差函数来描述点与点之间的空间相关性。常用的半方差函数包括指数型、高斯型等。 2. 通过最小二乘估计半方差函数的参数:使用已知点的位置和数值,以及半方差函数,通过最小二乘来估计半方差函数的参数。 3. 进行克里金插值:对于未知点,通过半方差函数的参数和已知点的距离,来计算未知点的值。可以使用简单克里金或普通克里金进行插值。 最后,通过调用克里金插值函数,输入需要插值的未知点的位置,即可得到对应的估计值。 值得注意的是,克里金的实现还有其他的细节需要处理,如数据的预处理、确定半方差函数的参数等。此外,还需要针对具体问题对插值结果进行后处理,如检验插值结果的准确性等。 ### 回答3: 普通克里金(Ordinary Kriging)是一种地质插值,用于估计未知位置的属性值。Python中,我们可以使用一些库来实现克里金,如GeostatsPy 和 scikit-gstat。 首先,我们需要准备用于插值的数据集。数据集应包含已知位置的样本点及其属性值。可以使用numpy库来创建这些样本点的坐标数组和属性值数组。 接下来,我们可以使用GeostatsPy库中的`OKModel()`函数来创建一个克里金模型对象。这个函数需要输入克里金变异函数的参数(如方差、粗糙度和变异范围)。 然后,我们可以使用`OK()`函数来进行克里金插值。这个函数需要输入样本点的坐标和属性值,以及待插值位置的坐标。插值结果将返回一个数组。 另一种实现克里金的方是使用scikit-gstat库。首先,我们需要使用`GSTools()`函数创建一个空的克里金模型对象。然后,我们可以使用`Simple()`函数为模型对象添加变异函数和模型参数。 接下来,我们使用`Krige()`函数来进行克里金插值。这个函数需要输入样本点的坐标和属性值,以及待插值位置的坐标。插值结果将返回一个数组。 最后,我们可以使用matplotlib和numpy库来对插值结果进行可视化,以便更好地理解。 总之,使用Python实现克里金需要准备数据集、调用库函数创建克里金模型对象,然后使用函数进行插值。最后,我们可以使用可视化库来展示插值结果。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

洛的地理研学

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值