libpysal.graph.Graph

class libpysal.graph.Graph(adjacency, transformation='O', is_sorted=False)[source]

Graph class encoding spatial weights matrices

The Graph is currently experimental and its API is incomplete and unstable.

__init__(adjacency, transformation='O', is_sorted=False)[source]

Weights base class based on adjacency list

It is recommenced to use one of the from_* or build_* constructors rather than invoking __init__ directly.

Each observation needs to be present in the focal, at least as a self-loop with a weight 0.

Parameters:
adjacencypandas.Series

A MultiIndexed pandas.Series with "focal" and "neigbor" levels encoding adjacency, and values encoding weights. By convention, isolates are encoded as self-loops with a weight 0.

transformationstr, default “O”

weights transformation used to produce the table.

  • O – Original

  • B – Binary

  • R – Row-standardization (global sum \(=n\))

  • D – Double-standardization (global sum \(=1\))

  • V – Variance stabilizing

  • C – Custom

is_sortedbool, default False

adjacency capturing the graph needs to be canonically sorted to initialize the class. The MultiIndex needs to be ordered i–>j on both focal and neighbor levels according to the order of ids in the original data from which the Graph is created. Sorting is performed by default based on the order of unique values in the focal level. Sorting needs to be reflected in both the values of the MultiIndex and also the underlying MultiIndex.codes. Set is_sorted=True to skip this step if the adjacency is already canonically sorted and you are certain about it.

Methods

__init__(adjacency[, transformation, is_sorted])

Weights base class based on adjacency list

aggregate(func)

Aggregate weights within a neighbor set

apply(y, func, **kwargs)

Apply a reduction across the neighbor sets

assign_self_weight([weight])

Assign values to edges representing self-weight.

asymmetry([intrinsic])

Asymmetry check.

build_block_contiguity(regimes)

Generate Graph from block contiguity (regime neighbors)

build_contiguity(geometry[, rook, ...])

Generate Graph from geometry based on contiguity

build_distance_band(data, threshold[, ...])

Generate Graph from geometry based on a distance band

build_fuzzy_contiguity(geometry[, ...])

Generate Graph from fuzzy contiguity

build_h3(ids[, order, weight])

Generate Graph from indices of H3 hexagons.

build_kernel(data[, kernel, k, bandwidth, ...])

Generate Graph from geometry data based on a kernel function

build_knn(data, k[, metric, p, coplanar])

Generate Graph from geometry data based on k-nearest neighbors search

build_raster_contiguity(da[, rook, z_value, ...])

Generate Graph from xarray.DataArray raster object

build_spatial_matches(data, k[, metric, ...])

Match locations in one dataset to at least n_matches locations in another (possibly identical) dataset by minimizing the total distance between matched locations.

build_travel_cost(df, network, threshold[, ...])

Generate a Graph based on shortest travel costs from a pandana.Network

build_triangulation(data[, method, ...])

Generate Graph from geometry based on triangulation

copy([deep])

Make a copy of this Graph's adjacency table and transformation

describe(y[, q, statistics])

Describe the distribution of y values within the neighbors of each node.

difference(right)

Provide the set difference between the graph on the left and the graph on the right.

eliminate_zeros()

Remove graph edges with zero weight

equals(right)

Check that two graphs are identical.

explore(gdf[, focal, nodes, color, ...])

Plot graph as an interactive Folium Map

from_W(w)

Create an experimental Graph from libpysal.weights.W object

from_adjacency(adjacency[, focal_col, ...])

Create a Graph from a pandas DataFrame formatted as an adjacency list

from_arrays(focal_ids, neighbor_ids, weight, ...)

Generate Graph from arrays of indices and weights of the same length

from_dicts(neighbors[, weights])

Generate Graph from dictionaries of neighbors and weights

from_sparse(sparse[, ids])

Convert a scipy.sparse array to a PySAL Graph object.

from_weights_dict(weights_dict)

Generate Graph from a dict of dicts

generate_da(y)

Creates xarray.DataArray object from passed data aligned with the Graph.

higher_order([k, shortest_path, diagonal, ...])

Contiguity weights object of order \(k\).

intersection(right)

Returns a binary Graph, that includes only those neighbor pairs that exist in both left and right.

intersects(right)

Returns True if left and right share at least one link, irrespective of weights value.

isomorphic(right)

Check that two graphs are isomorphic.

issubgraph(right)

Return True if every link in the left Graph also occurs in the right Graph.

lag(y[, categorical, ties])

Spatial lag operator

plot(gdf[, focal, nodes, color, edge_kws, ...])

Plot edges and nodes of the Graph

subgraph(ids)

Returns a subset of Graph containing only nodes specified in ids

summary([asymmetries])

Summary of the Graph properties

symmetric_difference(right)

Filter out links that are in both left and right Graph objects.

to_W()

Convert Graph to a libpysal.weights.W object

to_gal(path)

Save Graph to a GAL file

to_gwt(path)

Save Graph to a GWT file

to_networkx()

Convert Graph to a networkx graph.

to_parquet(path, **kwargs)

Save Graph to a Apache Parquet

transform(transformation)

Transformation of weights

union(right)

Provide the union of two Graph objects, collecing all links that are in either graph.

Attributes

adjacency

Return a copy of the adjacency list

cardinalities

Number of neighbors for each observation

component_labels

Get component labels per observation

index_pairs

Return focal-neighbor index pairs

isolates

Index of observations with no neighbors

n

Number of observations.

n_components

Get a number of connected components

n_edges

Number of edges.

n_nodes

Number of nodes.

neighbors

Get neighbors dictionary

nonzero

Number of nonzero weights.

pct_nonzero

Percentage of nonzero weights.

sparse

Return a scipy.sparse array (CSR)

unique_ids

Unique IDs used in the Graph

weights

Get weights dictionary

property adjacency

Return a copy of the adjacency list

Returns:
pandas.Series

Underlying adjacency list

aggregate(func)[source]

Aggregate weights within a neighbor set

Apply a custom aggregation function to a group of weights of the same focal geometry.

Parameters:
funccallable()

A callable accepted by pandas groupby.agg method

Returns:
pd.Series

Aggregated weights

apply(y, func, **kwargs)[source]

Apply a reduction across the neighbor sets

Applies func over groups of y defined by neighbors for each focal.

Parameters:
yarray_like

array of values to be grouped. Can be 1-D or 2-D and will be coerced to a pandas object

funcfunction, str, list, dict or None

Function to use for aggregating the data passed to pandas GroupBy.apply.

Returns:
Series | DataFrame

pandas object indexed by unique_ids

assign_self_weight(weight=1)[source]

Assign values to edges representing self-weight.

The value for each focal == neighbor location in the graph is set to weight.

