/* * GeoTools - The Open Source Java GIS Toolkit * http://geotools.org * * (C) 2014, Open Source Geospatial Foundation (OSGeo) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. */ package org.geotools.process.spatialstatistics.autocorrelation; import java.util.logging.Logger; import org.geotools.data.simple.SimpleFeatureCollection; import org.geotools.process.spatialstatistics.core.SSUtils; import org.geotools.process.spatialstatistics.core.SSUtils.StatEnum; import org.geotools.process.spatialstatistics.core.SpatialEvent; import org.geotools.process.spatialstatistics.core.WeightMatrixBuilder; import org.geotools.process.spatialstatistics.enumeration.DistanceMethod; import org.geotools.process.spatialstatistics.enumeration.SpatialConcept; import org.geotools.process.spatialstatistics.enumeration.StandardizationMethod; import org.geotools.util.logging.Logging; /** * Measures spatial autocorrelation based on feature locations and attribute values using the Global Moran's I statistic. * * @author Minpa Lee, MangoSystem * * @source $URL$ */ public class GlobalMoranIStatisticOperation extends AbstractStatisticsOperation { protected static final Logger LOGGER = Logging.getLogger(GlobalMoranIStatisticOperation.class); public GlobalMoranIStatisticOperation() { // Default Setting this.setDistanceType(DistanceMethod.Euclidean); this.setSpatialConceptType(SpatialConcept.InverseDistance); this.setStandardizationType(StandardizationMethod.None); this.setSelfNeighbors(false); } public MoransI execute(SimpleFeatureCollection inputFeatures, String inputField) { swMatrix = new WeightMatrixBuilder(getSpatialConceptType(), getStandardizationType()); swMatrix.setDistanceMethod(getDistanceType()); swMatrix.setDistanceBandWidth(getDistanceBand()); swMatrix.setSelfNeighbors(isSelfNeighbors()); swMatrix.buildWeightMatrix(inputFeatures, inputField); double zijWSum = 0.0; double wijSum = 0.0; // sum of all weights (wij) double ziSum2 = 0.0; double ziSum4 = 0.0; double sumS1 = 0.0; double sumS2 = 0.0; double n = swMatrix.getEvents().size(); double meanX = swMatrix.sumX / n; for (SpatialEvent source : swMatrix.getEvents()) { double jwijSum = 0.0; double jwjiSum = 0.0; double zi = source.xVal - meanX; ziSum2 += Math.pow(zi, 2.0); ziSum4 += Math.pow(zi, 4.0); for (SpatialEvent target : swMatrix.getEvents()) { if (!isSelfNeighbors() && source.id == target.id) { continue; } double wij = swMatrix.getWeight(source, target); double wji = wij; wij = swMatrix.standardizeWeight(source, wij); wji = swMatrix.standardizeWeight(target, wji); if (wij == 0) { continue; } double zj = target.xVal - meanX; zijWSum += wij * zi * zj; wijSum += wij; jwijSum += wij; jwjiSum += wji; sumS1 += Math.pow(wij + wji, 2.0); } sumS2 += Math.pow(jwijSum + jwjiSum, 2.0); } ziSum2 = ziSum2 / n; // standard deviation ziSum4 = ziSum4 / n; double dExpected = -1.0 / (n - 1.0); if (wijSum <= 0.0) { MoransI moransI = new MoransI(0d, dExpected, 0d); moransI.setConceptualization(getSpatialConceptType()); moransI.setDistanceMethod(getDistanceType()); moransI.setRowStandardization(getStandardizationType()); moransI.setDistanceThreshold(swMatrix.getDistanceBandWidth()); return moransI; } double dObserved = zijWSum / (ziSum2 * wijSum); double s1 = 0.5 * sumS1; double wijSum2 = Math.pow(wijSum, 2.0); double b2 = ziSum4 / (ziSum2 * ziSum2); // sample kurtosis double a = n * ((Math.pow(n, 2.0) - (3.0 * n) + 3.0) * s1 - (n * sumS2) + 3.0 * wijSum2); double b = b2 * ((Math.pow(n, 2.0) - n) * s1 - (2.0 * n * sumS2) + 6.0 * wijSum2); double c = ((n - 1.0) * (n - 2.0) * (n - 3.0) * (Math.pow(wijSum, 2.0))); double zVariance = (a / c) - (b / c) - Math.pow(dExpected, 2.0); // finally build result MoransI moransI = new MoransI(dObserved, dExpected, zVariance); moransI.setConceptualization(getSpatialConceptType()); moransI.setDistanceMethod(getDistanceType()); moransI.setRowStandardization(getStandardizationType()); moransI.setDistanceThreshold(swMatrix.getDistanceBandWidth()); return moransI; } public static final class MoransI { double observedIndex = 0.0; double expectedIndex = 0.0; double zVariance = 0.0; SpatialConcept conceptualization = SpatialConcept.InverseDistance; DistanceMethod distanceMethod = DistanceMethod.Euclidean; StandardizationMethod rowStandardization = StandardizationMethod.None; double distanceThreshold = 0.0; public MoransI() { } public MoransI(double observedIndex, double expectedIndex, double zVariance) { setObservedIndex(observedIndex); setExpectedIndex(expectedIndex); setZVariance(zVariance); } public void setObservedIndex(double observedIndex) { this.observedIndex = observedIndex; } public double getObservedIndex() { return observedIndex; } public void setExpectedIndex(double expectedIndex) { this.expectedIndex = expectedIndex; } public double getExpectedIndex() { return expectedIndex; } public void setZVariance(double zVariance) { this.zVariance = zVariance; } public double getZVariance() { return zVariance; } public double getZScore() { double dX = observedIndex - expectedIndex; double dY = Math.pow(getZVariance(), 0.5); if (dY == 0) return 0.0; return dX / dY; } public double getPValue() { return SSUtils.zProb(this.getZScore(), StatEnum.BOTH); } public SpatialConcept getConceptualization() { return conceptualization; } public void setConceptualization(SpatialConcept conceptualization) { this.conceptualization = conceptualization; } public DistanceMethod getDistanceMethod() { return distanceMethod; } public void setDistanceMethod(DistanceMethod distanceMethod) { this.distanceMethod = distanceMethod; } public StandardizationMethod getRowStandardization() { return rowStandardization; } public void setRowStandardization(StandardizationMethod rowStandardization) { this.rowStandardization = rowStandardization; } public double getDistanceThreshold() { return distanceThreshold; } public void setDistanceThreshold(double distanceThreshold) { this.distanceThreshold = distanceThreshold; } } }