Source code for esda.join_counts

"""
Spatial autocorrelation for binary attributes

"""

__author__ = "Sergio J. Rey <srey@asu.edu> , Luc Anselin <luc.anselin@asu.edu>"

import warnings

import numpy as np
import pandas as pd
from libpysal.weights import W
from scipy.stats import chi2, chi2_contingency

from .crand import njit as _njit
from .tabular import _univariate_handler

__all__ = ["Join_Counts"]

PERMUTATIONS = 999


[docs] class Join_Counts: # noqa: N801 """Binary Join Counts Parameters ---------- y : array binary variable measured across n spatial units w : W | Graph spatial weights instance as W or Graph aligned with y permutations : int number of random permutations for calculation of pseudo-p_values Attributes ---------- y : array original variable w : W original w object permutations : int number of permutations bb : float number of black-black joins ww : float number of white-white joins bw : float number of black-white joins J : float number of joins sim_bb : array (if permutations>0) vector of bb values for permuted samples p_sim_bb : array (if permutations>0) p-value based on permutations (one-sided) null: spatial randomness alternative: the observed bb is greater than under randomness mean_bb : float average of permuted bb values min_bb : float minimum of permuted bb values max_bb : float maximum of permuted bb values sim_bw : array (if permutations>0) vector of bw values for permuted samples p_sim_bw : array (if permutations>0) p-value based on permutations (one-sided) null: spatial randomness alternative: the observed bw is greater than under randomness mean_bw : float average of permuted bw values min_bw : float minimum of permuted bw values max_bw : float maximum of permuted bw values chi2 : float Chi-square statistic on contingency table for join counts chi2_p : float Analytical p-value for chi2 chi2_dof : int Degrees of freedom for analytical chi2 crosstab : DataFrame Contingency table for observed join counts expected : DataFrame Expected contingency table for the null p_sim_chi2 : float p-value for chi2 under random spatial permutations drop_islands : bool (default True) Whether or not to preserve islands as entries in the adjacency list. By default, observations with no neighbors do not appear in the adjacency list. If islands are kept, they are coded as self-neighbors with zero weight. See ``libpysal.weights.to_adjlist()``. Examples -------- >>> import numpy as np >>> import libpysal >>> w = libpysal.weights.lat2W(4, 4) >>> y = np.ones(16) >>> y[0:8] = 0 >>> np.random.seed(12345) >>> from esda.join_counts import Join_Counts >>> jc = Join_Counts(y, w) >>> jc.bb 10.0 >>> jc.bw 4.0 >>> jc.ww 10.0 >>> jc.J 24.0 >>> len(jc.sim_bb) 999 >>> round(jc.p_sim_bb, 3) 0.003 >>> round(np.mean(jc.sim_bb), 3) 5.547 >>> np.max(jc.sim_bb) 10.0 >>> np.min(jc.sim_bb) 0.0 >>> len(jc.sim_bw) 999 >>> jc.p_sim_bw 1.0 >>> np.mean(jc.sim_bw) 12.811811811811811 >>> np.max(jc.sim_bw) 24.0 >>> np.min(jc.sim_bw) 7.0 >>> round(jc.chi2_p, 3) 0.004 >>> jc.p_sim_chi2 0.002 Notes ----- Technical details and derivations can be found in :cite:`cliff81`. """
[docs] def __init__(self, y, w, permutations=PERMUTATIONS, drop_islands=True): y = np.asarray(y).flatten() if isinstance(w, W): w.transform = "b" # ensure we have binary weights self.w = w self.adj_list = self.w.to_adjlist( remove_symmetric=False, drop_islands=drop_islands ) else: w = w.transform("b") self.w = w if drop_islands: self.adj_list = w.adjacency.drop(w.isolates).reset_index() else: self.adj_list = w.adjacency.reset_index() self.y = y self.permutations = permutations s0 = w.s0 if isinstance(w, W) else w._adjacency.sum() self.J = s0 / 2.0 results = self.__calc(self.y) self.bb = results[0] self.ww = results[1] self.bw = results[2] self.chi2 = results[3] self.chi2_p = results[4] self.chi2_dof = results[5] self.autocorr_pos = self.bb + self.ww self.autocorr_neg = self.bw crosstab = pd.DataFrame(data=results[-1]) id_names = ["W", "B"] idx = pd.Index(id_names, name="Focal") crosstab.set_index(idx, inplace=True) crosstab.columns = pd.Index(id_names, name="Neighbor") self.crosstab = crosstab expected = pd.DataFrame(data=results[6]) expected.set_index(idx, inplace=True) expected.columns = pd.Index(id_names, name="Neighbor") self.expected = expected self.calc = self.__calc if permutations: sim = [] i = 0 while i < permutations: try: res = self.__calc(np.random.permutation(self.y)) sim.append(res) i += 1 except ValueError: # expected count of 0 -> inadmissible pass sim_jc = np.array(sim, dtype=object) self.sim_bb = sim_jc[:, 0] self.min_bb = np.min(self.sim_bb) self.mean_bb = np.mean(self.sim_bb) self.max_bb = np.max(self.sim_bb) self.sim_bw = sim_jc[:, 2] self.min_bw = np.min(self.sim_bw) self.mean_bw = np.mean(self.sim_bw) self.max_bw = np.max(self.sim_bw) self.sim_autocurr_pos = sim_jc[:, 0] + sim_jc[:, 1] self.sim_autocurr_neg = sim_jc[:, 2] self.sim_chi2 = sim_jc[:, 3] stat = (self.autocorr_pos - np.mean(self.sim_autocurr_pos)) ** 2 / np.mean( self.sim_autocurr_pos ) ** 2 + ( self.autocorr_neg - np.mean(self.sim_autocurr_neg) ) ** 2 / np.mean(self.sim_autocurr_pos) ** 2 self.sim_autocorr_chi2 = 1 - chi2.cdf(stat, 1) p_sim_bb = self.__pseudop(self.sim_bb, self.bb) p_sim_bw = self.__pseudop(self.sim_bw, self.bw) p_sim_chi2 = self.__pseudop(self.sim_chi2, self.chi2) p_sim_autocorr_pos = self.__pseudop( self.sim_autocurr_pos, self.autocorr_pos ) p_sim_autocorr_neg = self.__pseudop( self.sim_autocurr_neg, self.autocorr_neg ) self.p_sim_bb = p_sim_bb self.p_sim_bw = p_sim_bw self.p_sim_chi2 = p_sim_chi2 self.p_sim_autocorr_pos = p_sim_autocorr_pos self.p_sim_autocorr_neg = p_sim_autocorr_neg
def __calc(self, z): adj_list = self.adj_list zseries = pd.Series( z, index=self.w.id_order if hasattr(self.w, "id_order") else self.w.unique_ids, ) focal = zseries.loc[adj_list.focal].values neighbor = zseries.loc[adj_list.neighbor].values sim = focal == neighbor dif = 1 - sim bb = (focal * sim).sum() / 2 ww = ((1 - focal) * sim).sum() / 2 bw = (focal * dif).sum() / 2 wb = ((1 - focal) * dif).sum() / 2 table = [[ww, wb], [bw, bb]] chi2 = chi2_contingency(table) stat, pvalue, dof, expected = chi2 return (bb, ww, bw + wb, stat, pvalue, dof, expected, np.array(table)) def __pseudop(self, sim, jc): above = sim >= jc larger = sum(above) psim = (larger + 1.0) / (self.permutations + 1.0) return psim @property def _statistic(self): return self.bw
[docs] @classmethod def by_col( cls, df, cols, w=None, inplace=False, pvalue="sim", outvals=None, **stat_kws ): """ Function to compute a Join_Count statistic on a dataframe Parameters ---------- df : pandas.DataFrame a pandas dataframe with a geometry column cols : string or list of string name or list of names of columns to use to compute the statistic w : W | Graph spatial weights instance as W or Graph aligned with the dataframe. If not provided, this is searched for in the dataframe's metadata inplace : bool a boolean denoting whether to operate on the dataframe inplace or to return a series contaning the results of the computation. If operating inplace, the derived columns will be named 'column_join_count' pvalue : string a string denoting which pvalue should be returned. Refer to the the Join_Count statistic's documentation for available p-values outvals : list of strings list of arbitrary attributes to return as columns from the Join_Count statistic **stat_k : dict options to pass to the underlying statistic. For this, see the documentation for the Join_Count statistic. Returns -------- If inplace, None, and operation is conducted on dataframe in memory. Otherwise, returns a copy of the dataframe with the relevant columns attached. """ msg = ( "The `.by_col()` methods are deprecated and will be " "removed in a future version of `esda`." ) warnings.warn(msg, FutureWarning, stacklevel=2) if outvals is None: outvals = [] outvals.extend(["bb", "p_sim_bw", "p_sim_bb"]) pvalue = "" return _univariate_handler( df, cols, w=w, inplace=inplace, pvalue=pvalue, outvals=outvals, stat=cls, swapname="bw", **stat_kws, )
# -------------------------------------------------------------- # Conditional Randomization Function Implementations # -------------------------------------------------------------- @_njit(fastmath=True) def _local_join_count_crand(): raise NotImplementedError