Parameters:
weightfloat | array_like

Defines the value(s) to which the weight representing the relationship with itself should be set. If a constant is passed then each self-weight will get this value (default is 1). An array of length Graph.n can be passed to set explicit values to each self-weight (assumed to be in the same order as original data).

Returns:
Graph

A new Graph with added self-weights.

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> contiguity_weights = contiguity.assign_self_weight(0.5)
>>> contiguity_weights.adjacency
focal          neighbor
Staten Island  Staten Island    0.5
Queens         Queens           0.5
               Brooklyn         1.0
               Manhattan        1.0
               Bronx            1.0
Brooklyn       Queens           1.0
               Brooklyn         0.5
               Manhattan        1.0
Manhattan      Queens           1.0
               Brooklyn         1.0
               Manhattan        0.5
               Bronx            1.0
Bronx          Queens           1.0
               Manhattan        1.0
               Bronx            0.5
Name: weight, dtype: float64
asymmetry(intrinsic=True)[source]

Asymmetry check.

Parameters:
intrinsicbool, optional

Default is True. Intrinsic symmetry is defined as:

\[w_{i,j} == w_{j,i}\]

If intrinsic is False symmetry is defined as:

\[i \in N_j \ \& \ j \in N_i\]

where \(N_j\) is the set of neighbors for \(j\), e.g., True requires equality of the weight to consider two links equal, False requires only a presence of a link with a non-zero weight.

Returns:
pandas.Series

A Series of (i,j) pairs of asymmetries sorted ascending by the focal observation (index value), where i is the focal and j is the neighbor. An empty Series is returned if no asymmetries are found.

classmethod build_block_contiguity(regimes)[source]

Generate Graph from block contiguity (regime neighbors)

Block contiguity structures are relevant when defining neighbor relations based on membership in a regime. For example, all counties belonging to the same state could be defined as neighbors, in an analysis of all counties in the US.

Parameters:
regimeslist-like

list-like of regimes. If pandas.Series, its index is used to encode Graph. Otherwise a default RangeIndex is used.

Returns:
Graph

libpysal.graph.Graph encoding block contiguity

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> france = gpd.read_file(get_path('geoda guerry')).set_index('Dprmnt')

In the GeoDa Guerry dataset, the Region column reflects the region (North, East, West, South or Central) to which each department belongs.

>>> france[['Region', 'geometry']].head()
             Region                                           geometry
Dprtmnt
Ain               E  POLYGON ((801150.000 2092615.000, 800669.000 2...
Aisne             N  POLYGON ((729326.000 2521619.000, 729320.000 2...
Allier            C  POLYGON ((710830.000 2137350.000, 711746.000 2...
Basses-Alpes      E  POLYGON ((882701.000 1920024.000, 882408.000 1...
Hautes-Alpes      E  POLYGON ((886504.000 1922890.000, 885733.000 1...

Using the "Region" labels as regimes then identifies all departments within the region as neighbors.

>>> block_contiguity = graph.Graph.build_block_contiguity(france['Region'])
>>> block_contiguity.adjacency
focal   neighbor
Ain     Basses-Alpes       1
        Hautes-Alpes       1
        Aube               1
        Cote-d'Or          1
        Doubs              1
                          ..
Vienne  Mayenne            1
        Morbihan           1
        Basses-Pyrenees    1
        Deux-Sevres        1
        Vendee             1
Name: weight, Length: 1360, dtype: int32
classmethod build_contiguity(geometry, rook=True, by_perimeter=False, strict=False)[source]

Generate Graph from geometry based on contiguity

Contiguity builder assumes that all geometries are forming a coverage, i.e. a non-overlapping mesh and neighbouring geometries share only points or segments of their exterior boundaries. In practice, build_contiguity is capable of creating a Graph of partially overlapping geometries when strict=False, by_perimeter=False, but that would not strictly follow the definition of queen or rook contiguity.

Parameters:
geometryarray_like of shapely.Geometry objects

Could be geopandas.GeoSeries or geopandas.GeoDataFrame, in which case the resulting Graph is indexed by the original index. If an array of shapely.Geometry objects is passed, Graph will assume a RangeIndex.

rookbool, optional

Contiguity method. If True, two geometries are considered neighbours if they share at least one edge. If False, two geometries are considered neighbours if they share at least one vertex. By default True

by_perimeterbool, optional

If True, weight represents the length of the shared boundary between adjacent units, by default False. For row-standardized version of perimeter weights, use Graph.build_contiguity(gdf, by_perimeter=True).transform("r").

strictbool, optional

Use the strict topological method. If False, the contiguity is determined based on shared coordinates or coordinate sequences representing edges. This assumes geometry coverage that is topologically correct. This method is faster but can miss some relations. If True, the contiguity is determined based on geometric relations that do not require precise topology. This method is slower but will result in correct contiguity even if the topology of geometries is not optimal. By default False.

Returns:
Graph

libpysal.graph.Graph encoding contiguity weights

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> contiguity.adjacency
focal          neighbor
Staten Island  Staten Island    0
Queens         Brooklyn         1
               Manhattan        1
               Bronx            1
Brooklyn       Queens           1
               Manhattan        1
Manhattan      Queens           1
               Brooklyn         1
               Bronx            1
Bronx          Queens           1
               Manhattan        1
Name: weight, dtype: int64

Weight by perimeter instead of binary weights:

>>> contiguity_perimeter = graph.Graph.build_contiguity(nybb, by_perimeter=True)
>>> contiguity_perimeter.adjacency
focal          neighbor
Staten Island  Staten Island        0.000000
Queens         Brooklyn         50867.502055
               Manhattan          103.745207
               Bronx                5.777002
Brooklyn       Queens           50867.502055
               Manhattan         5736.546898
Manhattan      Queens             103.745207
               Brooklyn          5736.546898
               Bronx             5258.300879
Bronx          Queens               5.777002
               Manhattan         5258.300879
Name: weight, dtype: float64
classmethod build_distance_band(data, threshold, binary=True, alpha=-1.0, kernel=None, bandwidth=None)[source]

Generate Graph from geometry based on a distance band

Parameters:
datanumpy.ndarray, geopandas.GeoSeries, geopandas.GeoDataFrame

geometries containing locations to compute the delaunay triangulation. If a geopandas object with Point geometry is provided, the .geometry attribute is used. If a numpy.ndarray with shapely geometry is used, then the coordinates are extracted and used. If a numpy.ndarray of a shape (2,n) is used, it is assumed to contain x, y coordinates.

thresholdfloat

distance band

binarybool, optional

If True \(w_{ij}=1\) if \(d_{i,j}<=threshold\), otherwise \(w_{i,j}=0\). If False \(wij=dij^{alpha}\), by default True.

alphafloat, optional

distance decay parameter for weight (default -1.0) if alpha is positive the weights will not decline with distance. Ignored if binary=True or kernel is not None.

kernelstr, optional

kernel function to use in order to weight the output graph. See Graph.build_kernel() for details. Ignored if binary=True.

bandwidthfloat (default: None)

distance to use in the kernel computation. Should be on the same scale as the input coordinates. Ignored if binary=True or kernel=None.

Returns:
Graph

libpysal.graph.Graph encoding distance band weights

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]

Note that the method requires point geometry (or an array of coordinates representing points) as an input.

The threshold distance is in the units of the geometry projection. You can check it using the nybb.crs property.

>>> distance_band = graph.Graph.build_distance_band(nybb.centroid, 45000)
>>> distance_band.adjacency
focal          neighbor
Staten Island  Staten Island    0
Queens         Brooklyn         1
Brooklyn       Queens           1
Manhattan      Bronx            1
Bronx          Manhattan        1
Name: weight, dtype: int64

The larger threshold yields more neighbors.

>>> distance_band = graph.Graph.build_distance_band(nybb.centroid, 110000)
>>> distance_band.adjacency
focal          neighbor
Staten Island  Queens           1
               Brooklyn         1
               Manhattan        1
Queens         Staten Island    1
               Brooklyn         1
               Manhattan        1
               Bronx            1
Brooklyn       Staten Island    1
               Queens           1
               Manhattan        1
               Bronx            1
Manhattan      Staten Island    1
               Queens           1
               Brooklyn         1
               Bronx            1
Bronx          Queens           1
               Brooklyn         1
               Manhattan        1
Name: weight, dtype: int64

Instead of binary weights you can use inverse distance.

>>> distance_band = graph.Graph.build_distance_band(
...     nybb.centroid,
...     45000,
...     binary=False,
... )
>>> distance_band.adjacency
focal          neighbor
Staten Island  Staten Island    0.000000
Queens         Brooklyn         0.000024
Brooklyn       Queens           0.000024
Manhattan      Bronx            0.000026
Bronx          Manhattan        0.000026
Name: weight, dtype: float64

Or specify the kernel function to derive weight from the distance.

>>> distance_band = graph.Graph.build_distance_band(
...     nybb.centroid,
...     45000,
...     binary=False,
...     kernel='bisquare',
...     bandwidth=60000,
... )
>>> distance_band.adjacency
focal          neighbor
Staten Island  Staten Island    0.000000
Queens         Brooklyn         0.232079
Brooklyn       Queens           0.232079
Manhattan      Bronx            0.309825
Bronx          Manhattan        0.309825
Name: weight, dtype: float64
classmethod build_fuzzy_contiguity(geometry, tolerance=None, buffer=None, predicate='intersects', **kwargs)[source]

Generate Graph from fuzzy contiguity

Fuzzy contiguity relaxes the notion of contiguity neighbors for the case of geometry collections that violate the condition of planar enforcement. It handles three types of conditions present in such collections that would result in missing links when using the regular contiguity methods.

The first are edges for nearby polygons that should be shared, but are digitized separately for the individual polygons and the resulting edges do not coincide, but instead the edges intersect. This case can also be covered by build_contiguty with the strict=False parameter.

The second case is similar to the first, only the resultant edges do not intersect but are “close”. The optional buffering of geometry then closes the gaps between the polygons and a resulting intersection is encoded as a link.

The final case arises when one polygon is “inside” a second polygon but is not encoded to represent a hole in the containing polygon.

It is also possible to create a contiguity based on a custom spatial predicate.

Parameters:
geomsarray_like of shapely.Geometry objects

Could be geopandas.GeoSeries or geopandas.GeoDataFrame, in which case the resulting Graph is indexed by the original index. If an array of shapely.Geometry objects is passed, Graph will assume a RangeIndex.

tolerancefloat, optional

The percentage of the length of the minimum side of the bounding rectangle for the geoms to use in determining the buffering distance. Either tolerance or buffer may be specified but not both. By default None.

bufferfloat, optional

Exact buffering distance in the units of geoms.crs. Either tolerance or buffer may be specified but not both. By default None.

predicatestr, optional

The predicate to use for determination of neighbors. Default is ‘intersects’. If None is passed, neighbours are determined based on the intersection of bounding boxes. See the documentation of geopandas.GeoSeries.sindex.query for allowed predicates.

**kwargs

Keyword arguments passed to geopandas.GeoSeries.buffer.

Returns:
Graph

libpysal.graph.Graph encoding fuzzy contiguity

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]

Example using the default parameters:

>>> fuzzy_contiguity = graph.Graph.build_fuzzy_contiguity(nybb)
>>> fuzzy_contiguity
<Graph of 5 nodes and 10 nonzero edges indexed by
 ['Staten Island', 'Queens', 'Brooklyn', 'Manhattan', 'Bronx']>

Example using the tolerance of 0.05:

>>> fuzzy_contiguity = graph.Graph.build_fuzzy_contiguity(nybb, tolerance=0.05)
>>> fuzzy_contiguity
<Graph of 5 nodes and 12 nonzero edges indexed by
 ['Staten Island', 'Queens', 'Brooklyn', 'Manhattan', 'Bronx']>

Example using a buffer of 10000 feet (CRS of nybb is in feet):

>>> fuzzy_contiguity = graph.Graph.build_fuzzy_contiguity(nybb, buffer=10000)
>>> fuzzy_contiguity
<Graph of 5 nodes and 14 nonzero edges indexed by
 ['Staten Island', 'Queens', 'Brooklyn', 'Manhattan', 'Bronx']>
classmethod build_h3(ids, order=1, weight='distance')[source]

Generate Graph from indices of H3 hexagons.

Encode a graph from a set of H3 hexagons. The graph is generated by considering the H3 hexagons as nodes and connecting them based on their contiguity. The contiguity is defined by the order parameter, which specifies the number of steps to consider as neighbors. The weight parameter defines the type of weight to assign to the edges.

Requires the h3 library.

Parameters:
idsarray_like

Array of H3 IDs encoding focal geometries

orderint, optional

Order of contiguity, by default 1

weightstr, optional

Type of weight. Options are:

  • distance: raw topological distance between cells

  • binary: 1 for neighbors, 0 for non-neighbors

  • inverse: 1 / distance between cells

By default “distance”.

Returns:
Graph

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> from tobler.util import h3fy
>>> gdf = gpd.read_file(get_path("geoda guerry"))
>>> h3 = h3fy(gdf, resolution=4)
>>> h3.head()
                                                          geometry
hex_id
841f94dffffffff  POLYGON ((609346.657 2195981.397, 604556.817 2...
841fa67ffffffff  POLYGON ((722074.162 2561038.244, 717442.706 2...
84186a3ffffffff  POLYGON ((353695.287 2121176.341, 329999.974 2...
8418609ffffffff  POLYGON ((387747.482 2509794.492, 364375.032 2...
8418491ffffffff  POLYGON ((320872.289 1846157.662, 296923.464 1...
>>> h3_contiguity = graph.Graph.build_h3(h3.index)
>>> h3_contiguity
<Graph of 320 nodes and 1740 nonzero edges indexed by
 ['841f94dffffffff', '841fa67ffffffff', '84186a3ffffffff', ...]>
classmethod build_kernel(data, kernel='gaussian', k=None, bandwidth=None, metric='euclidean', p=2, coplanar='raise')[source]

Generate Graph from geometry data based on a kernel function

Parameters:
datanumpy.ndarray, geopandas.GeoSeries, geopandas.GeoDataFrame

geometries over which to compute a kernel. If a geopandas object with Point geoemtry is provided, the .geometry attribute is used. If a numpy.ndarray with shapely geoemtry is used, then the coordinates are extracted and used. If a numpy.ndarray of a shape (2,n) is used, it is assumed to contain x, y coordinates. If metric=”precomputed”, data is assumed to contain a precomputed distance metric.

kernelstr or callable() (default: ‘gaussian’)

kernel function to apply over the distance matrix computed by metric. The following kernels are supported:

  • "triangular":

  • "parabolic":

  • "gaussian":

  • "bisquare":

  • "cosine":

  • 'boxcar'/discrete: all distances less than bandwidth are 1, and all other distances are 0

  • "identity"/None : do nothing, weight similarity based on raw distance

  • callable : a user-defined function that takes the distance vector and the bandwidth and returns the kernel: kernel(distances, bandwidth)

kint (default: None)

number of nearest neighbors used to truncate the kernel. This is assumed to be constant across samples. If None, no truncation is conduted.

bandwidthfloat (default: None)

distance to use in the kernel computation. Should be on the same scale as the input coordinates.

metricstr or callable() (default: ‘euclidean’)

distance function to apply over the input coordinates. Supported options depend on whether or not scikit-learn is installed. If so, then any distance function supported by scikit-learn is supported here. Otherwise, only euclidean, minkowski, and manhattan/cityblock distances are admitted.

pint (default: 2)

parameter for minkowski metric, ignored if metric != “minkowski”.

coplanar: str, optional (default “raise”)

Method for handling coplanar points when k is not None. Options are 'raise' (raising an exception when coplanar points are present), 'jitter' (randomly displace coplanar points to produce uniqueness), & 'clique' (induce fully-connected sub cliques for coplanar points).

Returns:
Graph

libpysal.graph.Graph encoding kernel weights

classmethod build_knn(data, k, metric='euclidean', p=2, coplanar='raise')[source]

Generate Graph from geometry data based on k-nearest neighbors search

Parameters:
datanumpy.ndarray, geopandas.GeoSeries, geopandas.GeoDataFrame

geometries over which to compute a kernel. If a geopandas object with Point geoemtry is provided, the .geometry attribute is used. If a numpy.ndarray with shapely geoemtry is used, then the coordinates are extracted and used. If a numpy.ndarray of a shape (2,n) is used, it is assumed to contain x, y coordinates.

kint

number of nearest neighbors.

metricstr or callable() (default: ‘euclidean’)

distance function to apply over the input coordinates. Supported options depend on whether or not scikit-learn is installed. If so, then any distance function supported by scikit-learn is supported here. Otherwise, only euclidean, minkowski, and manhattan/cityblock distances are admitted.

pint (default: 2)

parameter for minkowski metric, ignored if metric != “minkowski”.

coplanar: str, optional (default “raise”)

Method for handling coplanar points. Options include 'raise' (raising an exception when coplanar points are present), 'jitter' (randomly displace coplanar points to produce uniqueness), & 'clique' (induce fully-connected sub cliques for coplanar points).

Returns:
Graph

libpysal.graph.Graph encoding KNN weights

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index('BoroName')
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
>>> knn3 = graph.Graph.build_knn(nybb.centroid, k=3)
>>> knn3.adjacency
focal           neighbor
Staten Island   Queens           1
                Brooklyn         1
                Manhattan        1
Queens          Brooklyn         1
                Manhattan        1
                Bronx            1
Brooklyn        Staten Island    1
                Queens           1
                Manhattan        1
Manhattan       Queens           1
                Brooklyn         1
                Bronx            1
Bronx           Queens           1
                Brooklyn         1
                Manhattan        1
Name: weight, dtype: int32

Specifying k=1 identifies the nearest neighbor (note that this can be asymmetrical):

>>> knn1 = graph.Graph.build_knn(nybb.centroid, k=1)
>>> knn1.adjacency
focal          neighbor
Staten Island  Brooklyn     1
Queens         Brooklyn     1
Brooklyn       Queens       1
Manhattan      Bronx        1
Bronx          Manhattan    1
Name: weight, dtype: int32
classmethod build_raster_contiguity(da, rook=False, z_value=None, coords_labels=None, k=1, include_nodata=False, n_jobs=1)[source]

Generate Graph from xarray.DataArray raster object

Create Graph object encoding contiguity of raster cells from xarray.DataArray object. The coordinates are flatten to tuples representing the location of each cell within the raster.

Parameters:
daxarray.DataArray

Input 2D or 3D DataArray with shape=(z, y, x)

rookbool, optional

Contiguity method. If True, two cells are considered neighbours if they share at least one edge. If False, two geometries are considered neighbours if they share at least one vertex. By default True

z_value{int, str, float}, optional

Select the z_value of 3D DataArray with multiple layers. By default None

coords_labelsdict, optional

Pass dimension labels for coordinates and layers if they do not belong to default dimensions, which are (band/time, y/lat, x/lon) e.g. coords_labels = {"y_label": "latitude", "x_label": "longitude", "z_label": "year"} When None, defaults to empty dictionary.

kint, optional

Order of contiguity, this will select all neighbors up to k-th order. Default is 1.

include_nodatabool, optional

If True, missing values will be assumed as non-missing when selecting higher_order neighbors, Default is False

n_jobsint, optional

Number of cores to be used in the sparse weight construction. If -1, all available cores are used. Default is 1. Requires joblib.

Returns:
Graph

libpysal.graph.Graph encoding raster contiguity

classmethod build_spatial_matches(data, k, metric='euclidean', solver=None, allow_partial_match=False, **metric_kwargs)[source]

Match locations in one dataset to at least n_matches locations in another (possibly identical) dataset by minimizing the total distance between matched locations.

Letting \(d_{ij}\) be

\[ \begin{align}\begin{aligned}\text{minimize} \sum_i^n \sum_j^n d_{ij}m_{ij}\\\text{subject to} \sum_j^n m_{ij} >= k \forall i\\ m_{ij} \in {0,1} \forall ij\end{aligned}\end{align} \]
Parameters:
xnumpy.ndarray, geopandas.GeoSeries, geopandas.GeoDataFrame

geometries that need matches. If a geopandas.Geo* object is provided, the .geometry attribute is used. If a numpy.ndarray with a geometry dtype is used, then the coordinates are extracted and used.

ynumpy.ndarray, geopandas.GeoSeries, geopandas.GeoDataFrame (default: None)

geometries that are used as a source for matching. If a geopandas object is provided, the .geometry attribute is used. If a numpy.ndarray with a geometry dtype is used, then the coordinates are extracted and used. If none, matches are made within x.

n_matchesint (default: None)

number of matches

metricstr or callable() (default: ‘euclidean’)

distance function to apply over the input coordinates. Supported options depend on whether or not scikit-learn is installed. If so, then any distance function supported by scikit-learn is supported here. Otherwise, only euclidean, minkowski, and manhattan/cityblock distances are admitted.

solversolver from pulp (default: None)

a solver defined by the pulp optimization library. If no solver is provided, pulp’s default solver will be used. This is generally pulp.COIN(), but this may vary depending on your configuration.

return_mipbool (default: False)

whether or not to return the instance of the pulp.LpProblem. By default, the problem is not returned to the user.

allow_partial_matchbool (default: False)

whether to allow for partial matching. A partial match may have a weight between zero and one, while a “full” match (by default) must have a weight of either zero or one. A partial matching may have a shorter total distance, but will result in a weighted graph.

classmethod build_travel_cost(df, network, threshold, kernel=None, mapping_distance=None)[source]

Generate a Graph based on shortest travel costs from a pandana.Network

Parameters:
dfgeopandas.GeoDataFrame

geodataframe representing observations which are snapped to the nearest node in the pandana.Network. CRS should be the same as the locations of node_x and node_y in the pandana.Network (usually 4326 if network comes from OSM, but sometimes projected to improve snapping quality).

networkpandana.Network

pandana Network object describing travel costs between nodes in the study area. See <https://udst.github.io/pandana/> for more

thresholdint

threshold representing maximum cost distances. This is measured in the same units as the pandana.Network (not influenced by the df.crs in any way). For travel modes with relatively constant speeds like walking or biking, this is usually distance (e.g. meters if the Network is constructed from OSM). For a a multimodal or auto network with variable travel speeds, this is usually some measure of travel time

kernelstr or callable(), optional

kernel transformation applied to the weights. See libpysal.graph.Graph.build_kernel for more information on kernel transformation options. Default is None, in which case the Graph weight is pure distance between focal and neighbor

mapping_distanceint

snapping tolerance passed to pandana.Network.get_node_ids that defines the maximum range at which observations are snapped to nearest nodes in the network. Default is None

Returns:
Graph

Examples

>>> import geodatasets
>>> import geopandas as gpd
>>> import osmnx as ox
>>> import pandana as pdna

Read an example geodataframe:

>>> df = gpd.read_file(geodatasets.get_path("geoda Cincinnati")).to_crs(4326)

Download a walk network using osmnx

>>> osm_graph = ox.graph_from_polygon(df.union_all(), network_type="walk")
>>> nodes, edges = ox.utils_graph.graph_to_gdfs(osm_graph)
>>> edges = edges.reset_index()

Generate a routable pandana network from the OSM nodes and edges

>>> network = pdna.Network(
>>>     edge_from=edges["u"],
>>>     edge_to=edges["v"],
>>>     edge_weights=edges[["length"]],
>>>     node_x=nodes["x"],
>>>     node_y=nodes["y"],)

Use the pandana network to compute shortest paths between gdf centroids and generate a Graph

>>> G = Graph.build_travel_cost(df.set_geometry(df.centroid), network, 500)
>>> G.adjacency.head()
focal  neighbor
0       62          385.609009
        65          309.471985
        115         346.858002
        116           0.000000
        117         333.639008
Name: weight, dtype: float64
classmethod build_triangulation(data, method='delaunay', bandwidth=inf, kernel='boxcar', clip='bounding_box', rook=True, coplanar='raise')[source]

Generate Graph from geometry based on triangulation

Parameters:
datanumpy.ndarray, geopandas.GeoSeries, geopandas.GeoDataFrame

geometries containing locations to compute the delaunay triangulation. If a geopandas object with Point geoemtry is provided, the .geometry attribute is used. If a numpy.ndarray with shapely geoemtry is used, then the coordinates are extracted and used. If a numpy.ndarray of a shape (2,n) is used, it is assumed to contain x, y coordinates.

methodstr, (default “delaunay”)

method of extracting the weights from triangulation. Supports:

  • "delaunay"

  • "gabriel"

  • "relative_neighborhood"

  • "voronoi"

bandwidthfloat, optional

distance to use in the kernel computation. Should be on the same scale as the input coordinates, by default numpy.inf

kernelstr, optional

kernel function to use in order to weight the output graph. See Graph.build_kernel() for details. By default “boxcar”

clipstr (default: ‘bbox’)

Clipping method when method="voronoi". Ignored otherwise. Default is 'bounding_box'. Options are as follows.

None

No clip is applied. Voronoi cells may be arbitrarily larger that the source map. Note that this may lead to cells that are many orders of magnitude larger in extent than the original map. Not recommended.

'bounding_box'

Clip the voronoi cells to the bounding box of the input points.

'convex_hull'

Clip the voronoi cells to the convex hull of the input points.

'alpha_shape'

Clip the voronoi cells to the tightest hull that contains all points (e.g. the smallest alpha shape, using libpysal.cg.alpha_shape_auto()).

shapely.Polygon

Clip to an arbitrary Polygon.

rookbool, optional

Contiguity method when method="voronoi". Ignored otherwise. If True, two geometries are considered neighbours if they share at least one edge. If False, two geometries are considered neighbours if they share at least one vertex. By default True

coplanar: str, optional (default “raise”)

Method for handling coplanar points. Options include 'raise' (raising an exception when coplanar points are present), 'jitter' (randomly displace coplanar points to produce uniqueness), & 'clique' (induce fully-connected sub cliques for coplanar points).

Returns:
Graph

libpysal.graph.Graph encoding triangulation weights

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]

Note that the method requires point geometry (or an array of coordinates representing points) as an input.

>>> triangulation = graph.Graph.build_triangulation(nybb.centroid)
>>> triangulation.adjacency
focal          neighbor
Staten Island  Brooklyn         1
               Manhattan        1
Queens         Brooklyn         1
               Manhattan        1
               Bronx            1
Brooklyn       Staten Island    1
               Queens           1
               Manhattan        1
Manhattan      Staten Island    1
               Queens           1
               Brooklyn         1
               Bronx            1
Bronx          Queens           1
               Manhattan        1
Name: weight, dtype: int64
property cardinalities

Number of neighbors for each observation

Returns:
pandas.Series

Series with a number of neighbors per each observation

property component_labels

Get component labels per observation

Returns:
numpy.array

Array of component labels

copy(deep=True)[source]

Make a copy of this Graph’s adjacency table and transformation

Parameters:
deepbool, optional

Make a deep copy of the adjacency table, by default True

Returns:
Graph

libpysal.graph.Graph as a copy of the original

describe(y, q=None, statistics=None)[source]

Describe the distribution of y values within the neighbors of each node.

Given the graph, computes the descriptive statistics of values within the neighbourhood of each node. Optionally, the values can be limited to a certain quantile range before computing the statistics.

Parameters:
yNDArray[np.float64] | Series

An 1D array of numeric values to be described.

qtuple[float, float] | None, optional

Tuple of percentages for the percentiles to compute. Values must be between 0 and 100 inclusive. When set, values below and above the percentiles will be discarded before computation of the statistics. The percentiles are computed for each neighborhood. By default None.

statisticslist[str] | None

A list of stats functions to compute. If None, compute all available functions - “count”, “mean”, “median”, “std”, “min”, “max”, “sum”, “nunique”, “mode”. By default None.

Returns:
DataFrame

A DataFrame with descriptive statistics.

Notes

The index of values must match the index of the graph.

Weight values do not affect the calculations, only adjacency does.

Returns numpy.nan for all isolates.

The numba package is used extensively in this function to accelerate the computation of statistics. Without numba, these computations may become slow on large data.

eliminate_zeros()[source]

Remove graph edges with zero weight

Eliminates edges with weight == 0 that do not encode an isolate. This is useful to clean-up edges that will make no effect in operations like lag().

Returns:
Graph

subset of Graph with zero-weight edges eliminated

explore(gdf, focal=None, nodes=True, color='black', edge_kws=None, node_kws=None, focal_kws=None, m=None, **kwargs)[source]

Plot graph as an interactive Folium Map

Parameters:
gdfgeopandas.GeoDataFrame

geodataframe used to instantiate to Graph

focallist, optional

subset of focal observations to plot in the map, by default None. If none, all relationships are plotted

nodesbool, optional

whether to display observations as nodes in the map, by default True

colorstr, optional

color applied to nodes and edges, by default “black”

edge_kwsdict, optional

additional keyword arguments passed to geopandas explore function when plotting edges, by default None

node_kwsdict, optional

additional keyword arguments passed to geopandas explore function when plotting nodes, by default None

focal_kwsdict, optional

additional keyword arguments passed to geopandas explore function when plotting focal observations, by default None. Only applicable when passing a subset of nodes with the focal argument

mFolilum.Map, optional

folium map objecto to plot on top of, by default None

**kwargsdict, optional

additional keyword arguments are passed directly to geopandas.explore, when m=None by default None

Returns:
folium.Map

folium map

classmethod from_W(w)[source]

Create an experimental Graph from libpysal.weights.W object

Parameters:
wlibpysal.weights.W
Returns:
Graph

libpysal.graph.Graph from W

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> queen_w = weights.Queen.from_dataframe(nybb, use_index=True)
>>> queen_graph = graph.Graph.from_W(queen_w)
>>> queen_graph
<Graph of 5 nodes and 10 nonzero edges indexed by
 ['Staten Island', 'Queens', 'Brooklyn', 'Manhattan', 'Bronx']>
classmethod from_adjacency(adjacency, focal_col='focal', neighbor_col='neighbor', weight_col='weight')[source]

Create a Graph from a pandas DataFrame formatted as an adjacency list

Parameters:
adjacencypandas.DataFrame

a dataframe formatted as an ajacency list. Should have columns “focal”, “neighbor”, and “weight”, or columns that can be mapped to these (e.g. origin, destination, cost)

focalstr, optional

name of column holding focal/origin index, by default ‘focal’

neighborstr, optional

name of column holding neighbor/destination index, by default ‘neighbor’

weightstr, optional

name of column holding weight values, by default ‘weight’

Returns:
Graph

libpysal.graph.Graph

classmethod from_arrays(focal_ids, neighbor_ids, weight, **kwargs)[source]

Generate Graph from arrays of indices and weights of the same length

The arrays needs to be sorted in a way ensuring that focal_ids.unique() is equal to the index of original observations from which the Graph is being built

Parameters:
focal_indexarray_like

focal indices

neighbor_indexarray_like

neighbor indices

weightarray_like

weights

**kwargs

keyword arguments passed to the class constructor

Returns:
Graph

libpysal.graph.Graph based on arrays

classmethod from_dicts(neighbors, weights=None)[source]

Generate Graph from dictionaries of neighbors and weights

Parameters:
neighborsdict

dictionary of neighbors with the {focal: [neighbor1, neighbor2]} structure

weightsdict, optional

dictionary of neighbors with the {focal: [weight1, weight2]} structure. If None, assumes binary weights.

Returns:
Graph

libpysal.graph.Graph based on dictionaries

Examples

>>> neighbors = {
...     'Africa': ['Asia'],
...     'Asia': ['Africa', 'Europe'],
...     'Australia': [],
...     'Europe': ['Asia'],
...     'North America': ['South America'],
...     'South America': ['North America'],
... }
>>> connectivity = graph.Graph.from_dicts(neighbors)
>>> connectivity.adjacency
focal          neighbor
Africa         Asia             1
Asia           Africa           1
               Europe           1
Australia      Australia        0
Europe         Asia             1
North America  South America    1
South America  North America    1
Name: weight, dtype: float64

You can also specify weights (for example based on the length of the shared border):

>>> weights = {
...     'Africa': [1],
...     'Asia': [0.2, 0.8],
...     'Australia': [],
...     'Europe': [1],
...     'North America': [1],
...     'South America': [1],
... }
>>> connectivity = graph.Graph.from_dicts(neighbors, weights)
>>> connectivity.adjacency
focal          neighbor
Africa         Asia             1.0
Asia           Africa           0.2
               Europe           0.8
Australia      Australia        0.0
Europe         Asia             1.0
North America  South America    1.0
South America  North America    1.0
Name: weight, dtype: float64
classmethod from_sparse(sparse, ids=None)[source]

Convert a scipy.sparse array to a PySAL Graph object.

Parameters:
sparsescipy.sparse array

sparse representation of a graph

idslist-like, default None

list-like of ids for geometries that is mappable to positions from sparse. If None, the positions are used as labels.

Returns:
Graph

libpysal.graph.Graph based on sparse

classmethod from_weights_dict(weights_dict)[source]

Generate Graph from a dict of dicts

Parameters:
weights_dictdictionary of dictionaries

weights dictionary with the {focal: {neighbor: weight}} structure.

Returns:
Graph

libpysal.graph.Graph based on weights dictionary of dictionaries

generate_da(y)[source]

Creates xarray.DataArray object from passed data aligned with the Graph.

Parameters:
yarray_like

flat array that shall be reshaped into a DataArray with dimensionality conforming to Graph

Returns:
xarray.DataArray

instance of xarray.DataArray that can be aligned with the DataArray from which Graph was built

higher_order(k=2, shortest_path=True, diagonal=False, lower_order=False)[source]

Contiguity weights object of order \(k\).

Proper higher order neighbors are returned such that \(i\) and \(j\) are \(k\)-order neighbors if the shortest path from \(i-j\) is of length \(k\).

Parameters:
kint, optional

Order of contiguity. By default 2.

shortest_pathbool, optional

If True, \(i,j\) and \(k\)-order neighbors if the shortest path for \(i,j\) is \(k\). If False, \(i,j\) are k-order neighbors if there is a path from \(i,j\) of length \(k\). By default True.

diagonalbool, optional

If True, keep \(k\)-order (\(i,j\)) joins when \(i==j\). If False, remove \(k\)-order (\(i,j\)) joins when \(i==j\). By default False.

lower_orderbool, optional

If True, include lower order contiguities. If False return only weights of order \(k\). By default False.

Returns:
Graph

higher order weights

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> gdf = gpd.read_file(get_path("geoda guerry"))
>>> contiguity = graph.Graph.build_contiguity(gdf)
>>> contiguity
<Graph of 85 nodes and 420 nonzero edges indexed by
 [0, 1, 2, 3, 4, ...]>
>>> contiguity.higher_order(k=2)
<Graph of 85 nodes and 756 nonzero edges indexed by
 [0, 1, 2, 3, 4, ...]>
>>> contiguity.higher_order(lower_order=True)
<Graph of 85 nodes and 1176 nonzero edges indexed by
 [0, 1, 2, 3, 4, ...]>
property index_pairs

Return focal-neighbor index pairs

Returns:
tuple(Index, Index)

tuple of two aligned pandas.Index objects encoding all edges of the Graph by their nodes

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> focal, neighbor = contiguity.index_pairs
>>> focal
Index(['Staten Island', 'Queens', 'Queens', 'Queens', 'Brooklyn', 'Brooklyn',
       'Manhattan', 'Manhattan', 'Manhattan', 'Bronx', 'Bronx'],
      dtype='object', name='focal')
>>> neighbor
Index(['Staten Island', 'Brooklyn', 'Manhattan', 'Bronx', 'Queens',
       'Manhattan', 'Queens', 'Brooklyn', 'Bronx', 'Queens', 'Manhattan'],
      dtype='object', name='neighbor')
property isolates

Index of observations with no neighbors

Isolates are encoded as a self-loop with the weight == 0 in the adjacency table.

Returns:
pandas.Index

Index with a subset of observations that do not have any neighbor

lag(y, categorical=False, ties='raise')[source]

Spatial lag operator

Constructs spatial lag based on neighbor relations of the graph.

Parameters:
yarray

numpy array with dimensionality conforming to w

categoricalbool

True if y is categorical, False if y is continuous.

ties{‘raise’, ‘random’, ‘tryself’}, optional

Policy on how to break ties when a focal unit has multiple modes for a categorical lag. - ‘raise’: This will raise an exception if ties are encountered to alert the user (Default). - ‘random’: modal label ties Will be broken randomly. - ‘tryself’: check if focal label breaks the tie between label modes. If the focal label does not break the modal tie, the tie will be be broken randomly. If the focal unit has a self-weight, focal label is not used to break any tie, rather any tie will be broken randomly.

Returns:
numpy.ndarray

array of numeric|categorical values for the spatial lag

Examples

>>> import numpy as np
>>> import pandas as pd
>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> aus = gpd.read_file(get_path("abs.australia_states_territories")).set_index(
...     "STE_NAME21"
... )
>>> aus = aus[aus.geometry.notna()]
>>> contiguity = graph.Graph.build_contiguity(aus)

Spatial lag operator for continuous variables.

>>> y = np.arange(9)
>>> contiguity.lag(y)
array([21.,  3.,  9., 13.,  9.,  0.,  9.,  0.,  0.])

You can also perform transformation of weights.

>>> contiguity_r = contiguity.transform("r")
>>> contiguity_r.lag(y)
array([4.2, 1.5, 3. , 2.6, 4.5, 0. , 3. , 0. , 0. ])
property n

Number of observations.

property n_components

Get a number of connected components

Returns:
int

number of components

property n_edges

Number of edges.

property n_nodes

Number of nodes.

property neighbors

Get neighbors dictionary

Returns:
dict

dict of tuples representing neighbors

Notes

It is recommended to work directly with Graph.adjacency() rather than using the Graph.neighbors().

property nonzero

Number of nonzero weights.

property pct_nonzero

Percentage of nonzero weights.

plot(gdf, focal=None, nodes=True, color='k', edge_kws=None, node_kws=None, focal_kws=None, ax=None, figsize=None, limit_extent=False)[source]

Plot edges and nodes of the Graph

Creates a maptlotlib plot based on the topology stored in the Graph and spatial location defined in gdf.

Parameters:
gdfgeopandas.GeoDataFrame

Geometries indexed using the same index as Graph. Geometry types other than points are converted to centroids encoding start and end point of Graph edges.

focalhashable | array_like[hashable] | None, optional

ID or an array-like of IDs of focal geometries whose weights shall be plotted. If None, all weights from all focal geometries are plotted. By default None

nodesbool, optional

Plot nodes as points, by default True

colorstr, optional

The color of all objects, by default “k”

edge_kwsdict, optional

Keyword arguments dictionary to send to LineCollection, which provides fine-grained control over the aesthetics of the edges in the plot. By default None

node_kwsdict, optional

Keyword arguments dictionary to send to ax.scatter, which provides fine-grained control over the aesthetics of the nodes in the plot. By default None

focal_kwsdict, optional

Keyword arguments dictionary to send to ax.scatter, which provides fine-grained control over the aesthetics of the focal nodes in the plot on top of generic node_kws. Values of node_kws are updated from focal_kws. Ignored if focal=None. By default None

axmatplotlib.axes.Axes, optional

Axis on which to plot the weights. If None, a new figure and axis are created. By default None

figsizetuple, optional

figsize used to create a new axis. By default None

limit_extentbool, optional

limit the extent of the axis to the extent of the plotted graph, by default False

Returns:
matplotlib.axes.Axes

Axis with the resulting plot

Notes

If you’d like to overlay the actual geometries from the geopandas.GeoDataFrame, create an axis by plotting the GeoDataFrame and plot the Graph on top.

ax = gdf.plot() gdf_graph.plot(gdf, ax=ax)

property sparse

Return a scipy.sparse array (CSR)

Returns:
scipy.sparse.CSR

sparse representation of the adjacency

subgraph(ids)[source]

Returns a subset of Graph containing only nodes specified in ids

The resulting subgraph contains only the nodes in ids and the edges between them or zero-weight self-loops in case of isolates.

The order of ids reflects a new canonical order of the resulting subgraph. This means ids should be equal to the index of the DataFrame containing data linked to the graph to ensure alignment of sparse representation of subgraph.

Parameters:
idsarray_like

An array of node IDs to be retained

Returns:
Graph

A new Graph that is a subset of the original

Notes

Unlike the implementation in networkx, this creates a copy since Graphs in libpysal are immutable.

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> contiguity.subgraph(["Queens", "Brooklyn", "Manhattan", "Bronx"])
<Graph of 4 nodes and 10 nonzero edges indexed by
 ['Queens', 'Brooklyn', 'Manhattan', 'Bronx']>
summary(asymmetries=False)[source]

Summary of the Graph properties

Returns a GraphSummary object with the statistical attributes summarising the Graph and its basic properties. See the docstring of the GraphSummary for details and all the available attributes.

Parameters:
asymmetriesbool

whether to compute n_asymmetries, which is considerably more expensive than the other attributes. By default False.

Returns:
GraphSummary

a class containing a summary statisitcs about the graph

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> contiguity
<Graph of 5 nodes and 10 nonzero edges indexed by
 ['Staten Island', 'Queens', 'Brooklyn', 'Manhattan', 'Bronx']>
>>> summary = contiguity.summary(asymmetries=True)
>>> summary
Graph Summary Statistics
========================
Graph indexed by:
['Staten Island', 'Queens', 'Brooklyn', 'Manhattan', 'Bronx']
==============================================================
Number of nodes:                                             5
Number of edges:                                            10
Number of connected components:                              2
Number of isolates:                                          1
Number of non-zero edges:                                   10
Percentage of non-zero edges:                           44.00%
Number of asymmetries:                                       0
--------------------------------------------------------------
Cardinalities
==============================================================
Mean:                       2    25%:                        2
Standard deviation:         1    50%:                        2
Min:                        0    75%:                        3
Max:                        3
--------------------------------------------------------------
Weights
==============================================================
Mean:                       1    25%:                        1
Standard deviation:         0    50%:                        1
Min:                        0    75%:                        1
Max:                        1
--------------------------------------------------------------
Sum of weights
==============================================================
S0:                                                         10
S1:                                                         20
S2:                                                        104
--------------------------------------------------------------
Traces
==============================================================
GG:                                                         10
G'G:                                                        10
G'G + GG:                                                   20
>>> summary.s1
20
to_W()[source]

Convert Graph to a libpysal.weights.W object

Returns:
libpysal.weights.W

representation of graph as a weights.W object

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> contiguity.adjacency
focal          neighbor
Staten Island  Staten Island    0
Queens         Brooklyn         1
               Manhattan        1
               Bronx            1
Brooklyn       Queens           1
               Manhattan        1
Manhattan      Queens           1
               Brooklyn         1
               Bronx            1
Bronx          Queens           1
               Manhattan        1
Name: weight, dtype: int64
>>> w = contiguity.to_W()
>>> w.neighbors
{'Bronx': ['Queens', 'Manhattan'],
 'Brooklyn': ['Queens', 'Manhattan'],
 'Manhattan': ['Queens', 'Brooklyn', 'Bronx'],
 'Queens': ['Brooklyn', 'Manhattan', 'Bronx'],
 'Staten Island': []}
to_gal(path)[source]

Save Graph to a GAL file

Graph is serialized to the GAL file format.

Parameters:
pathstr

path to the GAL file

See also

read_gal

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> contiguity.to_gal("contiguity.gal")
to_gwt(path)[source]

Save Graph to a GWT file

Graph is serialized to the GWT file format.

Parameters:
pathstr

path to the GWT file

See also

read_gwt

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb).transform("r")
>>> contiguity.to_gwt("contiguity.gwt")
to_networkx()[source]

Convert Graph to a networkx graph.

If Graph is symmetric, returns nx.Graph, otherwise returns a nx.DiGraph.

Returns:
networkx.Graph | networkx.DiGraph

Representation of libpysal Graph as networkx graph

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> nx_graph = contiguity.to_networkx()
to_parquet(path, **kwargs)[source]

Save Graph to a Apache Parquet

Graph is serialized to the Apache Parquet using the underlying adjacency object stored as a Parquet table and custom metadata containing transformation.

Requires pyarrow package.

Parameters:
pathstr | pyarrow.NativeFile

path or any stream supported by pyarrow

**kwargs

additional keyword arguments passed to pyarrow.parquet.write_table

See also

read_parquet

Examples

>>> import geopandas as gpd
>>> from geodatasets import get_path
>>> nybb = gpd.read_file(get_path("nybb")).set_index("BoroName")
>>> nybb
               BoroCode  ...                                           geometry
BoroName                 ...
Staten Island         5  ...  MULTIPOLYGON (((970217.022 145643.332, 970227....
Queens                4  ...  MULTIPOLYGON (((1029606.077 156073.814, 102957...
Brooklyn              3  ...  MULTIPOLYGON (((1021176.479 151374.797, 102100...
Manhattan             1  ...  MULTIPOLYGON (((981219.056 188655.316, 980940....
Bronx                 2  ...  MULTIPOLYGON (((1012821.806 229228.265, 101278...
[5 rows x 4 columns]
>>> contiguity = graph.Graph.build_contiguity(nybb)
>>> contiguity.to_parquet("contiguity.parquet")
transform(transformation)[source]

Transformation of weights

Parameters:
transformationstr | callable()

Transformation method. The following are valid transformations.

  • B – Binary

  • R – Row-standardization (global sum \(=n\))

  • D – Double-standardization (global sum \(=1\))

  • V – Variance stabilizing

Alternatively, you can pass your own callable passed to self.adjacency.groupby(level=0).transform().

Returns:
Graph

transformed weights

Raises:
ValueError

Value error for unsupported transformation

property unique_ids

Unique IDs used in the Graph

property weights

Get weights dictionary

Returns:
dict

dict of tuples representing weights

Notes

It is recommended to work directly with Graph.adjacency() rather than using the Graph.weights().