Title: | Simulation Methods for Legislative Redistricting |
---|---|
Description: | Enables researchers to sample redistricting plans from a pre-specified target distribution using Sequential Monte Carlo and Markov Chain Monte Carlo algorithms. The package allows for the implementation of various constraints in the redistricting process such as geographic compactness and population parity requirements. Tools for analysis such as computation of various summary statistics and plotting functionality are also included. The package implements the SMC algorithm of McCartan and Imai (2023) <doi:10.1214/23-AOAS1763>, the enumeration algorithm of Fifield, Imai, Kawahara, and Kenny (2020) <doi:10.1080/2330443X.2020.1791773>, the Flip MCMC algorithm of Fifield, Higgins, Imai and Tarr (2020) <doi:10.1080/10618600.2020.1739532>, the Merge-split/Recombination algorithms of Carter et al. (2019) <arXiv:1911.01503> and DeFord et al. (2021) <doi:10.1162/99608f92.eb30390f>, and the Short-burst optimization algorithm of Cannon et al. (2020) <arXiv:2011.02288>. |
Authors: | Christopher T. Kenny [aut, cre], Cory McCartan [aut], Ben Fifield [aut], Kosuke Imai [aut], Jun Kawahara [ctb], Alexander Tarr [ctb], Michael Higgins [ctb] |
Maintainer: | Christopher T. Kenny <[email protected]> |
License: | GPL (>= 2) |
Version: | 4.2.0 |
Built: | 2024-12-22 05:07:42 UTC |
Source: | https://github.com/alarm-redist/redist |
This function facilitates comparing an existing (i.e., non-simulated) redistricting plan to a set of simulated plans.
add_reference(plans, ref_plan, name = NULL)
add_reference(plans, ref_plan, name = NULL)
plans |
a |
ref_plan |
an integer vector containing the reference plan. It will be
renumbered to 1.. |
name |
a human-readable name for the reference plan. Defaults to the
name of |
a modified redist_plans
object containing the reference plan
Takes a column of a redist_plans
object and averages it across a set of
draws
for each precinct.
avg_by_prec(plans, x, draws = NA)
avg_by_prec(plans, x, draws = NA)
plans |
a |
x |
an expression to average. Tidy-evaluated in |
draws |
which draws to average. |
a vector of length matching the number of precincts, containing the average.
Applies hierarchical clustering to a distance matrix computed from a set of
plans and takes the first k
splits.
classify_plans(dist_mat, k = 8, method = "complete")
classify_plans(dist_mat, k = 8, method = "complete")
dist_mat |
a distance matrix, the output of |
k |
the number of groupings to create |
method |
the clustering method to use. See |
An object of class redist_classified
, which is a list with two
elements:
groups |
A character vector of group labels of the form |
splits |
A list of splits in the hierarchical clustering. Each list element is a list of two mutually exclusive vectors of plan indices, labeled by their group classification, indicating the plans on each side of the split. |
Use plot.redist_classified()
for a visual summary.
This function provides one way to identify the structural differences between two sets of redistricting plans. It operates by computing the precinct co-occurrence matrix (a symmetric matrix where the i,j-th entry is the fraction of plans where precinct i and j are in the same district) for each set, and then computing the first eigenvalue of the difference in these two matrices (in each direction). These eigenvalues identify the important parts of the map.
compare_plans( plans, set1, set2, shp = NULL, plot = "fill", thresh = 0.1, labs = c("Set 1", "Set 2"), ncores = 1 )
compare_plans( plans, set1, set2, shp = NULL, plot = "fill", thresh = 0.1, labs = c("Set 1", "Set 2"), ncores = 1 )
plans |
a redist_plans object |
set1 |
|
set2 |
|
shp |
a shapefile for plotting. |
plot |
If |
thresh |
the value to threshold the eigenvector at in determining the relevant set of precincts for comparison. |
labs |
the names of the panels in the plot. |
ncores |
the number of parallel cores to use. |
The co-occurrence matrices are regularized with a
prior, which is useful for when either
set1
or set2
is small.
If possible, makes a comparison plot according to plot
. Otherwise
returns the following list:
eigen1 |
A numeric vector containing the first eigenvector of
|
eigen2 |
A numeric vector containing the first eigenvector of
|
group_1a , group_1b
|
Lists of precincts. Compared to |
group_2a , group_2b
|
Lists of precincts. Compared to |
cooccur_sep_1 |
The difference in the average co-occurrence of precincts
in |
cooccur_sep_2 |
The difference in the average co-occurrence of precincts
in |
data(iowa) iowa_map <- redist_map(iowa, ndists = 4, pop_tol = 0.05) plans1 <- redist_smc(iowa_map, 100, silent = TRUE) plans2 <- redist_mergesplit(iowa_map, 200, warmup = 100, silent = TRUE) compare_plans(plans1, plans2, shp = iowa_map) compare_plans(plans2, as.integer(draw) <= 20, as.integer(draw) > 20, shp = iowa_map, plot = "line")
data(iowa) iowa_map <- redist_map(iowa, ndists = 4, pop_tol = 0.05) plans1 <- redist_smc(iowa_map, 100, silent = TRUE) plans2 <- redist_mergesplit(iowa_map, 200, warmup = 100, silent = TRUE) compare_plans(plans1, plans2, shp = iowa_map) compare_plans(plans2, as.integer(draw) <= 20, as.integer(draw) > 20, shp = iowa_map, plot = "line")
Currently only implements the competitiveness function in equation (5) of Cho & Liu 2016.
competitiveness(map, rvote, dvote, .data = cur_plans()) redist.competitiveness(plans, rvote, dvote, alpha = 1, beta = 1)
competitiveness(map, rvote, dvote, .data = cur_plans()) redist.competitiveness(plans, rvote, dvote, alpha = 1, beta = 1)
map |
a |
rvote |
A numeric vector with the Republican vote for each precinct. |
dvote |
A numeric vector with the Democratic vote for each precinct. |
.data |
a |
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. Required. |
alpha |
A numeric value for the alpha parameter for the talisman metric |
beta |
A numeric value for the beta parameter for the talisman metric |
Numeric vector with competitiveness scores
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] # old: comp <- redist.competitiveness(plans_05, fl25$mccain, fl25$obama) comp <- compet_talisman(plans_05, fl25, mccain, obama)
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] # old: comp <- redist.competitiveness(plans_05, fl25$mccain, fl25$obama) comp <- compet_talisman(plans_05, fl25, mccain, obama)
The redist_smc()
and redist_mergesplit()
algorithms in this package allow
for additional constraints on the redistricting process to be encoded in the
target distribution for sampling. These functions are provided to specify
these constraints. All arguments are quoted and evaluated in the context of
the data frame provided to redist_constr()
.
add_constr_status_quo(constr, strength, current) add_constr_grp_pow( constr, strength, group_pop, total_pop = NULL, tgt_group = 0.5, tgt_other = 0.5, pow = 1 ) add_constr_grp_hinge( constr, strength, group_pop, total_pop = NULL, tgts_group = c(0.55) ) add_constr_grp_inv_hinge( constr, strength, group_pop, total_pop = NULL, tgts_group = c(0.55) ) add_constr_compet(constr, strength, dvote, rvote, pow = 0.5) add_constr_incumbency(constr, strength, incumbents) add_constr_splits(constr, strength, admin) add_constr_multisplits(constr, strength, admin) add_constr_total_splits(constr, strength, admin) add_constr_pop_dev(constr, strength) add_constr_segregation(constr, strength, group_pop, total_pop = NULL) add_constr_polsby(constr, strength, perim_df = NULL) add_constr_fry_hold( constr, strength, total_pop = NULL, ssdmat = NULL, denominator = 1 ) add_constr_log_st(constr, strength, admin = NULL) add_constr_edges_rem(constr, strength) add_constr_custom(constr, strength, fn)
add_constr_status_quo(constr, strength, current) add_constr_grp_pow( constr, strength, group_pop, total_pop = NULL, tgt_group = 0.5, tgt_other = 0.5, pow = 1 ) add_constr_grp_hinge( constr, strength, group_pop, total_pop = NULL, tgts_group = c(0.55) ) add_constr_grp_inv_hinge( constr, strength, group_pop, total_pop = NULL, tgts_group = c(0.55) ) add_constr_compet(constr, strength, dvote, rvote, pow = 0.5) add_constr_incumbency(constr, strength, incumbents) add_constr_splits(constr, strength, admin) add_constr_multisplits(constr, strength, admin) add_constr_total_splits(constr, strength, admin) add_constr_pop_dev(constr, strength) add_constr_segregation(constr, strength, group_pop, total_pop = NULL) add_constr_polsby(constr, strength, perim_df = NULL) add_constr_fry_hold( constr, strength, total_pop = NULL, ssdmat = NULL, denominator = 1 ) add_constr_log_st(constr, strength, admin = NULL) add_constr_edges_rem(constr, strength) add_constr_custom(constr, strength, fn)
constr |
A |
strength |
The strength of the constraint. Higher values mean a more restrictive constraint. |
current |
The reference map for the status quo constraint. |
group_pop |
A vector of group population |
total_pop |
A vector of total population. Defaults to the population vector used for sampling. |
tgt_group , tgt_other
|
Target group shares for the power-type constraint. |
pow |
The exponent for the power-type constraint. |
tgts_group |
A vector of target group shares for the hinge-type constraint. |
dvote , rvote
|
A vector of Democratic or Republican vote counts |
incumbents |
A vector of unit indices for incumbents. For example, if three incumbents live in the precincts that correspond to rows 1, 2, and 100 of your redist_map, entering incumbents = c(1, 2, 100) would avoid having two or more incumbents be in the same district. |
admin |
A vector indicating administrative unit membership |
perim_df |
A dataframe output from |
ssdmat |
Squared distance matrix for Fryer Holden constraint |
denominator |
Fryer Holden minimum value to normalize by. Default is 1 (no normalization). |
fn |
A function |
All constraints are fed into a Gibbs measure, with coefficients on each
constraint set by the corresponding strength
parameter.
The strength can be any real number, with zero corresponding to no constraint.
Higher and higher strength
values will eventually cause the algorithm's
accuracy and efficiency to suffer. Whenever you use constraints, be sure to
check all sampling diagnostics.
The status_quo
constraint adds a term measuring the variation of
information distance between the plan and the reference, rescaled to [0, 1].
The grp_hinge
constraint takes a list of target group percentages. It
matches each district to its nearest target percentage, and then applies a
penalty of the form , summing across
districts. This penalizes districts which are below their target percentage.
Use
plot.redist_constr()
to visualize the effect of this constraint and
calibrate strength
appropriately.
The grp_inv_hinge
constraint takes a list of target group percentages. It
matches each district to its nearest target percentage, and then applies a
penalty of the form , summing across
districts. This penalizes districts which are above their target percentage.
Use
plot.redist_constr()
to visualize the effect of this constraint and
calibrate strength
appropriately.
The grp_pow
constraint (for expert use) adds a term of the form
, which
encourages districts to have group shares near either
tgt_group
or tgt_other
. Values of strength
depend heavily on the values of these
parameters and especially the pow
parameter.
Use plot.redist_constr()
to visualize the effect of this constraint and
calibrate strength
appropriately.
The compet
constraint encourages competitiveness by applying the grp_pow
constraint with target percentages set to 50%. For convenience, it is
specified with Democratic and Republican vote shares.
The incumbency
constraint adds a term counting the number of districts
containing paired-up incumbents.
Values of strength
should generally be small, given that the underlying values are counts.
The splits
constraint adds a term counting the number of
counties which are split once or more.
Values of strength
should generally be small, given that the underlying values are counts.
The multisplits
constraint adds a term counting the number of
counties which are split twice or more.
Values of strength
should generally be small, given that the underlying values are counts.
The total_splits
constraint adds a term counting the total number of times
each county is split, summed across counties (i.e., counting the number of
excess district-county pairs). Values of strength
should generally be
small, given that the underlying values are counts.
The edges_rem
constraint adds a term counting the number of edges removed from the
adjacency graph. This is only usable with redist_flip()
, as other algorithms
implicitly use this via the compactness
parameter. Values of strength
should
generally be small, given that the underlying values are counts.
The log_st
constraint constraint adds a term counting the log number of spanning
trees. This is only usable with redist_flip()
, as other algorithms
implicitly use this via the compactness
parameter.
The polsby
constraint adds a term encouraging compactness as defined by the
Polsby Popper metric. Values of strength
may be of moderate size.
The fry_hold
constraint adds a term encouraging compactness as defined by the
Fryer Holden metric. Values of strength
should be extremely small, as the
underlying values are massive when the true minimum Fryer Holden denominator is not known.
The segregation
constraint adds a term encouraging segregation among minority groups,
as measured by the dissimilarity index.
The pop_dev
constraint adds a term encouraging plans to have smaller population deviations
from the target population.
The custom
constraint allows the user to specify their own constraint using
a function which evaluates districts one at a time. The provided function
fn
should take two arguments: a vector describing the current plan
assignment for each unit as its first argument, and an integer describing the
district which to evaluate in the second argument. which([plans == distr])
would give the indices of the units that are assigned to a district distr
in any iteration. The function must return a single scalar for each plan -
district combination, where a value of 0 indicates no penalty is applied. If
users want to penalize an entire plan, they can have the penalty function
return a scalar that does not depend on the district. It is important that
fn
not use information from precincts not included in distr
, since in the
case of SMC these precincts may not be assigned any district at all (plan
will take the value of 0 for these precincts). The flexibility of this
constraint comes with an additional computational cost, since the other
constraints are written in C++ and so are more performant.
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) constr <- redist_constr(iowa_map) constr <- add_constr_splits(constr, strength = 1.5, admin = name) constr <- add_constr_grp_hinge(constr, strength = 100, dem_08, tot_08, tgts_group = c(0.5, 0.6)) # encourage districts to have the same number of counties constr <- add_constr_custom(constr, strength = 1000, fn = function(plan, distr) { # notice that we only use information on precincts in `distr` abs(sum(plan == distr) - 99/4) }) print(constr)
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) constr <- redist_constr(iowa_map) constr <- add_constr_splits(constr, strength = 1.5, admin = name) constr <- add_constr_grp_hinge(constr, strength = 100, dem_08, tot_08, tgts_group = c(0.5, 0.6)) # encourage districts to have the same number of counties constr <- add_constr_custom(constr, strength = 1000, fn = function(plan, distr) { # notice that we only use information on precincts in `distr` abs(sum(plan == distr) - 99/4) }) print(constr)
Count County Splits
county_splits(map, counties, .data = cur_plans()) redist.splits(plans, counties)
county_splits(map, counties, .data = cur_plans()) redist.splits(plans, counties)
map |
a |
counties |
A vector of county names or county ids. |
.data |
a |
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. Required. |
integer vector with one number for each map
redist.compactness
is used to compute different compactness statistics for a
shapefile. It currently computes the Polsby-Popper, Schwartzberg score, Length-Width Ratio,
Convex Hull score, Reock score, Boyce Clark Index, Fryer Holden score, Edges Removed number,
and the log of the Spanning Trees.
distr_compactness(map, measure = "FracKept", .data = cur_plans(), ...) redist.compactness( shp = NULL, plans, measure = c("PolsbyPopper"), total_pop = NULL, adj = NULL, draw = 1, ncores = 1, counties = NULL, planarize = 3857, ppRcpp, perim_path, perim_df )
distr_compactness(map, measure = "FracKept", .data = cur_plans(), ...) redist.compactness( shp = NULL, plans, measure = c("PolsbyPopper"), total_pop = NULL, adj = NULL, draw = 1, ncores = 1, counties = NULL, planarize = 3857, ppRcpp, perim_path, perim_df )
map |
a |
measure |
A vector with a string for each measure desired. "PolsbyPopper", "Schwartzberg", "LengthWidth", "ConvexHull", "Reock", "BoyceClark", "FryerHolden", "EdgesRemoved", "FracKept", and "logSpanningTree" are implemented. Defaults to "PolsbyPopper". Use "all" to return all implemented measures. |
.data |
a |
... |
passed on to |
shp |
A SpatialPolygonsDataFrame or sf object. Required unless "EdgesRemoved" and "logSpanningTree" with adjacency provided. |
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. Required. |
total_pop |
A numeric vector with the population for every observation. Is only necessary when "FryerHolden" is used for measure. Defaults to NULL. |
adj |
A zero-indexed adjacency list. Only used for "PolsbyPopper",
EdgesRemoved" and "logSpanningTree". Created with |
draw |
A numeric to specify draw number. Defaults to 1 if only one map provided
and the column number if multiple maps given. Can also take a factor input, which will become the
draw column in the output if its length matches the number of entries in plans. If the |
ncores |
Number of cores to use for parallel computing. Default is 1. |
counties |
A numeric vector from 1:ncounties corresponding to counties. Required for "logSpanningTree". |
planarize |
a number, indicating the CRS to project the shapefile to if it is latitude-longitude based. Set to FALSE to avoid planarizing. |
ppRcpp |
Boolean, whether to run Polsby Popper and Schwartzberg using Rcpp. It has a higher upfront cost, but quickly becomes faster. Becomes TRUE if ncol(district_membership > 8) and not manually set. |
perim_path |
it checks for an Rds, if no rds exists at the path,
it creates an rds with borders and saves it.
This can be created in advance with |
perim_df |
A dataframe output from |
This function computes specified compactness scores for a map. If there is more than one shape specified for a single district, it combines them, if necessary, and computes one score for each district.
Polsby-Popper is computed as
where A is the area function, the district is d, and P is the perimeter function. All values are between 0 and 1, where larger values are more compact.
Schwartzberg is computed as
where A is the area function, the district is d, and P is the perimeter function. All values are between 0 and 1, where larger values are more compact.
The Length Width ratio is computed as
where length is the shorter of the maximum x distance and the maximum y distance. Width is the longer of the two values. All values are between 0 and 1, where larger values are more compact.
The Convex Hull score is computed as
where A is the area function, d is the district, and CVH is the convex hull of the district. All values are between 0 and 1, where larger values are more compact.
The Reock score is computed as
where A is the area function, d is the district, and MBC is the minimum bounding circle of the district. All values are between 0 and 1, where larger values are more compact.
The Boyce Clark Index is computed as
.
The are the distances of the 16 radii computed from the geometric
centroid of the shape to the most outward point of the shape that intersects
the radii, if the centroid is contained within the shape. If the centroid
lies outside of the shape, a point on the surface is used, which will naturally
incur a penalty to the score. All values are between 0 and 1,
where larger values are more compact.
The Fryer Holden score for each district is computed with
,
where is the population product matrix. Each element is the
product of the i-th and j-th precinct's populations. D represents the distance,
where the matrix is the distance between each precinct. To fully compute this
index, for any map, the sum of these values should be used as the numerator.
The denominator can be calculated from the full enumeration of districts as the
smallest calculated numerator. This produces very large numbers, where smaller
values are more compact.
The log spanning tree measure is the logarithm of the product of the number of spanning trees which can be drawn on each district.
The edges removed measure is number of edges removed from the underlying adjacency graph. A smaller number of edges removed is more compact.
The fraction kept measure is the fraction of edges that were not removed from the underlying adjacency graph. This takes values 0 - 1, where 1 is more compact.
A tibble with a column that specifies the district, a column for each specified measure, and a column that specifies the map number.
Boyce, R., & Clark, W. 1964. The Concept of Shape in Geography. Geographical Review, 54(4), 561-572.
Cox, E. 1927. A Method of Assigning Numerical and Percentage Values to the Degree of Roundness of Sand Grains. Journal of Paleontology, 1(3), 179-183.
Fryer R, Holden R. 2011. Measuring the Compactness of Political Districting Plans. Journal of Law and Economics.
Harris, Curtis C. 1964. “A scientific method of districting”. Behavioral Science 3(9), 219–225.
Maceachren, A. 1985. Compactness of Geographic Shape: Comparison and Evaluation of Measures. Geografiska Annaler. Series B, Human Geography, 67(1), 53-67.
Polsby, Daniel D., and Robert D. Popper. 1991. “The Third Criterion: Compactness as a procedural safeguard against partisan gerrymandering.” Yale Law & Policy Review 9 (2): 301–353.
Reock, E. 1961. A Note: Measuring Compactness as a Requirement of Legislative Apportionment. Midwest Journal of Political Science, 5(1), 70-74.
Schwartzberg, Joseph E. 1966. Reapportionment, Gerrymanders, and the Notion of Compactness. Minnesota Law Review. 1701.
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] # old redist.compactness( # shp = fl25, plans = plans_05[, 1:3], # measure = c("PolsbyPopper", "EdgesRemoved") # ) comp_polsby(plans_05[, 1:3], fl25) comp_edges_rem(plans_05[, 1:3], fl25, fl25$adj)
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] # old redist.compactness( # shp = fl25, plans = plans_05[, 1:3], # measure = c("PolsbyPopper", "EdgesRemoved") # ) comp_polsby(plans_05[, 1:3], fl25) comp_edges_rem(plans_05[, 1:3], fl25, fl25$adj)
This data contains NAD83 (HARN) EPSG codes for every U.S. state.
Since redist
uses projected geometries, it is often a good idea to use
projections tailored to a particular state, rather than, for example, a
Mercator projection. Use these codes along with sf::st_transform()
to
project your shapefiles nicely.
data("EPSG")
data("EPSG")
named list containing EPSG codes for each U.S. state. Codes are indexed by state abbreviations.
data(EPSG) EPSG$WA # 2855
data(EPSG) EPSG$WA # 2855
This data set contains the 25-precinct shapefile and related data for each precinct.
All possible partitions of the 25 precincts into three contiguous
congressional districts are stored in fl25_enum
, and the
corresponding adjacency graph is stored in fl25_adj
.
This is generally useful for demonstrating basic algorithms locally.
data("fl25")
data("fl25")
sf data.frame containing columns for useful data related to the redistricting process, subsetted from real data in Florida, and sf geometry column.
geoid
Contains unique identifier for each precinct which can be matched to the full Florida dataset.
pop
Contains the population of each precinct.
vap
Contains the voting age population of each precinct.
obama
Contains the 2012 presidential vote for Obama.
mccain
Contains the 2012 presidential vote for McCain.
TotPop
Contains the population of each precinct. Identical to pop.
BlackPop
Contains the black population of each precinct.
HispPop
Contains the Hispanic population of each precinct.
VAP
Contains the voting age population of each precinct. Identical to vap.
BlackVAP
Contains the voting age population of black constituents of each precinct.
HispVAP
Contains the voting age population of hispanic constituents of each precinct.
geometry
Contains sf geometry of each precinct.
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
data(fl25)
data(fl25)
This data set contains the 25-precinct shapefile and related data for each precinct.
All possible partitions of the 25 precincts into three contiguous
congressional districts are stored in fl25_enum
, and the
corresponding adjacency graph is stored in fl25_adj
.
A list storing the adjacency graph for the 25-precinct subset of Florida.
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
data(fl25_adj)
data(fl25_adj)
This data set contains demographic and geographic information about 25
contiguous precincts in the state of Florida. The data lists all possible
partitions of the 25 precincts into three contiguous congressional districts.
The 25-precinct shapefile may be found in fl25
data("fl25_enum")
data("fl25_enum")
A list with two entries:
plans
A matrix containing every partition of the 25 precincts into three contiguous congressional districts, with no population constraint.
pop_dev
A vector containing the maximum population deviation across the three districts for each plan.
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
Massey, Douglas and Nancy Denton. (1987) "The Dimensions of Social Segregation". Social Forces.
data(fl25_enum)
data(fl25_enum)
This data set contains the 250 Precinct shapefile and related data for each precinct.
data("fl250")
data("fl250")
sf data.frame containing columns for useful data related to the redistricting process, subsetted from real data in Florida, and sf geometry column.
geoid
Contains unique identifier for each precinct which can be matched to the full Florida dataset.
pop
Contains the population of each precinct.
vap
Contains the voting age population of each precinct.
obama
Contains the 2012 presidential vote for Obama.
mccain
Contains the 2012 presidential vote for McCain.
TotPop
Contains the population of each precinct. Identical to pop.
BlackPop
Contains the black population of each precinct.
HispPop
Contains the Hispanic population of each precinct.
VAP
Contains the voting age population of each precinct. Identical to vap.
BlackVAP
Contains the voting age population of black constituents of each precinct.
HispVAP
Contains the voting age population of hispanic constituents of each precinct.
geometry
Contains sf geometry of each precinct.
It is a random 70 precinct connected subset from Florida's precincts. This was introduced by doi:10.1080/2330443X.2020.1791773
Benjamin Fifield, Kosuke Imai, Jun Kawahara & Christopher T. Kenny (2020) The Essential Role of Empirical Validation in Legislative Redistricting Simulation, Statistics and Public Policy, 7:1, 52-68, doi:10.1080/2330443X.2020.1791773
data(fl250)
data(fl250)
This data set contains the 70 Precinct shapefile and related data for each precinct.
data("fl70")
data("fl70")
sf data.frame containing columns for useful data related to the redistricting process, subsetted from real data in Florida, and sf geometry column.
geoid
Contains unique identifier for each precinct which can be matched to the full Florida dataset.
pop
Contains the population of each precinct.
vap
Contains the voting age population of each precinct.
obama
Contains the 2012 presidential vote for Obama.
mccain
Contains the 2012 presidential vote for McCain.
TotPop
Contains the population of each precinct. Identical to pop.
BlackPop
Contains the black population of each precinct.
HispPop
Contains the Hispanic population of each precinct.
VAP
Contains the voting age population of each precinct. Identical to vap.
BlackVAP
Contains the voting age population of black constituents of each precinct.
HispVAP
Contains the voting age population of hispanic constituents of each precinct.
geometry
Contains sf geometry of each precinct.
It is a random 70 precinct connected subset from Florida's precincts. This was introduced by doi:10.1080/2330443X.2020.1791773
Benjamin Fifield, Kosuke Imai, Jun Kawahara & Christopher T. Kenny (2020) The Essential Role of Empirical Validation in Legislative Redistricting Simulation, Statistics and Public Policy, 7:1, 52-68, doi:10.1080/2330443X.2020.1791773
data(fl70)
data(fl70)
Freeze Parts of a Map
freeze(freeze_row, plan, .data = cur_map()) redist.freeze(adj, freeze_row, plan = rep(1, length(adj)))
freeze(freeze_row, plan, .data = cur_map()) redist.freeze(adj, freeze_row, plan = rep(1, length(adj)))
freeze_row |
Required, logical vector where TRUE freezes and FALSE lets a precinct stay free or a vector of indices to freeze |
plan |
A vector of district assignments, which if provided will create
separate groups by district. Recommended. In |
.data |
a |
adj |
Required, zero indexed adjacency list. |
integer vector to group by
library(redist) library(dplyr) data(fl25) data(fl25_enum) data(fl25_adj) plan <- fl25_enum$plans[, 5118] freeze_id <- redist.freeze(adj = fl25_adj, freeze_row = (plan == 2), plan = plan) data(iowa) map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.02) map <- map %>% merge_by(freeze(cd_2010 == 1, .data = .))
library(redist) library(dplyr) data(fl25) data(fl25_enum) data(fl25_adj) plan <- fl25_enum$plans[, 5118] freeze_id <- redist.freeze(adj = fl25_adj, freeze_row = (plan == 2), plan = plan) data(iowa) map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.02) map <- map %>% merge_by(freeze(cd_2010 == 1, .data = .))
redist_map
objectGet and set the adjacency graph from a redist_map
object
get_adj(x) set_adj(x, adj)
get_adj(x) set_adj(x, adj)
x |
the |
adj |
a new adjacency list. |
a zero-indexed adjacency list (get_adj
)
the modified redist_map
object (set_adj
)
redist_map
objectExtract the existing district assignment from a redist_map
object
get_existing(x)
get_existing(x)
x |
the |
an integer vector of district numbers
Extract the Metropolis Hastings Acceptance Rate
get_mh_acceptance_rate(plans)
get_mh_acceptance_rate(plans)
plans |
the |
a numeric acceptance rate
Extract the matrix of district assignments from a redistricting simulation
get_plans_matrix(x) ## S3 method for class 'redist_plans' as.matrix(x, ...)
get_plans_matrix(x) ## S3 method for class 'redist_plans' as.matrix(x, ...)
x |
the |
... |
ignored |
matrix
matrix
May be NULL
if no weights exist (MCMC or optimization methods).
get_plans_weights(plans) ## S3 method for class 'redist_plans' weights(object, ...)
get_plans_weights(plans) ## S3 method for class 'redist_plans' weights(object, ...)
plans , object
|
the |
... |
Ignored. |
A numeric vector of weights, with an additional attribute
resampled
indicating whether the plans have been resampled according
to these weights. If weights have been resampled, this returns the weights
before resampling (i.e., they do not correspond to the resampled plans).
numeric vector
redist_map
objectGet and set the population tolerance from a redist_map
object
get_pop_tol(map) set_pop_tol(map, pop_tol)
get_pop_tol(map) set_pop_tol(map, pop_tol)
map |
the |
pop_tol |
the population tolerance |
For get_pop_tol
, a single numeric value, the population
tolerance
For seet_pop_tol
, an updated redist_map
object
Extract the sampling information from a redistricting simulation
get_sampling_info(plans)
get_sampling_info(plans)
plans |
the |
a list of parameters and information about the sampling problem.
redist_map
objectExtract the target district population from a redist_map
object
get_target(x)
get_target(x)
x |
the |
a single numeric value, the target population
redist.group.percent
computes the proportion that a group makes up in
each district across a matrix of maps.
group_frac( map, group_pop, total_pop = map[[attr(map, "pop_col")]], .data = pl() ) redist.group.percent(plans, group_pop, total_pop, ncores = 1)
group_frac( map, group_pop, total_pop = map[[attr(map, "pop_col")]], .data = pl() ) redist.group.percent(plans, group_pop, total_pop, ncores = 1)
map |
a |
group_pop |
A numeric vector with the population of the group for every precinct. |
total_pop |
A numeric vector with the population for every precinct. |
.data |
a |
plans |
A matrix with one row for each precinct and one column for each map. Required. |
ncores |
Number of cores to use for parallel computing. Default is 1. |
matrix with percent for each district
data(fl25) data(fl25_enum) cd <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] fl25_map = redist_map(fl25, ndists=3, pop_tol=0.1) fl25_plans = redist_plans(cd, fl25_map, algorithm="enumpart") group_frac(fl25_map, BlackPop, TotPop, fl25_plans)
data(fl25) data(fl25_enum) cd <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] fl25_map = redist_map(fl25, ndists=3, pop_tol=0.1) fl25_plans = redist_plans(cd, fl25_map, algorithm="enumpart") group_frac(fl25_map, BlackPop, TotPop, fl25_plans)
This data contains geographic and demographic information on the 99 counties of the state of Iowa.
data("iowa")
data("iowa")
sf tibble containing columns for useful data related to the redistricting process
fips
The FIPS code for the county.
cd_2010
The 2010 congressional district assignments.
pop
The total population of the precinct, according to the 2010 Census.
white
The non-Hispanic white population of the precinct.
black
The non-Hispanic Black population of the precinct.
hisp
The Hispanic population (of any race) of the precinct.
vap
The voting-age population of the precinct.
wvap
The white voting-age population of the precinct.
bvap
The Black voting-age population of the precinct.
hvap
The Hispanic voting-age population of the precinct.
tot_08
Number of total votes for president in the county in 2008.
dem_08
Number of votes for Barack Obama in 2008.
rep_08
Number of votes for John McCain in 2008.
region
The 28E agency regions for counties.
geometry
The sf geometry column containing the geographic information.
data(iowa) print(iowa)
data(iowa) print(iowa)
redist_map
object is contiguousCheck that a redist_map
object is contiguous
is_contiguous(x)
is_contiguous(x)
x |
the object |
TRUE
if contiguous.
Identify which counties are split by a plan
is_county_split(plan, counties)
is_county_split(plan, counties)
plan |
A vector of precinct/unit assignments |
counties |
A vector of county names or county ids. |
A logical vector which is TRUE
for precincts belonging to
counties which are split
Extract the last plan from a set of plans
last_plan(plans)
last_plan(plans)
plans |
A |
An integer vector containing the final plan assignment.
Creates a grouping ID to unite geographies and perform analysis on a smaller
set of precincts. It identifies all precincts more than boundary
edges
of a district district boundary. Each contiguous group of precincts more than
boundary
steps away from another district gets it own group. Some
districts may have multiple, disconnected components that make up the core,
but each of these is assigned a separate grouping id so that a call to
sf::st_union()
would produce only connected pieces.
make_cores(.data = cur_map(), boundary = 1, focus = NULL) redist.identify.cores(adj, plan, boundary = 1, focus = NULL, simplify = TRUE)
make_cores(.data = cur_map(), boundary = 1, focus = NULL) redist.identify.cores(adj, plan, boundary = 1, focus = NULL, simplify = TRUE)
.data |
a |
boundary |
Number of steps to check for. Defaults to 1. |
focus |
Optional. Integer. A single district to focus on. |
adj |
zero indexed adjacency list. |
plan |
An integer vector or matrix column of district assignments. |
simplify |
Optional. Logical. Whether to return extra information or just grouping ID. |
This is a loose interpretation of the
NCSL's summary
of redistricting criteria to preserve the cores of prior districts. Using the
adjacency graph for a given plan, it will locate the precincts on the
boundary of the district, within boundary
steps of the edge. Each of
these is given their own group. Each remaining entry that is not near the
boundary of the district is given an id that can be used to group the
remainder of the district by connected component. This portion is deemed the
core of the district.
integer vector (if simplify is false). Otherwise it returns a tibble with the grouping
variable as group_id
and additional information on connected components.
redist.plot.cores()
for a plotting function
data(fl250) fl250_map <- redist_map(fl250, ndists = 4, pop_tol = 0.01) plan <- as.matrix(redist_smc(fl250_map, 20, silent = TRUE)) core <- redist.identify.cores(adj = fl250_map$adj, plan = plan) redist.plot.cores(shp = fl250, plan = plan, core = core)
data(fl250) fl250_map <- redist_map(fl250, ndists = 4, pop_tol = 0.01) plan <- as.matrix(redist_smc(fl250_map, 20, silent = TRUE)) core <- redist.identify.cores(adj = fl250_map$adj, plan = plan) redist.plot.cores(shp = fl250, plan = plan, core = core)
District numbers in simulated plans are by and large random. This function attempts to renumber the districts across all simulated plans to match the numbers in a provided plan, using the Hungarian algorithm.
match_numbers( data, plan, total_pop = attr(data, "prec_pop"), col = "pop_overlap" )
match_numbers( data, plan, total_pop = attr(data, "prec_pop"), col = "pop_overlap" )
data |
a |
plan |
a character vector giving the name of the plan to match to (e.g., for a reference plan), or an integer vector containing the plan itself. |
total_pop |
a vector of population counts. Should not be needed for most
|
col |
the name of a new column to store the vector of population overlap
with the reference plan: the fraction of the total population who are in
the same district under each plan and the reference plan. Set to
|
a modified redist_plans
object. New district numbers will be
stored as an ordered factor variable in the district
column. The
district numbers in the plan matrix will match the levels of this factor.
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_smc(iowa_map, 100, silent = TRUE) match_numbers(plans, "cd_2010")
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_smc(iowa_map, 100, silent = TRUE) match_numbers(plans, "cd_2010")
In performing a county-level or cores-based analysis it is often necessary to merge several units together into a larger unit. This function performs this operation, modifying the adjacency graph as needed and attempting to properly aggregate other data columns.
merge_by(.data, ..., by_existing = TRUE, drop_geom = TRUE, collapse_chr = TRUE)
merge_by(.data, ..., by_existing = TRUE, drop_geom = TRUE, collapse_chr = TRUE)
.data |
a |
... |
|
by_existing |
if an existing assignment is present, whether to also group by it |
drop_geom |
whether to drop the geometry column. Recommended, as otherwise a costly geometric merge is required. |
collapse_chr |
if |
A merged redist_map
object
This function computes a minimal set of population moves (e.g., 5 people from district 1 to district 3) to maximally balance the population between districts. The moves are only allowed between districts that share the territory of a county, so that any boundary adjustments are guaranteed to preserve all unbroken county boundaries.
min_move_parity(map, plan, counties = NULL, penalty = 0.2)
min_move_parity(map, plan, counties = NULL, penalty = 0.2)
map |
|
plan |
an integer vector containing the plan to be balanced. Tidy-evaluated. |
counties |
an optional vector of counties, whose boundaries will be preserved. Tidy-evaluated. |
penalty |
the larger this value, the more to encourage sparsity. |
a list with components:
moves
A tibble describing the population moves
pop_old
The current district populations
pop_new
The district populations after the moves
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) min_move_parity(iowa_map, cd_2010)
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) min_move_parity(iowa_map, cd_2010)
Counts the total number of municpalities that are split. Municipalities in this interpretation do not need to cover the entire state, which differs from counties.
muni_splits(map, munis, .data = cur_plans()) redist.muni.splits(plans, munis)
muni_splits(map, munis, .data = cur_plans()) redist.muni.splits(plans, munis)
map |
a |
munis |
A vector of municipality names or ids. |
.data |
a |
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. Required. |
integer vector of length ndist by ncol(plans)
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.01) plans <- redist_smc(ia, 50, silent = TRUE) ia$region[1:10] <- NA #old redist.muni.splits(plans, ia$region) splits_sub_admin(plans, ia, region)
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.01) plans <- redist_smc(ia, 50, silent = TRUE) ia$region[1:10] <- NA #old redist.muni.splits(plans, ia$region) splits_sub_admin(plans, ia, region)
District numbers in simulated plans are by and large random. This function will renumber the districts across all simulated plans in order of a provided quantity of interest.
number_by(data, x, desc = FALSE)
number_by(data, x, desc = FALSE)
data |
a |
x |
|
desc |
|
a modified redist_plans
object. New district numbers will be
stored as an ordered factor variable in the district
column. The
district numbers in the plan matrix will match the levels of this factor.
redist.metrics
is used to compute different gerrymandering metrics for a
set of maps.
partisan_metrics(map, measure, rvote, dvote, ..., .data = cur_plans()) redist.metrics( plans, measure = "DSeats", rvote, dvote, tau = 1, biasV = 0.5, respV = 0.5, bandwidth = 0.01, draw = 1 )
partisan_metrics(map, measure, rvote, dvote, ..., .data = cur_plans()) redist.metrics( plans, measure = "DSeats", rvote, dvote, tau = 1, biasV = 0.5, respV = 0.5, bandwidth = 0.01, draw = 1 )
map |
a |
measure |
A vector with a string for each measure desired from list "DSeats", "DVS", "EffGap", "EffGapEqPop", "TauGap", "MeanMedian", "Bias", "BiasV", "Declination", "Responsiveness", "LopsidedWins", "RankedMarginal", and "SmoothedSeat". Use "all" to get all metrics. "DSeats" and "DVS" are always computed, so it is recommended to always return those values. |
rvote |
A numeric vector with the Republican vote for each precinct. |
dvote |
A numeric vector with the Democratic vote for each precinct. |
... |
passed on to |
.data |
a |
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. Required. |
tau |
A non-negative number for calculating Tau Gap. Only used with option "TauGap". Defaults to 1. |
biasV |
A value between 0 and 1 to compute bias at. Only used with option "BiasV". Defaults to 0.5. |
respV |
A value between 0 and 1 to compute responsiveness at. Only used with option "Responsiveness". Defaults to 0.5. |
bandwidth |
A value between 0 and 1 for computing responsiveness. Only used with option "Responsiveness." Defaults to 0.01. |
draw |
A numeric to specify draw number. Defaults to 1 if only one map provided
and the column number if multiple maps given. Can also take a factor input, which will become the
draw column in the output if its length matches the number of entries in plans. If the |
This function computes specified compactness scores for a map. If there is more than one precinct specified for a map, it aggregates to the district level and computes one score.
DSeats is computed as the expected number of Democratic seats with no change in votes.
DVS is the Democratic Vote Share, which is the two party vote share with Democratic votes as the numerator.
EffGap is the Efficiency Gap, calculated with votes directly.
EffGapEqPop is the Efficiency Gap under an Equal Population assumption, calculated with the DVS.
TauGap is the Tau Gap, computed with the Equal Population assumption.
MeanMedian is the Mean Median difference.
Bias is the Partisan Bias computed at 0.5.
BiasV is the Partisan Bias computed at value V.
Declination is the value of declination at 0.5.
Responsiveness is the responsiveness at the user-supplied value with the user-supplied bandwidth.
LopsidedWins computed the Lopsided Outcomes value, but does not produce a test statistic.
RankedMarginal computes the Ranked Marginal Deviation (0-1, smaller is better). This is also known as the "Gerrymandering Index" and is sometimes presented as this value divided by 10000.
SmoothedSeat computes the Smoothed Seat Count Deviation (0-1, smaller is R Bias, bigger is D Bias).
A tibble with a column for each specified measure and a column that specifies the map number.
Jonathan N. Katz, Gary King, and Elizabeth Rosenblatt. 2020. Theoretical Foundations and Empirical Evaluations of Partisan Fairness in District-Based Democracies. American Political Science Review, 114, 1, Pp. 164-178.
Gregory S. Warrington. 2018. "Quantifying Gerrymandering Using the Vote Distribution." Election Law Journal: Rules, Politics, and Policy. Pp. 39-57.http://doi.org/10.1089/elj.2017.0447
Samuel S.-H. Wang. 2016. "Three Tests for Practical Evaluation of Partisan Gerrymandering." Stanford Law Review, 68, Pp. 1263 - 1321.
Gregory Herschlag, Han Sung Kang, Justin Luo, Christy Vaughn Graves, Sachet Bangia, Robert Ravier & Jonathan C. Mattingly (2020) Quantifying Gerrymandering in North Carolina, Statistics and Public Policy, 7:1, 30-38, DOI: 10.1080/2330443X.2020.1796400
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] # old: redist.metrics(plans_05, measure = "DSeats", rvote = fl25$mccain, dvote = fl25$obama) part_dseats(plans_05, fl25, mccain, obama)
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] # old: redist.metrics(plans_05, measure = "DSeats", rvote = fl25$mccain, dvote = fl25$obama) part_dseats(plans_05, fl25, mccain, obama)
redist_plans()
ObjectUseful inside piped expressions and dplyr
functions.
pl()
pl()
A redist_plans
object, or NULL
if not called from inside a
dplyr
function.
pl()
pl()
Compute Distance between Partitions
plan_distances(plans, measure = "variation of information", ncores = 1) redist.distances(plans, measure = "Hamming", ncores = 1, total_pop = NULL)
plan_distances(plans, measure = "variation of information", ncores = 1) redist.distances(plans, measure = "Hamming", ncores = 1, total_pop = NULL)
plans |
A matrix with one row for each precinct and one column for each map. Required. |
measure |
String vector indicating which distances to compute. Implemented currently are "Hamming", "Manhattan", "Euclidean", and "variation of information", Use "all" to return all implemented measures. Not case sensitive, and any unique substring is enough, e.g. "ham" for Hamming, or "info" for variation of information. |
ncores |
Number of cores to use for parallel computing. Default is 1. |
total_pop |
The vector of precinct populations. Used only if computing variation of information. If not provided, equal population of precincts will be assumed, i.e. the VI will be computed with respect to the precincts themselves, and not the population. |
Hamming distance measures the number of different precinct assignments between plans. Manhattan and Euclidean distances are the 1- and 2-norms for the assignment vectors. All three of the Hamming, Manhattan, and Euclidean distances implemented here are not invariant to permutations of the district labels; permuting will cause large changes in measured distance, and maps which are identical up to a permutation may be computed to be maximally distant.
Variation of Information is a metric on population partitions (i.e., districtings) which is invariant to permutations of the district labels, and arises out of information theory. It is calculated as
where are the partitions,
the individual
districts,
is the population, and
the total
population of the state. VI is also expressible as the difference between
the joint entropy and the mutual information (see references).
distance_matrix
returns a numeric distance matrix for the
chosen metric.
a named list of distance matrices, one for each distance measure selected.
Cover, T. M. and Thomas, J. A. (2006). Elements of information theory. John Wiley & Sons, 2 edition.
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] distances <- redist.distances(plans_05) distances$Hamming[1:5, 1:5]
data(fl25) data(fl25_enum) plans_05 <- fl25_enum$plans[, fl25_enum$pop_dev <= 0.05] distances <- redist.distances(plans_05) distances$Hamming[1:5, 1:5]
Returns the off-diagonal elements of the variation of information distance matrix for a sample of plans, which can be used as a diagnostic measure to assess the diversity of a set of plans. While the exact scale varies depending on the number of precincts and districts, generally diversity is good if most of the values are greater than 0.5. Conversely, if there are many values close to zero, then the sample has many similar plans and may not be a good approximation to the target distribution.
plans_diversity( plans, chains = 1, n_max = 100, ncores = 1, total_pop = attr(plans, "prec_pop") )
plans_diversity( plans, chains = 1, n_max = 100, ncores = 1, total_pop = attr(plans, "prec_pop") )
plans |
a |
chains |
For plans objects with multiple chains, which ones to compute diversity for. Defaults to the first. Specify "all" to use all chains. |
n_max |
the maximum number of plans to sample in computing the distances. Larger numbers will have less sampling error but will require more computation time. |
ncores |
the number of cores to use in computing the distances. |
total_pop |
The vector of precinct populations. Used only if computing variation of information. If not provided, equal population of precincts will be assumed, i.e. the VI will be computed with respect to the precincts themselves, and not the population. |
A numeric vector of off-diagonal variation of information distances.
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) plans <- redist_smc(ia, 100, silent = TRUE) hist(plans_diversity(plans))
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) plans <- redist_smc(ia, 100, silent = TRUE) hist(plans_diversity(plans))
Plot a plan classification
## S3 method for class 'redist_classified' plot(x, plans, shp, type = "fill", which = NULL, ...)
## S3 method for class 'redist_classified' plot(x, plans, shp, type = "fill", which = NULL, ...)
x |
a |
plans |
a redist_plans object. |
shp |
a shapefile or redist_map object. |
type |
either |
which |
indices of the splits to plot. Defaults to all |
... |
passed on to |
ggplot comparison plot
Plots the constraint strength versus some running variable. Currently
supports visualizing the grp_hinge
, grp_inv_hinge
, and grp_pow
constraints.
## S3 method for class 'redist_constr' plot(x, y, type = "group", xlim = c(0, 1), ...)
## S3 method for class 'redist_constr' plot(x, y, type = "group", xlim = c(0, 1), ...)
x |
A redist_constr object. |
y |
Ignored. |
type |
What type of constraint to visualize. Currently supports only
|
xlim |
Range of group shares to visualize. |
... |
additional arguments (ignored) |
A ggplot object
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) constr <- redist_constr(iowa_map) constr <- add_constr_grp_hinge(constr, strength = 30, dem_08, tot_08, tgts_group = 0.5) constr <- add_constr_grp_hinge(constr, strength = -20, dem_08, tot_08, tgts_group = 0.3) plot(constr)
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) constr <- redist_constr(iowa_map) constr <- add_constr_grp_hinge(constr, strength = 30, dem_08, tot_08, tgts_group = 0.5) constr <- add_constr_grp_hinge(constr, strength = -20, dem_08, tot_08, tgts_group = 0.3) plot(constr)
redist_map
Plot a redist_map
## S3 method for class 'redist_map' plot(x, fill = NULL, by_distr = FALSE, adj = FALSE, ...)
## S3 method for class 'redist_map' plot(x, fill = NULL, by_distr = FALSE, adj = FALSE, ...)
x |
the |
fill |
|
by_distr |
if |
adj |
if |
... |
passed on to |
ggplot2 object
data(fl25) d <- redist_map(fl25, ndists = 3, pop_tol = 0.05) plot(d) plot(d, BlackPop/pop) data(fl25_enum) fl25$dist <- fl25_enum$plans[, 5118] d <- redist_map(fl25, existing_plan = dist) plot(d)
data(fl25) d <- redist_map(fl25, ndists = 3, pop_tol = 0.05) plot(d) plot(d, BlackPop/pop) data(fl25_enum) fl25$dist <- fl25_enum$plans[, 5118] d <- redist_map(fl25, existing_plan = dist) plot(d)
\link{redist_plans}
If no arguments are passed, defaults to plotting the sampling weights for
the redist_plans
object. If no weights exist, plots district
populations.
## S3 method for class 'redist_plans' plot(x, ..., type = "distr_qtys")
## S3 method for class 'redist_plans' plot(x, ..., type = "distr_qtys")
x |
the |
... |
passed on to the underlying function |
type |
the name of the plotting function to use. Will have
|
Extract the district assignments for a precinct across all simulated plans
prec_assignment(prec, .data = pl())
prec_assignment(prec, .data = pl())
prec |
the precinct number |
.data |
a |
integer vector, a row from a plans matrix
For a map with n
precincts Returns an n
-by-n
matrix, where each
entry measures the fraction of the plans in which the row and column
precincts were in the same district.
prec_cooccurrence(plans, which = NULL, sampled_only = TRUE, ncores = 1)
prec_cooccurrence(plans, which = NULL, sampled_only = TRUE, ncores = 1)
plans |
a redist_plans object. |
which |
|
sampled_only |
if |
ncores |
the number of parallel cores to use in the computation. |
a symmetric matrix the size of the number of precincts.
Print redist_classified objects
## S3 method for class 'redist_classified' print(x, ...)
## S3 method for class 'redist_classified' print(x, ...)
x |
redist_classified object |
... |
additional arguments |
prints to console
Generic to print redist_constr
## S3 method for class 'redist_constr' print(x, header = TRUE, details = TRUE, ...)
## S3 method for class 'redist_constr' print(x, header = TRUE, details = TRUE, ...)
x |
redist_constr |
header |
if FALSE, then suppress introduction / header line |
details |
if FALSE, then suppress the details of each constraint |
... |
additional arguments |
Prints to console and returns input redist_constr
Generic to print redist_map
## S3 method for class 'redist_map' print(x, ...)
## S3 method for class 'redist_map' print(x, ...)
x |
redist_map |
... |
additional arguments |
Prints to console and returns input redist_map
redist_plans
Print method for redist_plans
## S3 method for class 'redist_plans' print(x, ...)
## S3 method for class 'redist_plans' print(x, ...)
x |
a redist_plans object |
... |
additional arguments (ignored) |
The original object, invisibly.
Merging map units through merge_by
or summarize
changes the indexing of each unit. Use this function to take a set of
redistricting plans from a redist
algorithm and re-index them to
be compatible with the original set of units.
pullback(plans, map = NULL)
pullback(plans, map = NULL)
plans |
a |
map |
optionally, a |
a new, re-indexed, redist_plans
object
Only works when all the sets are compatible—generated from the same map,
with the same number of districts. Sets of plans will be indexed by the
chain
column.
## S3 method for class 'redist_plans' rbind(..., deparse.level = 1)
## S3 method for class 'redist_plans' rbind(..., deparse.level = 1)
... |
The |
deparse.level |
Ignored. |
A new redist_plans
object.
Builds a confidence interval for a quantity of interest. If multiple runs are available, uses the between-run variation to estimate the standard error. If only one run is available, uses information on the SMC particle/plan genealogy to estimate the standard error, using a variant of the method of Olson & Douc (2019). The multiple-run estimator is more reliable, especially for situations with many districts, and should be used when parallelism is available. All reference plans are ignored.
redist_ci(plans, x, district = 1L, conf = 0.9, by_chain = FALSE) redist_smc_ci(plans, x, district = 1L, conf = 0.9, by_chain = FALSE) redist_mcmc_ci(plans, x, district = 1L, conf = 0.9, by_chain = FALSE)
redist_ci(plans, x, district = 1L, conf = 0.9, by_chain = FALSE) redist_smc_ci(plans, x, district = 1L, conf = 0.9, by_chain = FALSE) redist_mcmc_ci(plans, x, district = 1L, conf = 0.9, by_chain = FALSE)
plans |
a redist_plans object. |
x |
the quantity to build an interval for. Tidy-evaluated within |
district |
for redist_plans objects with multiple districts, which
|
conf |
the desired confidence level. |
by_chain |
Whether the confidence interval should indicate overall
sampling uncertainty ( |
A tibble with three columns: X
, X_lower
, and
X_upper
, where X
is the name of the vector of interest,
containing the mean and confidence interval. When used inside
summarize()
this will create three columns in the
output data.
redist_smc_ci()
: Compute confidence intervals for SMC output.
redist_mcmc_ci()
: Compute confidence intervals for MCMC output.
Lee, A., & Whiteley, N. (2018). Variance estimation in the particle filter. Biometrika, 105(3), 609-625.
Olsson, J., & Douc, R. (2019). Numerically stable online estimation of variance in particle filters. Bernoulli, 25(2), 1504-1535.
H. P. Chan and T. L. Lai. A general theory of particle filters in hidden Markov models and some applications. Ann. Statist., 41(6):2877–2904, 2013.
library(dplyr) data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_mergesplit_parallel(iowa_map, nsims = 200, chains = 2, silent = TRUE) %>% mutate(dem = group_frac(iowa_map, dem_08, dem_08 + rep_08)) %>% number_by(dem) redist_smc_ci(plans, dem)
library(dplyr) data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_mergesplit_parallel(iowa_map, nsims = 200, chains = 2, silent = TRUE) %>% mutate(dem = group_frac(iowa_map, dem_08, dem_08 + rep_08)) %>% number_by(dem) redist_smc_ci(plans, dem)
redist_constr
objects are used to specify constraints when sampling
redistricting plans with redist_smc()
and redist_mergesplit()
. Each
constraint is specified as a function which scores a given plan. Higher
scores are penalized and sampled less frequently.
redist_constr(map = tibble())
redist_constr(map = tibble())
map |
a |
The redist_constr
object keeps track of sampling constraints in a nested list.
You can view the exact structure of this list by calling str()
.
Constraints may be added by using one of the following functions:
More information about each constraint can be found on the relevant constraint page.
a redist_constr
object, which is just a list with a certain nested structure.
data(iowa) map_ia <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) constr <- redist_constr(map_ia) constr <- add_constr_splits(constr, strength = 1.5, admin = region) print(constr)
data(iowa) map_ia <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) constr <- redist_constr(map_ia) constr <- add_constr_splits(constr, strength = 1.5, admin = region) print(constr)
This function allows users to simulate redistricting plans using a Markov Chain Monte Carlo algorithm (Fifield, Higgins, Imai, and Tarr 2020). Several constraints corresponding to substantive requirements in the redistricting process are implemented, including population parity and geographic compactness. In addition, the function includes multiple-swap and simulated tempering functionality to improve the mixing of the Markov Chain.
redist_flip( map, nsims, warmup = 0, init_plan, constraints = add_constr_edges_rem(redist_constr(map), 0.4), thin = 1, eprob = 0.05, lambda = 0, temper = FALSE, betaseq = "powerlaw", betaseqlength = 10, betaweights = NULL, adapt_lambda = FALSE, adapt_eprob = FALSE, exact_mh = FALSE, adjswaps = TRUE, init_name = NULL, verbose = TRUE, nthin )
redist_flip( map, nsims, warmup = 0, init_plan, constraints = add_constr_edges_rem(redist_constr(map), 0.4), thin = 1, eprob = 0.05, lambda = 0, temper = FALSE, betaseq = "powerlaw", betaseqlength = 10, betaweights = NULL, adapt_lambda = FALSE, adapt_eprob = FALSE, exact_mh = FALSE, adjswaps = TRUE, init_name = NULL, verbose = TRUE, nthin )
map |
A |
nsims |
The number of samples to draw, not including warmup. |
warmup |
The number of warmup samples to discard. |
init_plan |
A vector containing the congressional district labels
of each geographic unit. The default is |
constraints |
A |
thin |
The amount by which to thin the Markov Chain. The
default is |
eprob |
The probability of keeping an edge connected. The
default is |
lambda |
lambda The parameter determining the number of swaps to attempt
each iteration of the algorithm. The number of swaps each iteration is
equal to Pois( |
temper |
Whether to use simulated tempering algorithm. Default is FALSE. |
betaseq |
Sequence of beta values for tempering. The default is
|
betaseqlength |
Length of beta sequence desired for
tempering. The default is |
betaweights |
betaweights Sequence of weights for different values of
beta. Allows the user to upweight certain values of beta over
others. The default is |
adapt_lambda |
adapt_lambda Whether to adaptively tune the lambda parameter so that the Metropolis-Hastings acceptance probability falls between 20% and 40%. Default is FALSE. |
adapt_eprob |
eprob Whether to adaptively tune the edgecut probability parameter so that the Metropolis-Hastings acceptance probability falls between 20% and 40%. Default is FALSE. |
exact_mh |
Whether to use the approximate (FALSE) or exact (TRUE) Metropolis-Hastings ratio calculation for accept-reject rule. Default is FALSE. |
adjswaps |
Flag to restrict swaps of beta so that only
values adjacent to current constraint are proposed. The default is
|
init_name |
a name for the initial plan, or |
verbose |
Whether to print initialization statement. Default is |
nthin |
Deprecated. Use |
redist_flip
allows for Gibbs constraints to be supplied via a list object
passed to constraints
.
redist_flip
uses a small compactness constraint by default, as this improves
the realism of the maps greatly and also leads to large speed improvements.
(One of the most time consuming aspects of the flip MCMC backend is checking for
district shattering, which is slowed down even further by non-compact districts.
As such, it is recommended that all flip simulations use at least a minimal compactness
constraint, even if you weaken it from the default settings.) The default is
a compact
constraint using the edges-removed
metric with a
weight of 0.6. For very small maps (< 100 precincts), you will likely want to
weaken (lower) this constraint, while for very large maps (> 5000 precincts),
you will likely want to strengthen (increase) this constraint. Otherwise,
for most maps, the default constraint should be a good starting place.
redist_flip
samples from a known target distribution which can be described
using the constraints
. The following describes the constraints available. The general
advice is to set weights in a way that gets between 20% and 40% acceptance
on average, though more tuning advice is available in the vignette on using
MCMC methods.Having too small of an acceptance rate indicates that the weights
within constraints
are too large and will impact sampling efficiency.
If the Metropolis Hastings acceptance rate is too large, this may impact the
target distribution, but may be fine for general exploration of possible maps.
There are currently 9 implemented constraint types, though 'compact
and
partisan
have sub-types which are specified via a character metric
within their respective list objects. The constraints are as follows:
compact
- biases the algorithm towards drawing more compact districts.
weight - the coefficient to put on the Gibbs constraint
metric - which metric to use. Must be one of edges-removed
(the default),
polsby-popper
, fryer-holden
, or log-st
. Using Polsby Popper
is generally not recommended, as edges-removed
is faster and highly correlated.
log-st
can be used to match the target distribution of redist_smc
or
redist_mergesplit
.
areas - Only used with polsby-popper
- A vector of precinct areas.
borderlength_mat - Only used with polsby-popper
- A matrix of precinct
border lengths.
ssdmat - Only used with fryer-holden
- A matrix of squared distances between
precinct centroids.
ssd_denom - Only used with fryer-holden
- a positive integer to use
as the normalizing constant for the Relative Proximity Index.
population
- A Gibbs constraint to complement the hard population
constraint set by pop_tol
. This penalizes moves which move away from smaller
population parity deviations. It is very useful when an init_plan
sits
outside of the desired pop_tol
but there are substantive reasons to use
that plan. This constraint uses the input to total_pop
.
weight - the coefficient to put on the Gibbs constraint
countysplit
This is a Gibbs constraint to minimize county splits. Unlike
SMC's county constraint, this allows for more than ndists - 1
splits and
does not require that counties are contiguous.
weight - the coefficient to put on the Gibbs constraint
hinge
This uses the proportion of a group in a district and matches to the
nearest target proportion, and then creates a penalty of
.
weight - the coefficient to put on the Gibbs constraint
minorityprop - A numeric vector of minority proportions (between 0 and 1) which districts should aim to have
vra
This takes two target proportions of the presence of a minority group
within a district.
weight - the coefficient to put on the Gibbs constraint
target_min - the target minority percentage. Often, this is set to 0.55 to encourage minority majority districts.
target_other - the target minority percentage for non majority minority districts.
minority
This constraint sorts the districts by the proportion of a group in
a district and compares the highest districts to the entries of minorityprop.
This takes the form where n
is the length of minorityprop input.
weight - the coefficient to put on the Gibbs constraint
minorityprop - A numeric vector of minority proportions (between 0 and 1) which districts should aim to have
similarity
This is a status-quo constraint which penalizes plans which
are very different from the starting place. It is useful for local exploration.
weight - the coefficient to put on the Gibbs constraint
partisan
This is a constraint which minimizes partisan bias, either as
measured as the difference from proportional representation or as the magnitude of
the efficiency gap.
weight - the coefficient to put on the Gibbs constraint
rvote - An integer vector of votes for Republicans or other party
dvote - An integer vector of votes for Democrats or other party
metric - which metric to use. Must be one of proportional-representation
or efficiency-gap
.
segregation
This constraint attempts to minimize the degree of dissimilarity
between districts by group population.
weight - the coefficient to put on the Gibbs constraint
A redist_plans
object containing the simulated plans.
Fifield, B., Higgins, M., Imai, K., & Tarr, A. (2020). Automated redistricting simulation using Markov chain Monte Carlo. Journal of Computational and Graphical Statistics, 29(4), 715-728.
data(iowa) iowa_map <- redist_map(iowa, ndists = 4, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.05) sims <- redist_flip(map = iowa_map, nsims = 100)
data(iowa) iowa_map <- redist_map(iowa, ndists = 4, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.05) sims <- redist_flip(map = iowa_map, nsims = 100)
redist_flip_anneal
simulates congressional redistricting plans
using Markov chain Monte Carlo methods coupled with simulated annealing.
redist_flip_anneal( map, nsims, warmup = 0, init_plan = NULL, constraints = redist_constr(), num_hot_steps = 40000, num_annealing_steps = 60000, num_cold_steps = 20000, eprob = 0.05, lambda = 0, adapt_lambda = FALSE, adapt_eprob = FALSE, exact_mh = FALSE, maxiterrsg = 5000, verbose = TRUE )
redist_flip_anneal( map, nsims, warmup = 0, init_plan = NULL, constraints = redist_constr(), num_hot_steps = 40000, num_annealing_steps = 60000, num_cold_steps = 20000, eprob = 0.05, lambda = 0, adapt_lambda = FALSE, adapt_eprob = FALSE, exact_mh = FALSE, maxiterrsg = 5000, verbose = TRUE )
map |
A |
nsims |
The number of samples to draw, not including warmup. |
warmup |
The number of warmup samples to discard. |
init_plan |
A vector containing the congressional district labels
of each geographic unit. The default is |
constraints |
A |
num_hot_steps |
The number of steps to run the simulator at beta = 0. Default is 40000. |
num_annealing_steps |
The number of steps to run the simulator with linearly changing beta schedule. Default is 60000 |
num_cold_steps |
The number of steps to run the simulator at beta = 1. Default is 20000. |
eprob |
The probability of keeping an edge connected. The
default is |
lambda |
The parameter determining the number of swaps to attempt
each iteration of the algorithm. The number of swaps each iteration is
equal to Pois( |
adapt_lambda |
Whether to adaptively tune the lambda parameter so that the Metropolis-Hastings acceptance probability falls between 20% and 40%. Default is FALSE. |
adapt_eprob |
Whether to adaptively tune the edgecut probability parameter so that the Metropolis-Hastings acceptance probability falls between 20% and 40%. Default is FALSE. |
exact_mh |
Whether to use the approximate (0) or exact (1) Metropolis-Hastings ratio calculation for accept-reject rule. Default is FALSE. |
maxiterrsg |
Maximum number of iterations for random seed-and-grow algorithm to generate starting values. Default is 5000. |
verbose |
Whether to print initialization statement.
Default is |
redist_plans
redist_map
object.Sets up a redistricting problem.
redist_map( ..., existing_plan = NULL, pop_tol = NULL, total_pop = c("pop", "population", "total_pop", "POP100"), ndists = NULL, pop_bounds = NULL, adj = NULL, adj_col = "adj", planarize = 3857 ) as_redist_map(x)
redist_map( ..., existing_plan = NULL, pop_tol = NULL, total_pop = c("pop", "population", "total_pop", "POP100"), ndists = NULL, pop_bounds = NULL, adj = NULL, adj_col = "adj", planarize = 3857 ) as_redist_map(x)
... |
column elements to be bound into a |
existing_plan |
|
pop_tol |
|
total_pop |
|
ndists |
|
pop_bounds |
|
adj |
the adjacency graph for the object. Defaults to being computed from the data if it is coercible to a shapefile. |
adj_col |
the name of the adjacency graph column |
planarize |
a number, indicating the CRS to project the shapefile to if it is latitude-longitude based. Set to NULL or FALSE to avoid planarizing. |
x |
an object to be coerced |
A redist_map
object is a tibble
which contains an
adjacency list and additional information about the number of districts and
population bounds. It supports all of the dplyr
generics, and will
adjust the adjacency list and attributes according to these functions; i.e.,
if we filter
to a subset of units, the graph will change to subset to
these units, and the population bounds will adjust accordingly. If an
existing map is also attached to the object, the number of districts will
also adjust. Subsetting with `[`
and `[[`
does not recompute
graphs or attributes.
Other useful methods for redist_map
objects:
A redist_map object
data(fl25) d <- redist_map(fl25, ndists = 3, pop_tol = 0.05, total_pop = pop) dplyr::filter(d, pop >= 10e3)
data(fl25) d <- redist_map(fl25, ndists = 3, pop_tol = 0.05, total_pop = pop) dplyr::filter(d, pop >= 10e3)
redist_mergesplit
uses a Markov Chain Monte Carlo algorithm (Carter et
al. 2019; based on DeFord et. al 2019) to generate congressional or legislative redistricting plans
according to contiguity, population, compactness, and administrative boundary
constraints. The MCMC proposal is the same as is used in the SMC sampler
(McCartan and Imai 2023); it is similar but not identical to those used in
the references. 1-level hierarchical Merge-split is supported through the
counties
parameter; unlike in the SMC algorithm, this does not
guarantee a maximum number of county splits.
redist_mergesplit( map, nsims, warmup = if (is.null(init_plan)) 10 else max(100, nsims%/%5), thin = 1L, init_plan = NULL, counties = NULL, compactness = 1, constraints = list(), constraint_fn = function(m) rep(0, ncol(m)), adapt_k_thresh = 0.99, k = NULL, init_name = NULL, verbose = FALSE, silent = FALSE )
redist_mergesplit( map, nsims, warmup = if (is.null(init_plan)) 10 else max(100, nsims%/%5), thin = 1L, init_plan = NULL, counties = NULL, compactness = 1, constraints = list(), constraint_fn = function(m) rep(0, ncol(m)), adapt_k_thresh = 0.99, k = NULL, init_name = NULL, verbose = FALSE, silent = FALSE )
map |
A |
nsims |
The number of samples to draw, including warmup. |
warmup |
The number of warmup samples to discard. Recommended to be at least the first 20% of samples, and in any case no less than around 100 samples, unless initializing from a random plan. |
thin |
Save every |
init_plan |
The initial state of the map. If not provided, will default to
the reference map of the |
counties |
A vector containing county (or other administrative or
geographic unit) labels for each unit, which may be integers ranging from 1
to the number of counties, or a factor or character vector. If provided,
the algorithm will generate maps tend to follow county lines. There is no
strength parameter associated with this constraint. To adjust the number of
county splits further, or to constrain a second type of administrative
split, consider using |
compactness |
Controls the compactness of the generated districts, with higher values preferring more compact districts. Must be nonnegative. See the 'Details' section for more information, and computational considerations. |
constraints |
A list containing information on constraints to implement. See the 'Details' section for more information. |
constraint_fn |
A function which takes in a matrix where each column is a redistricting plan and outputs a vector of log-weights, which will be added the the final weights. |
adapt_k_thresh |
The threshold value used in the heuristic to select a
value |
k |
The number of edges to consider cutting after drawing a spanning tree. Should be selected automatically in nearly all cases. |
init_name |
a name for the initial plan, or |
verbose |
Whether to print out intermediate information while sampling. Recommended. |
silent |
Whether to suppress all diagnostic information. |
This function draws samples from a specific target measure, controlled by the
map
, compactness
, and constraints
parameters.
Key to ensuring good performance is monitoring the acceptance rate, which
is reported at the sample level in the output.
Users should also check diagnostics of the sample by running
summary.redist_plans()
.
Higher values of compactness
sample more compact districts;
setting this parameter to 1 is computationally efficient and generates nicely
compact districts.
redist_mergesplit
returns an object of class
redist_plans
containing the simulated plans.
Carter, D., Herschlag, G., Hunter, Z., and Mattingly, J. (2019). A merge-split proposal for reversible Monte Carlo Markov chain sampling of redistricting plans. arXiv preprint arXiv:1911.01503.
McCartan, C., & Imai, K. (2023). Sequential Monte Carlo for Sampling Balanced and Compact Redistricting Plans. Annals of Applied Statistics 17(4). Available at doi:10.1214/23-AOAS1763.
DeFord, D., Duchin, M., and Solomon, J. (2019). Recombination: A family of Markov chains for redistricting. arXiv preprint arXiv:1911.05725.
data(fl25) fl_map <- redist_map(fl25, ndists = 3, pop_tol = 0.1) sampled_basic <- redist_mergesplit(fl_map, 10000) sampled_constr <- redist_mergesplit(fl_map, 10000, constraints = list( incumbency = list(strength = 1000, incumbents = c(3, 6, 25)) ))
data(fl25) fl_map <- redist_map(fl25, ndists = 3, pop_tol = 0.1) sampled_basic <- redist_mergesplit(fl_map, 10000) sampled_constr <- redist_mergesplit(fl_map, 10000, constraints = list( incumbency = list(strength = 1000, incumbents = c(3, 6, 25)) ))
redist_mergesplit_parallel()
runs redist_mergesplit()
on several
chains in parallel.
redist_mergesplit_parallel( map, nsims, chains = 1, warmup = if (is.null(init_plan)) 10 else max(100, nsims%/%5), thin = 1L, init_plan = NULL, counties = NULL, compactness = 1, constraints = list(), constraint_fn = function(m) rep(0, ncol(m)), adapt_k_thresh = 0.99, k = NULL, ncores = NULL, cl_type = "PSOCK", return_all = TRUE, init_name = NULL, verbose = FALSE, silent = FALSE )
redist_mergesplit_parallel( map, nsims, chains = 1, warmup = if (is.null(init_plan)) 10 else max(100, nsims%/%5), thin = 1L, init_plan = NULL, counties = NULL, compactness = 1, constraints = list(), constraint_fn = function(m) rep(0, ncol(m)), adapt_k_thresh = 0.99, k = NULL, ncores = NULL, cl_type = "PSOCK", return_all = TRUE, init_name = NULL, verbose = FALSE, silent = FALSE )
map |
A |
nsims |
The number of samples to draw, including warmup. |
chains |
the number of parallel chains to run. Each chain will have
|
warmup |
The number of warmup samples to discard. Recommended to be at least the first 20% of samples, and in any case no less than around 100 samples, unless initializing from a random plan. |
thin |
Save every |
init_plan |
The initial state of the map, provided as a single vector
to be shared across all chains, or a matrix with |
counties |
A vector containing county (or other administrative or
geographic unit) labels for each unit, which may be integers ranging from 1
to the number of counties, or a factor or character vector. If provided,
the algorithm will generate maps tend to follow county lines. There is no
strength parameter associated with this constraint. To adjust the number of
county splits further, or to constrain a second type of administrative
split, consider using |
compactness |
Controls the compactness of the generated districts, with higher values preferring more compact districts. Must be nonnegative. See the 'Details' section for more information, and computational considerations. |
constraints |
A list containing information on constraints to implement. See the 'Details' section for more information. |
constraint_fn |
A function which takes in a matrix where each column is a redistricting plan and outputs a vector of log-weights, which will be added the the final weights. |
adapt_k_thresh |
The threshold value used in the heuristic to select a
value |
k |
The number of edges to consider cutting after drawing a spanning tree. Should be selected automatically in nearly all cases. |
ncores |
the number of parallel processes to run. Defaults to the maximum available. |
cl_type |
the cluster type (see |
return_all |
if |
init_name |
a name for the initial plan, or |
verbose |
Whether to print out intermediate information while sampling. Recommended. |
silent |
Whether to suppress all diagnostic information. |
This function draws samples from a specific target measure, controlled by the
map
, compactness
, and constraints
parameters.
Key to ensuring good performance is monitoring the acceptance rate, which
is reported at the sample level in the output.
Users should also check diagnostics of the sample by running
summary.redist_plans()
.
Higher values of compactness
sample more compact districts;
setting this parameter to 1 is computationally efficient and generates nicely
compact districts.
A redist_plans
object with all of the simulated plans, and an
additional chain
column indicating the chain the plan was drawn from.
Carter, D., Herschlag, G., Hunter, Z., and Mattingly, J. (2019). A merge-split proposal for reversible Monte Carlo Markov chain sampling of redistricting plans. arXiv preprint arXiv:1911.01503.
McCartan, C., & Imai, K. (2023). Sequential Monte Carlo for Sampling Balanced and Compact Redistricting Plans. Annals of Applied Statistics 17(4). Available at doi:10.1214/23-AOAS1763.
DeFord, D., Duchin, M., and Solomon, J. (2019). Recombination: A family of Markov chains for redistricting. arXiv preprint arXiv:1911.05725.
## Not run: data(fl25) fl_map <- redist_map(fl25, ndists = 3, pop_tol = 0.1) sampled <- redist_mergesplit_parallel(fl_map, nsims = 100, chains = 100) ## End(Not run)
## Not run: data(fl25) fl_map <- redist_map(fl25, ndists = 3, pop_tol = 0.1) sampled <- redist_mergesplit_parallel(fl_map, nsims = 100, chains = 100) ## End(Not run)
A redist_plans
object is essentially a data frame of summary
information on each district and each plan, along with the matrix of district
assignments and information about the simulation process used to generate the
plans.
redist_plans(plans, map, algorithm, wgt = NULL, ...)
redist_plans(plans, map, algorithm, wgt = NULL, ...)
plans |
a matrix with |
map |
a |
algorithm |
the algorithm used to generate the plans (usually "smc" or "mcmc") |
wgt |
the weights to use, if any. |
... |
Other named attributes to set |
The first two columns of the data frame will be draw
, a factor indexing
the simulation draw, and district
, an integer indexing the districts
within a plan. The data frame will therefore have n_sims*ndists
rows.
As a data frame, the usual dplyr
methods will work.
Other useful methods for redist_plans
objects:
a new redist_plans
object.
data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) rsg_plan <- redist.rsg(iowa$adj, iowa$pop, ndists = 4, pop_tol = 0.05)$plan redist_plans(rsg_plan, iowa, "rsg")
data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) rsg_plan <- redist.rsg(iowa$adj, iowa$pop, ndists = 4, pop_tol = 0.05)$plan redist_plans(rsg_plan, iowa, "rsg")
Defined as pmin(x, quantile(x, 1 - length(x)^(-0.5)))
redist_quantile_trunc(x)
redist_quantile_trunc(x)
x |
the weights |
numeric vector
redist_quantile_trunc(c(1, 2, 3, 4))
redist_quantile_trunc(c(1, 2, 3, 4))
This function uses redist_mergesplit()
or redist_flip()
to optimize a
redistrict plan according to a user-provided criteria. It does so by running
the Markov chain for "short bursts" of usually 10 iterations, and then
starting the chain anew from the best plan in the burst, according to the
criteria. This implements the ideas in the below-referenced paper, "Voting
Rights, Markov Chains, and Optimization by Short Bursts."
redist_shortburst( map, score_fn = NULL, stop_at = NULL, burst_size = ifelse(backend == "mergesplit", 10L, 50L), max_bursts = 500L, maximize = TRUE, init_plan = NULL, counties = NULL, constraints = redist_constr(map), compactness = 1, adapt_k_thresh = 0.95, reversible = TRUE, fixed_k = NULL, return_all = TRUE, thin = 1L, backend = "mergesplit", flip_lambda = 0, flip_eprob = 0.05, verbose = TRUE )
redist_shortburst( map, score_fn = NULL, stop_at = NULL, burst_size = ifelse(backend == "mergesplit", 10L, 50L), max_bursts = 500L, maximize = TRUE, init_plan = NULL, counties = NULL, constraints = redist_constr(map), compactness = 1, adapt_k_thresh = 0.95, reversible = TRUE, fixed_k = NULL, return_all = TRUE, thin = 1L, backend = "mergesplit", flip_lambda = 0, flip_eprob = 0.05, verbose = TRUE )
map |
A redist_map object. |
score_fn |
A function which takes a matrix of plans and returns a score
(or, generally, a row vector) for each plan. Can also be a purrr-style
anonymous function. See |
stop_at |
A threshold to stop optimization at. When |
burst_size |
The size of each burst. 10 is recommended for the
|
max_bursts |
The maximum number of bursts to run before returning. |
maximize |
If |
init_plan |
The initial state of the map. If not provided, will default to
the reference map of the |
counties |
A vector containing county (or other administrative or
geographic unit) labels for each unit, which may be integers ranging from 1
to the number of counties, or a factor or character vector. If provided, the
algorithm will only generate maps which split up to |
constraints |
A |
compactness |
Controls the compactness of the generated districts, with
higher values preferring more compact districts. Must be non-negative. See
|
adapt_k_thresh |
The threshold value used in the heuristic to select a
value |
reversible |
If |
fixed_k |
If not |
return_all |
Whether to return all the burst results or just the best one (generally, the Pareto frontier). Recommended for monitoring purposes. |
thin |
Save every |
backend |
the MCMC algorithm to use within each burst, either "mergesplit" or "flip". |
flip_lambda |
The parameter determining the number of swaps to attempt each iteration of flip mcmc. The number of swaps each iteration is equal to Pois(lambda) + 1. The default is 0. |
flip_eprob |
The probability of keeping an edge connected in flip mcmc. The default is 0.05. |
verbose |
Whether to print out intermediate information while sampling. Recommended for monitoring purposes. |
a redist_plans
object containing the final best plan
(or the best plans after each burst, if return_all=TRUE
.
Cannon, S., Goldbloom-Helzner, A., Gupta, V., Matthews, J. N., & Suwal, B. (2020). Voting Rights, Markov Chains, and Optimization by Short Bursts. arXiv preprint arXiv:2011.02288.
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) redist_shortburst(iowa_map, scorer_frac_kept(iowa_map), max_bursts = 50) redist_shortburst(iowa_map, ~ 1 - scorer_frac_kept(iowa_map)(.), max_bursts = 50)
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) redist_shortburst(iowa_map, scorer_frac_kept(iowa_map), max_bursts = 50) redist_shortburst(iowa_map, ~ 1 - scorer_frac_kept(iowa_map)(.), max_bursts = 50)
redist_smc
uses a Sequential Monte Carlo algorithm (McCartan and Imai 2023)
to generate representative samples of congressional or legislative
redistricting plans according to contiguity, population, compactness, and
administrative boundary constraints.
redist_smc( map, nsims, counties = NULL, compactness = 1, constraints = list(), resample = TRUE, runs = 1L, ncores = 0L, init_particles = NULL, n_steps = NULL, adapt_k_thresh = 0.99, seq_alpha = 0.5, truncate = (compactness != 1), trunc_fn = redist_quantile_trunc, pop_temper = 0, final_infl = 1, est_label_mult = 1, ref_name = NULL, verbose = FALSE, silent = FALSE )
redist_smc( map, nsims, counties = NULL, compactness = 1, constraints = list(), resample = TRUE, runs = 1L, ncores = 0L, init_particles = NULL, n_steps = NULL, adapt_k_thresh = 0.99, seq_alpha = 0.5, truncate = (compactness != 1), trunc_fn = redist_quantile_trunc, pop_temper = 0, final_infl = 1, est_label_mult = 1, ref_name = NULL, verbose = FALSE, silent = FALSE )
map |
A |
nsims |
The number of samples to draw. |
counties |
A vector containing county (or other administrative or
geographic unit) labels for each unit, which may be integers ranging from 1
to the number of counties, or a factor or character vector. If provided,
the algorithm will only generate maps which split up to |
compactness |
Controls the compactness of the generated districts, with higher values preferring more compact districts. Must be nonnegative. See the 'Details' section for more information, and computational considerations. |
constraints |
A |
resample |
Whether to perform a final resampling step so that the
generated plans can be used immediately. Set this to |
runs |
How many independent parallel runs to conduct. Each run will
have |
ncores |
How many cores to use to parallelize plan generation within each
run. The default, 0, will use the number of available cores on the machine
as long as |
init_particles |
A matrix of partial plans to begin sampling from. For
advanced use only. The matrix must have |
n_steps |
How many steps to run the SMC algorithm for. Each step splits off a new district. Defaults to all remaining districts. If fewer than the number of remaining splits, reference plans are disabled. |
adapt_k_thresh |
The threshold value used in the heuristic to select a
value |
seq_alpha |
The amount to adjust the weights by at each resampling step; higher values prefer exploitation, while lower values prefer exploration. Must be between 0 and 1. |
truncate |
Whether to truncate the importance sampling weights at the
final step by |
trunc_fn |
A function which takes in a vector of weights and returns a truncated vector. If the loo package is installed (strongly recommended), will default to Pareto-smoothed Importance Sampling (PSIS) rather than naive truncation. |
pop_temper |
The strength of the automatic population tempering. Try values of 0.01-0.05 to start if the algorithm gets stuck on the final few splits. |
final_infl |
A multiplier for the population constraint on the final
iteration. Used to loosen the constraint when the sampler is getting stuck
on the final split. |
est_label_mult |
A multiplier for the number of importance samples to use in estimating the number of ways to sequentially label the districts. Lower values increase speed at the cost of accuracy. Only applied when there are more than 13 districts. |
ref_name |
a name for the existing plan, which will be added as a
reference plan, or |
verbose |
Whether to print out intermediate information while sampling. Recommended. |
silent |
Whether to suppress all diagnostic information. |
This function draws samples from a specific target measure controlled by
the map
, compactness
, and constraints
parameters.
Key to ensuring good performance is monitoring the efficiency of the resampling
process at each SMC stage. Unless silent=FALSE
, this function will print
out the effective sample size of each resampling step to allow the user to
monitor the efficiency. If verbose=TRUE
the function will also print
out information on the values automatically chosen and the
acceptance rate (based on the population constraint) at each step.
Users should also check diagnostics of the sample by running
summary.redist_plans()
.
Higher values of compactness
sample more compact districts;
setting this parameter to 1 is computationally efficient and generates nicely
compact districts. Values of other than 1 may lead to highly variable
importance sampling weights. In these cases, these weights are by default
truncated using redist_quantile_trunc()
to stabilize the resulting
estimates, but if truncation is used, a specific truncation function should
probably be chosen by the user.
redist_smc
returns a redist_plans object containing the simulated
plans.
McCartan, C., & Imai, K. (2023). Sequential Monte Carlo for Sampling Balanced and Compact Redistricting Plans. Annals of Applied Statistics 17(4). Available at doi:10.1214/23-AOAS1763.
data(fl25) fl_map <- redist_map(fl25, ndists = 3, pop_tol = 0.1) sampled_basic <- redist_smc(fl_map, 5000) constr <- redist_constr(fl_map) constr <- add_constr_incumbency(constr, strength = 100, incumbents = c(3, 6, 25)) sampled_constr <- redist_smc(fl_map, 5000, constraints = constr) # Multiple parallel independent runs redist_smc(fl_map, 1000, runs = 2) # One run with multiple cores redist_smc(fl_map, 1000, ncores = 2)
data(fl25) fl_map <- redist_map(fl25, ndists = 3, pop_tol = 0.1) sampled_basic <- redist_smc(fl_map, 5000) constr <- redist_constr(fl_map) constr <- add_constr_incumbency(constr, strength = 100, incumbents = c(3, 6, 25)) sampled_constr <- redist_smc(fl_map, 5000, constraints = constr) # Multiple parallel independent runs redist_smc(fl_map, 1000, runs = 2) # One run with multiple cores redist_smc(fl_map, 1000, ncores = 2)
Creates an adjacency list that is zero indexed with no skips
redist.adjacency(shp, plan)
redist.adjacency(shp, plan)
shp |
A SpatialPolygonsDataFrame or sf object. Required. |
plan |
A numeric vector (if only one map) or matrix with one row |
Adjacency list
Calculate Frontier Size
redist.calc.frontier.size(ordered_path)
redist.calc.frontier.size(ordered_path)
ordered_path |
path to ordered path created by redist.prep.enumpart |
List, four objects
max
numeric, maximum frontier size
average
numeric, average frontier size
average_sq
numeric, average((frontier size)^2)
sequence
numeric vector, lists out all sizes for every frontier
## Not run: data(fl25) adj <- redist.adjacency(fl25) redist.prep.enumpart(adj, "unordered", "ordered") redist.calc.frontier.size("ordered") ## End(Not run)
## Not run: data(fl25) adj <- redist.adjacency(fl25) redist.prep.enumpart(adj, "unordered", "ordered") redist.calc.frontier.size("ordered") ## End(Not run)
Coarsen Adjacency List
redist.coarsen.adjacency(adj, groups)
redist.coarsen.adjacency(adj, groups)
adj |
A zero-indexed adjacency list. Required. |
groups |
integer vector of elements of adjacency to group |
adjacency list coarsened
redist.mcmc.mpi
redist.combine.mpi
is used to combine successive runs of
redist.mcmc.mpi
into a single data object
redist.combine.mpi(savename, nloop, nthin, tempadj)
redist.combine.mpi(savename, nloop, nthin, tempadj)
savename |
The name (without the loop or |
nloop |
The number of loops being combined. |
nthin |
How much to thin the simulations being combined. |
tempadj |
The temperature adjacency object saved by
|
This function allows users to combine multiple successive runs of
redist.mcmc.mpi
into a single redist
object for analysis.
redist.combine.mpi
returns an object of class "redist".
The object redist
is a list that contains the following components (the
inclusion of some components is dependent on whether tempering
techniques are used):
plans |
Matrix of congressional district assignments generated by the algorithm. Each row corresponds to a geographic unit, and each column corresponds to a simulation. |
distance_parity |
Vector containing the maximum distance from parity for a particular simulated redistricting plan. |
mhdecisions |
A vector specifying whether a proposed redistricting plan was accepted (1) or rejected (0) in a given iteration. |
mhprob |
A vector containing the Metropolis-Hastings acceptance probability for each iteration of the algorithm. |
pparam |
A vector containing the draw of the |
constraint_pop |
A vector containing the value of the population constraint for each accepted redistricting plan. |
constraint_compact |
A vector containing the value of the compactness constraint for each accepted redistricting plan. |
constraint_vra |
A vector containing the value of the vra constraint for each accepted redistricting plan. |
constraint_similar |
A vector containing the value of the similarity constraint for each accepted redistricting plan. |
constraint_qps |
A vector containing the value of the QPS constraint for each accepted redistricting plan. |
beta_sequence |
A vector containing the value of beta for each iteration of the algorithm. Returned when tempering is being used. |
mhdecisions_beta |
A vector specifying whether a proposed beta value was accepted (1) or rejected (0) in a given iteration of the algorithm. Returned when tempering is being used. |
mhprob_beta |
A vector containing the Metropolis-Hastings acceptance probability for each iteration of the algorithm. Returned when tempering is being used. |
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
## Not run: # Cannot run on machines without Rmpi data(fl25) data(fl25_enum) data(fl25_adj) ## Code to run the simulations in Figure 4 in Fifield, Higgins, Imai and ## Tarr (2015) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] ## Run the algorithm redist.mcmc.mpi(adj = fl25_adj, total_pop = fl25$pop, init_plan = init_plan, nsims = 10000, nloops = 2, savename = "test") out <- redist.combine.mpi(savename = "test", nloop = 2, nthin = 10, tempadj = tempAdjMat) ## End(Not run)
## Not run: # Cannot run on machines without Rmpi data(fl25) data(fl25_enum) data(fl25_adj) ## Code to run the simulations in Figure 4 in Fifield, Higgins, Imai and ## Tarr (2015) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] ## Run the algorithm redist.mcmc.mpi(adj = fl25_adj, total_pop = fl25$pop, init_plan = init_plan, nsims = 10000, nloops = 2, savename = "test") out <- redist.combine.mpi(savename = "test", nloop = 2, nthin = 10, tempadj = tempAdjMat) ## End(Not run)
Create Constraints for SMC
redist.constraint.helper( constraints = "vra", tgt_min = 0.55, group_pop, total_pop, ndists, nmmd, strength_vra = 2500, pow_vra = 1.5 )
redist.constraint.helper( constraints = "vra", tgt_min = 0.55, group_pop, total_pop, ndists, nmmd, strength_vra = 2500, pow_vra = 1.5 )
constraints |
Vector of constraints to include. Currently only 'vra' implemented. |
tgt_min |
Defaults to 0.55. If 'vra' included, the minority percent to encourage in each district. |
group_pop |
A vector of populations for some subgroup of interest. |
total_pop |
A vector containing the populations of each geographic unit. |
ndists |
The total number of districts. |
nmmd |
The number of majority minority districts to target for 'vra' constraint |
strength_vra |
The strength of the 'vra' constraint. Defaults to 2500. |
pow_vra |
The exponent for the 'vra' constraint. Defaults to 1.5. |
list of lists for each constraint selected
Create County IDs
redist.county.id(counties)
redist.county.id(counties)
counties |
vector of counties, required. |
A vector with an ID that corresponds from 1:n counties
set.seed(2) counties <- sample(c(rep("a", 20), rep("b", 5))) redist.county.id(counties)
set.seed(2) counties <- sample(c(rep("a", 20), rep("b", 5))) redist.county.id(counties)
Relabel Discontinuous Counties
redist.county.relabel(adj, counties, simplify = TRUE)
redist.county.relabel(adj, counties, simplify = TRUE)
adj |
adjacency list |
counties |
character vector of county names |
simplify |
boolean - TRUE returns a numeric vector of ids, while FALSE appends a number when there are multiple connected components. |
character vector of county names
set.seed(2) data(fl25) data(fl25_adj) counties <- sample(c(rep("a", 20), rep("b", 5))) redist.county.relabel(fl25_adj, counties)
set.seed(2) data(fl25) data(fl25_adj) counties <- sample(c(rep("a", 20), rep("b", 5))) redist.county.relabel(fl25_adj, counties)
redist.crsg
generates redistricting plans using a random seed a grow
algorithm. This is the compact districting algorithm described in Chen and
Rodden (2013).
redist.crsg( adj, total_pop, shp, ndists, pop_tol, verbose = TRUE, maxiter = 5000 )
redist.crsg( adj, total_pop, shp, ndists, pop_tol, verbose = TRUE, maxiter = 5000 )
adj |
List of length N, where N is the number of precincts. Each list element is an integer vector indicating which precincts that precinct is adjacent to. It is assumed that precinct numbers start at 0. |
total_pop |
numeric vector of length N, where N is the number of precincts. Each element lists the population total of the corresponding precinct, and is used to enforce pop_tol constraints. |
shp |
An sf dataframe to compute area and centroids with. |
ndists |
integer, the number of districts we want to partition the precincts into. |
pop_tol |
numeric, indicating how close district population targets have to be to the target population before algorithm converges. pop_tol=0.05 for example means that all districts must be between 0.95 and 1.05 times the size of target.pop in population size. |
verbose |
boolean, indicating whether the time to run the algorithm is printed. |
maxiter |
integer, indicating maximum number of iterations to attempt before convergence to population constraint fails. If it fails once, it will use a different set of start values and try again. If it fails again, redist.rsg() returns an object of all NAs, indicating that use of more iterations may be advised. Default is 5000. |
list, containing three objects containing the completed redistricting plan.
plan
: A vector of length N, indicating the
district membership of each precinct.
district_list
A list of length Ndistrict. Each list contains a
vector of the precincts in the respective district.
district_pop
A vector of length Ndistrict, containing the
population totals of the respective districts.
Jowei Chen and Jonathan Rodden (2013) “Unintentional Gerrymandering: Political Geography and Electoral Bias in Legislatures.” Quarterly Journal of Political Science. 8(3): 239-269.
data("fl25") adj <- redist.adjacency(fl25) redist.crsg(adj = adj, total_pop = fl25$pop, shp = fl25, ndists = 2, pop_tol = .1)
data("fl25") adj <- redist.adjacency(fl25) redist.crsg(adj = adj, total_pop = fl25$pop, shp = fl25, ndists = 2, pop_tol = .1)
redist.diagplot
generates several common MCMC diagnostic plots.
redist.diagplot(sumstat, plot = c("trace", "autocorr", "densplot", "mean", "gelmanrubin"), logit = FALSE, savename = NULL)
redist.diagplot(sumstat, plot = c("trace", "autocorr", "densplot", "mean", "gelmanrubin"), logit = FALSE, savename = NULL)
sumstat |
A vector, list, |
plot |
The type of diagnostic plot to generate: one of "trace",
"autocorr", "densplot", "mean", "gelmanrubin". If |
logit |
Flag for whether to apply the logistic transformation for the
summary statistic. The default is |
savename |
Filename to save the plot. Default is |
This function allows users to generate several standard diagnostic plots from the MCMC literature, as implemented by Plummer et. al (2006). Diagnostic plots implemented include trace plots, autocorrelation plots, density plots, running means, and Gelman-Rubin convergence diagnostics (Gelman & Rubin 1992).
Returns a plot of file type .pdf
.
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
Gelman, Andrew and Donald Rubin. (1992) "Inference from iterative simulations using multiple sequences (with discussion)." Statistical Science.
Plummer, Martin, Nicky Best, Kate Cowles and Karen Vines. (2006) "CODA: Convergence Diagnosis and Output Analysis for MCMC." R News.
data(fl25) data(fl25_enum) data(fl25_adj) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] fl25$init_plan <- init_plan ## 25 precinct, three districts - no pop constraint ## fl_map <- redist_map(fl25, existing_plan = 'init_plan', adj = fl25_adj) alg_253 <- redist_flip(fl_map, nsims = 10000) ## Get Republican Dissimilarity Index from simulations rep_dmi_253 <- redistmetrics::seg_dissim(alg_253, fl25, mccain, pop) |> redistmetrics::by_plan(ndists = 3) ## Generate diagnostic plots redist.diagplot(rep_dmi_253, plot = "trace") redist.diagplot(rep_dmi_253, plot = "autocorr") redist.diagplot(rep_dmi_253, plot = "densplot") redist.diagplot(rep_dmi_253, plot = "mean") ## Gelman Rubin needs two chains, so we run a second alg_253_2 <- redist_flip(fl_map, nsims = 10000) rep_dmi_253_2 <- redistmetrics::seg_dissim(alg_253_2, fl25, mccain, pop) |> redistmetrics::by_plan(ndists = 3) ## Make a list out of the objects: rep_dmi_253_list <- list(rep_dmi_253, rep_dmi_253_2) ## Generate Gelman Rubin diagnostic plot redist.diagplot(sumstat = rep_dmi_253_list, plot = "gelmanrubin")
data(fl25) data(fl25_enum) data(fl25_adj) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] fl25$init_plan <- init_plan ## 25 precinct, three districts - no pop constraint ## fl_map <- redist_map(fl25, existing_plan = 'init_plan', adj = fl25_adj) alg_253 <- redist_flip(fl_map, nsims = 10000) ## Get Republican Dissimilarity Index from simulations rep_dmi_253 <- redistmetrics::seg_dissim(alg_253, fl25, mccain, pop) |> redistmetrics::by_plan(ndists = 3) ## Generate diagnostic plots redist.diagplot(rep_dmi_253, plot = "trace") redist.diagplot(rep_dmi_253, plot = "autocorr") redist.diagplot(rep_dmi_253, plot = "densplot") redist.diagplot(rep_dmi_253, plot = "mean") ## Gelman Rubin needs two chains, so we run a second alg_253_2 <- redist_flip(fl_map, nsims = 10000) rep_dmi_253_2 <- redistmetrics::seg_dissim(alg_253_2, fl25, mccain, pop) |> redistmetrics::by_plan(ndists = 3) ## Make a list out of the objects: rep_dmi_253_list <- list(rep_dmi_253, rep_dmi_253_2) ## Generate Gelman Rubin diagnostic plot redist.diagplot(sumstat = rep_dmi_253_list, plot = "gelmanrubin")
This implements Crespin's 2005 measure of district continuity, as applied to the geographies represented by a plan, typically precincts or voting districts. This implementation assumes none of the precincts in plan_old or plan_new are split.
redist.dist.pop.overlap(plan_old, plan_new, total_pop, normalize_rows = TRUE)
redist.dist.pop.overlap(plan_old, plan_new, total_pop, normalize_rows = TRUE)
plan_old |
The reference or original plan to compare against |
plan_new |
The new plan to compare to the reference plan |
total_pop |
The total population by precinct This can also take a redist_map object and will use the population in that object. If nothing is provided, it weights all entries in plan equally. |
normalize_rows |
Default TRUE. Normalize populations by row. If FALSE, normalizes by column. If NULL, does not normalize. |
matrix with length(unique(plan_old)) rows and length(unique(plan_new)) columns
"Using Geographic Information Systems to Measure District Change, 2000-02", Michael Crespin, Political Analysis (2005) 13(3): 253-260
set.seed(5) data(iowa) iowa_map <- redist_map(iowa, total_pop = pop, pop_tol = 0.01, ndists = 4) plans <- redist_smc(iowa_map, 2) plans_mat <- get_plans_matrix(plans) ov <- redist.dist.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map) round(ov, 2) ov_col <- redist.dist.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map, normalize_rows = FALSE) round(ov_col, 2) ov_un_norm <- redist.dist.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map, normalize_rows = NULL) round(ov_un_norm, 2) iowa_map_5 <- iowa_map <- redist_map(iowa, total_pop = pop, pop_tol = 0.01, ndists = 5) plan_5 <- get_plans_matrix(redist_smc(iowa_map_5, 1)) ov4_5 <- redist.dist.pop.overlap(plans_mat[, 1], plan_5, iowa_map) round(ov4_5, 2)
set.seed(5) data(iowa) iowa_map <- redist_map(iowa, total_pop = pop, pop_tol = 0.01, ndists = 4) plans <- redist_smc(iowa_map, 2) plans_mat <- get_plans_matrix(plans) ov <- redist.dist.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map) round(ov, 2) ov_col <- redist.dist.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map, normalize_rows = FALSE) round(ov_col, 2) ov_un_norm <- redist.dist.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map, normalize_rows = NULL) round(ov_un_norm, 2) iowa_map_5 <- iowa_map <- redist_map(iowa, total_pop = pop, pop_tol = 0.01, ndists = 5) plan_5 <- get_plans_matrix(redist_smc(iowa_map_5, 1)) ov4_5 <- redist.dist.pop.overlap(plans_mat[, 1], plan_5, iowa_map) round(ov4_5, 2)
Counts the total number of counties that are found within a district. This does not subtract out the number of counties that are found completely within a district.
redist.district.splits(plans, counties)
redist.district.splits(plans, counties)
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. Required. |
counties |
A vector of county names or county ids. |
integer matrix where each district is a
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.01) plans <- redist_smc(ia, 50, silent = TRUE) #old redist.district.splits(plans, ia$region) splits_count(plans, ia, region)
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.01) plans <- redist_smc(ia, 50, silent = TRUE) #old redist.district.splits(plans, ia$region) splits_count(plans, ia, region)
Single function for standard enumeration analysis, using ZDD methodology (Fifield, Imai, Kawahara, and Kenny 2020).
redist.enumpart( adj, unordered_path, ordered_path, out_path, ndists = 2, all = TRUE, n = NULL, weight_path = NULL, lower = NULL, upper = NULL, init = FALSE, read = TRUE, total_pop = NULL )
redist.enumpart( adj, unordered_path, ordered_path, out_path, ndists = 2, all = TRUE, n = NULL, weight_path = NULL, lower = NULL, upper = NULL, init = FALSE, read = TRUE, total_pop = NULL )
adj |
zero indexed adjacency list. |
unordered_path |
valid path to output the unordered adjacency map to |
ordered_path |
valid path to output the ordered adjacency map to |
out_path |
Valid path to output the enumerated districts |
ndists |
number of districts to enumerate |
all |
boolean. TRUE outputs all districts. FALSE samples n districts. |
n |
integer. Number of districts to output if all is FALSE. Returns districts selected from uniform random distribution. |
weight_path |
A path (not including ".dat") to a space-delimited file containing a vector of
vertex weights, to be used along with |
lower |
A lower bound on each partition's total weight, implemented by rejection sampling. |
upper |
An upper bound on each partition's total weight. |
init |
Runs redist.init.enumpart. Defaults to false. Should be run on first use. |
read |
boolean. Defaults to TRUE. reads |
total_pop |
the vector of precinct populations |
List with entries district_membership and parity.
Fifield, B., Imai, K., Kawahara, J., & Kenny, C. T. (2020). The essential role of empirical validation in legislative redistricting simulation. Statistics and Public Policy, 7(1), 52-68.
Given a percent goal for majority minority districts, this computes the average
value of minority in non-majority minority districts. This value is "tgt_other"
in redist_flip
and redist_smc
.
redist.find.target(tgt_min, group_pop, total_pop, ndists, nmmd)
redist.find.target(tgt_min, group_pop, total_pop, ndists, nmmd)
tgt_min |
target group population for majority minority district |
group_pop |
A vector of populations for some subgroup of interest. |
total_pop |
A vector containing the populations of each geographic unit. |
ndists |
The number of congressional districts. |
nmmd |
The number of majority minority districts. |
numeric value to target
redist_flip
redist.findparams
is used to find optimal parameter values of
redist_flip
for a given map.
redist.findparams( map, nsims, init_plan = NULL, adapt_lambda = FALSE, adapt_eprob = FALSE, params, ssdmat = NULL, group_pop = NULL, counties = NULL, nstartval_store = 1, maxdist_startval = 100, maxiterrsg = 5000, report_all = TRUE, parallel = FALSE, ncores = NULL, log = FALSE, verbose = TRUE )
redist.findparams( map, nsims, init_plan = NULL, adapt_lambda = FALSE, adapt_eprob = FALSE, params, ssdmat = NULL, group_pop = NULL, counties = NULL, nstartval_store = 1, maxdist_startval = 100, maxiterrsg = 5000, report_all = TRUE, parallel = FALSE, ncores = NULL, log = FALSE, verbose = TRUE )
map |
A |
nsims |
The number of simulations run before a save point. |
init_plan |
A vector containing the congressional district labels
of each geographic unit. The default is |
adapt_lambda |
Whether to adaptively tune the lambda parameter so that the Metropolis-Hastings acceptance probability falls between 20% and 40%. Default is FALSE. |
adapt_eprob |
Whether to adaptively tune the edgecut probability parameter so that the Metropolis-Hastings acceptance probability falls between 20% and 40%. Default is FALSE. |
params |
A matrix of parameter values to test, such as the output of
|
ssdmat |
A matrix of squared distances between geographic
units. The default is |
group_pop |
A vector of populations for some sub-group of
interest. The default is |
counties |
A vector of county membership assignments. The default is |
nstartval_store |
The number of maps to sample from the preprocessing chain for use as starting values in future simulations. Default is 1. |
maxdist_startval |
The maximum distance from the starting map that sampled maps should be. Default is 100 (no restriction). |
maxiterrsg |
Maximum number of iterations for random seed-and-grow algorithm to generate starting values. Default is 5000. |
report_all |
Whether to report all summary statistics for each set of
parameter values. Default is |
parallel |
Whether to run separate parameter settings in parallel.
Default is |
ncores |
Number of parallel tasks to run, declared outside of the
function. Default is |
log |
Whether to open a log to track progress for each parameter combination being tested. Default is FALSE. |
verbose |
Whether to print additional information about the tests.
Default is |
This function allows users to test multiple parameter settings of
redist_flip
in preparation for a longer run for analysis.
redist.findparams
returns a print-out of summary statistics
about each parameter setting.
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
data(fl25) data(fl25_enum) data(fl25_adj) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] params <- expand.grid(eprob = c(.01, .05, .1)) # Make map map_fl <- redist_map(fl25, ndists = 3, pop_tol = 0.2) ## Run the algorithm redist.findparams(map_fl, init_plan = init_plan, nsims = 10000, params = params)
data(fl25) data(fl25_enum) data(fl25_adj) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] params <- expand.grid(eprob = c(.01, .05, .1)) # Make map map_fl <- redist_map(fl25, ndists = 3, pop_tol = 0.2) ## Run the algorithm redist.findparams(map_fl, init_plan = init_plan, nsims = 10000, params = params)
This ensures that the enumerate partitions programs is prepared to run. This must be run once per install of the redist package.
redist.init.enumpart()
redist.init.enumpart()
0 on success
Benjamin Fifield, Kosuke Imai, Jun Kawahara, and Christopher T Kenny. "The Essential Role of Empirical Validation in Legislative Redistricting Simulation." Forthcoming, Statistics and Public Policy.
## Not run: redist.init.enumpart() ## End(Not run)
## Not run: redist.init.enumpart() ## End(Not run)
redist.ipw
properly weights and resamples simulated redistricting plans
so that the set of simulated plans resemble a random sample from the
underlying distribution. redist.ipw
is used to correct the sample when
population parity, geographic compactness, or other constraints are
implemented.
redist.ipw( plans, resampleconstraint = c("pop_dev", "edges_removed", "segregation", "status_quo"), targetbeta, targetpop = NULL, temper = 0 )
redist.ipw( plans, resampleconstraint = c("pop_dev", "edges_removed", "segregation", "status_quo"), targetbeta, targetpop = NULL, temper = 0 )
plans |
An object of class |
resampleconstraint |
The constraint implemented in the simulations: one of "pop", "compact", "segregation", or "similar". |
targetbeta |
The target value of the constraint. |
targetpop |
The desired level of population parity. |
temper |
A flag for whether simulated tempering was used to improve the
mixing of the Markov Chain. The default is |
This function allows users to resample redistricting plans using inverse probability weighting techniques described in Rubin (1987). This techniques reweights and resamples redistricting plans so that the resulting sample is representative of a random sample from the uniform distribution.
redist.ipw
returns an object of class "redist". The object
redist
is a list that contains the following components (the
inclusion of some components is dependent on whether tempering
techniques are used):
plans |
Matrix of congressional district assignments generated by the algorithm. Each row corresponds to a geographic unit, and each column corresponds to a simulation. |
distance_parity |
Vector containing the maximum distance from parity for a particular simulated redistricting plan. |
mhdecisions |
A vector specifying whether a proposed redistricting plan was accepted (1) or rejected (0) in a given iteration. |
mhprob |
A vector containing the Metropolis-Hastings acceptance probability for each iteration of the algorithm. |
pparam |
A vector containing the draw of the |
constraint_pop |
A vector containing the value of the population constraint for each accepted redistricting plan. |
constraint_compact |
A vector containing the value of the compactness constraint for each accepted redistricting plan. |
constraint_segregation |
A vector containing the value of the segregation constraint for each accepted redistricting plan. |
constraint_similar |
A vector containing the value of the similarity constraint for each accepted redistricting plan. |
constraint_vra |
A vector containing the value of the vra constraint for each accepted redistricting plan. |
constraint_partisan |
A vector containing the value of the partisan constraint for each accepted redistricting plan. |
constraint_minority |
A vector containing the value of the minority constraint for each accepted redistricting plan. |
constraint_hinge |
A vector containing the value of the hinge constraint for each accepted redistricting plan. |
constraint_qps |
A vector containing the value of the QPS constraint for each accepted redistricting plan. |
beta_sequence |
A vector containing the value of beta for each iteration of the algorithm. Returned when tempering is being used. |
mhdecisions_beta |
A vector specifying whether a proposed beta value was accepted (1) or rejected (0) in a given iteration of the algorithm. Returned when tempering is being used. |
mhprob_beta |
A vector containing the Metropolis-Hastings acceptance probability for each iteration of the algorithm. Returned when tempering is being used. |
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
Rubin, Donald. (1987) "Comment: A Noniterative Sampling/Importance Resampling Alternative to the Data Augmentation Algorithm for Creating a Few Imputations when Fractions of Missing Information are Modest: the SIR Algorithm." Journal of the American Statistical Association.
data(iowa) map_ia <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) cons <- redist_constr(map_ia) cons <- add_constr_pop_dev(cons, strength = 5.4) alg <- redist_flip(map_ia, nsims = 500, constraints = cons) alg_ipw <- redist.ipw(plans = alg, resampleconstraint = "pop_dev", targetbeta = 1, targetpop = 0.05)
data(iowa) map_ia <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) cons <- redist_constr(map_ia) cons <- add_constr_pop_dev(cons, strength = 5.4) alg <- redist_flip(map_ia, nsims = 500, constraints = cons) alg_ipw <- redist.ipw(plans = alg, resampleconstraint = "pop_dev", targetbeta = 1, targetpop = 0.05)
redist.mcmc.mpi
is used to simulate Congressional redistricting
plans using Markov Chain Monte Carlo methods.
redist.mcmc.mpi( adj, total_pop, nsims, ndists = NA, init_plan = NULL, loopscompleted = 0, nloop = 1, nthin = 1, eprob = 0.05, lambda = 0, pop_tol = NA, group_pop = NA, areasvec = NA, counties = NA, borderlength_mat = NA, ssdmat = NA, compactness_metric = "fryer-holden", rngseed = NA, constraint = NA, constraintweights = NA, betaseq = "powerlaw", betaseqlength = 10, adjswaps = TRUE, freq = 100, savename = NA, maxiterrsg = 5000, verbose = FALSE, cities = NULL )
redist.mcmc.mpi( adj, total_pop, nsims, ndists = NA, init_plan = NULL, loopscompleted = 0, nloop = 1, nthin = 1, eprob = 0.05, lambda = 0, pop_tol = NA, group_pop = NA, areasvec = NA, counties = NA, borderlength_mat = NA, ssdmat = NA, compactness_metric = "fryer-holden", rngseed = NA, constraint = NA, constraintweights = NA, betaseq = "powerlaw", betaseqlength = 10, adjswaps = TRUE, freq = 100, savename = NA, maxiterrsg = 5000, verbose = FALSE, cities = NULL )
adj |
An adjacency matrix, list, or object of class "SpatialPolygonsDataFrame." |
total_pop |
A vector containing the populations of each geographic unit. |
nsims |
The number of simulations run before a save point. |
ndists |
The number of congressional districts. The default is
|
init_plan |
A vector containing the congressional district labels
of each geographic unit. The default is |
loopscompleted |
Number of save points reached by the
algorithm. The default is |
nloop |
The total number of save points for the algorithm. The
default is |
nthin |
The amount by which to thin the Markov Chain. The default
is |
eprob |
The probability of keeping an edge connected. The default
is |
lambda |
The parameter determining the number of swaps to attempt
each iteration of the algorithm. The number of swaps each iteration is
equal to Pois( |
pop_tol |
The strength of the hard population
constraint. |
group_pop |
A vector of populations for some sub-group of
interest. The default is |
areasvec |
A vector of precinct areas for discrete Polsby-Popper.
The default is |
counties |
A vector of county membership assignments. The default is |
borderlength_mat |
A matrix of border length distances, where
the first two columns are the indices of precincts sharing a border and
the third column is its distance. Default is |
ssdmat |
A matrix of squared distances between geographic
units. The default is |
compactness_metric |
The compactness metric to use when constraining on
compactness. Default is |
rngseed |
Allows the user to set the seed for the
simulations. Default is |
constraint |
Which constraint to apply. Accepts any combination of |
constraintweights |
The weights to apply to each constraint. Should be a vector the same length as constraint. Default is NULL. |
betaseq |
Sequence of beta values for tempering. The default is
|
betaseqlength |
Length of beta sequence desired for
tempering. The default is |
adjswaps |
Flag to restrict swaps of beta so that only
values adjacent to current constraint are proposed. The default is
|
freq |
Frequency of between-chain swaps. Default to once every 100 iterations |
savename |
Filename to save simulations. Default is |
maxiterrsg |
Maximum number of iterations for random seed-and-grow algorithm to generate starting values. Default is 5000. |
verbose |
Whether to print initialization statement. Default is
|
cities |
integer vector of cities for QPS constraint. |
This function allows users to simulate redistricting plans using Markov Chain Monte Carlo methods. Several constraints corresponding to substantive requirements in the redistricting process are implemented, including population parity and geographic compactness. In addition, the function includes multiple-swap and parallel tempering functionality in MPI to improve the mixing of the Markov Chain.
redist.mcmc.mpi
returns an object of class "redist". The object
redist
is a list that contains the following components (the
inclusion of some components is dependent on whether tempering
techniques are used):
partitions |
Matrix of congressional district assignments generated by the algorithm. Each row corresponds to a geographic unit, and each column corresponds to a simulation. |
distance_parity |
Vector containing the maximum distance from parity for a particular simulated redistricting plan. |
mhdecisions |
A vector specifying whether a proposed redistricting plan was accepted (1) or rejected (0) in a given iteration. |
mhprob |
A vector containing the Metropolis-Hastings acceptance probability for each iteration of the algorithm. |
pparam |
A vector containing the draw of the |
constraint_pop |
A vector containing the value of the population constraint for each accepted redistricting plan. |
constraint_compact |
A vector containing the value of the compactness constraint for each accepted redistricting plan. |
constraint_vra |
A vector containing the value of the vra constraint for each accepted redistricting plan. |
constraint_similar |
A vector containing the value of the similarity constraint for each accepted redistricting plan. |
beta_sequence |
A vector containing the value of beta for each iteration of the algorithm. Returned when tempering is being used. |
mhdecisions_beta |
A vector specifying whether a proposed beta value was accepted (1) or rejected (0) in a given iteration of the algorithm. Returned when tempering is being used. |
mhprob_beta |
A vector containing the Metropolis-Hastings acceptance probability for each iteration of the algorithm. Returned when tempering is being used. |
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
## Not run: # Cannot run on machines without Rmpi data(fl25) data(fl25_enum) data(fl25_adj) ## Code to run the simulations in Figure 4 in Fifield, Higgins, Imai and ## Tarr (2015) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] ## Run the algorithm redist.mcmc.mpi(adj = fl25_adj, total_pop = fl25$pop, init_plan = init_plan, nsims = 10000, savename = "test") ## End(Not run)
## Not run: # Cannot run on machines without Rmpi data(fl25) data(fl25_enum) data(fl25_adj) ## Code to run the simulations in Figure 4 in Fifield, Higgins, Imai and ## Tarr (2015) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] ## Run the algorithm redist.mcmc.mpi(adj = fl25_adj, total_pop = fl25$pop, init_plan = init_plan, nsims = 10000, savename = "test") ## End(Not run)
Counts the total number of counties that are split across more than 2 districts.
redist.multisplits(plans, counties)
redist.multisplits(plans, counties)
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. Required. |
counties |
A vector of county names or county ids. |
integer matrix where each district is a
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.01) plans <- redist_smc(ia, 50, silent = TRUE) #old redist.multisplits(plans, ia$region) splits_multi(plans, ia, region)
data(iowa) ia <- redist_map(iowa, existing_plan = cd_2010, total_pop = pop, pop_tol = 0.01) plans <- redist_smc(ia, 50, silent = TRUE) #old redist.multisplits(plans, ia$region) splits_multi(plans, ia, region)
Computes the deviation from population parity from a plan. Higher values indicate that (at least) a single district in the map deviates from population parity. See Details.
redist.parity(plans, total_pop) plan_parity(map, .data = pl(), ...)
redist.parity(plans, total_pop) plan_parity(map, .data = pl(), ...)
plans |
A matrix with one row for each precinct and one column for each map. Required. |
total_pop |
A numeric vector with the population for every precinct. |
map |
a |
.data |
a |
... |
passed on to |
With a map with pop
representing the populations of each district,
the deviation from population parity is given as max(abs(pop - parity) / parity)
where parity = sum(pop)/length(pop)
is the population size for the
average district.
Therefore, the metric can be thought of as the maximum percent deviation from
equal population. For example, a value of 0.03 in this metric indicates that
all districts are within 3 percent of population parity.
numeric vector with the population parity for each column
Creates a Graph Overlay
redist.plot.adj( shp, adj = NULL, plan = NULL, centroids = TRUE, drop = FALSE, plot_shp = TRUE, zoom_to = NULL, title = "" )
redist.plot.adj( shp, adj = NULL, plan = NULL, centroids = TRUE, drop = FALSE, plot_shp = TRUE, zoom_to = NULL, title = "" )
shp |
A SpatialPolygonsDataFrame or sf object. Required. |
adj |
A zero-indexed adjacency list. Created with redist.adjacency if not supplied. Default is NULL. |
plan |
A numeric vector with one entry for each precinct in shp.
Used to remove edges that cross boundaries. Default is |
centroids |
A logical indicating if centroids should be plotted. Default is |
drop |
A logical indicating if edges that cross districts should be dropped. Default is |
plot_shp |
A logical indicating if the shp should be plotted under the
graph. Default is |
zoom_to |
|
title |
A string title of plot. Defaults to empty string. Optional. |
ggplot map
data(iowa) redist.plot.adj(shp = iowa, plan = iowa$cd_2010)
data(iowa) redist.plot.adj(shp = iowa, plan = iowa$cd_2010)
Plot Cores
redist.plot.cores(shp, plan = NULL, core = NULL, lwd = 2)
redist.plot.cores(shp, plan = NULL, core = NULL, lwd = 2)
shp |
A SpatialPolygonsDataFrame or sf object. Required. |
plan |
A numeric vector with one entry for each precinct in shp. Used to color the districts. Required. |
core |
Required. integer vector produced by |
lwd |
Line width. Defaults to 2. |
ggplot
Plots a boxplot of a quantity of interest across districts, with districts optionally sorted by this quantity. Adds reference points for each reference plan, if applicable.
redist.plot.distr_qtys( plans, qty, sort = "asc", geom = "jitter", color_thresh = NULL, size = 0.1, ref_geom, ref_label, ... )
redist.plot.distr_qtys( plans, qty, sort = "asc", geom = "jitter", color_thresh = NULL, size = 0.1, ref_geom, ref_label, ... )
plans |
the |
qty |
|
sort |
set to |
geom |
the |
color_thresh |
if a number, the threshold to use in coloring the points. Plans with quantities of interest above the threshold will be colored differently than plans below the threshold. |
size |
The dot size for |
ref_geom |
The reference plan geometry type. |
ref_label |
A human-readable name for the reference plan. By default
the name in the |
... |
passed on to |
A ggplot
ggdist
For custom functions in geom
, we can also create more complicated things like rainclouds
using the ggdist
package. For example:
raincloud <- function(...) { list( ggdist::stat_slab(aes(thickness = ggplot2::after_stat(pdf*n)), scale = 0.7), ggdist::stat_dotsinterval(side = "bottom", scale = 0.7, slab_size = NA, quantiles = 200) ) }
These functions can be then passed to geom
.
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) plans <- plans %>% mutate(pct_dem = group_frac(iowa, dem_08, tot_08)) redist.plot.distr_qtys(plans, pct_dem) # It also takes custom functions: redist.plot.distr_qtys(plans, pct_dem, geom = ggplot2::geom_violin) # With the raincloud example, if you have `ggdist`, you can run: # redist.plot.distr_qtys(plans, pct_dem, geom = raincloud) # The reference geom can also be changed via `reg_geom` r_geom <- function(...) ggplot2::geom_segment(ggplot2::aes(as.integer(.data$.distr_no) - 0.5, xend = as.integer(.data$.distr_no) + 0.5, yend = pct_dem, color = .data$draw), linewidth = 1.2, ...) # Finally, the `ref_label` argument can also be swapped for a function, like so: redist.plot.distr_qtys(plans, pct_dem, geom = ggplot2::geom_violin, ref_geom = r_geom, ref_label = function() ggplot2::labs(color = 'Ref.'))
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) plans <- plans %>% mutate(pct_dem = group_frac(iowa, dem_08, tot_08)) redist.plot.distr_qtys(plans, pct_dem) # It also takes custom functions: redist.plot.distr_qtys(plans, pct_dem, geom = ggplot2::geom_violin) # With the raincloud example, if you have `ggdist`, you can run: # redist.plot.distr_qtys(plans, pct_dem, geom = raincloud) # The reference geom can also be changed via `reg_geom` r_geom <- function(...) ggplot2::geom_segment(ggplot2::aes(as.integer(.data$.distr_no) - 0.5, xend = as.integer(.data$.distr_no) + 0.5, yend = pct_dem, color = .data$draw), linewidth = 1.2, ...) # Finally, the `ref_label` argument can also be swapped for a function, like so: redist.plot.distr_qtys(plans, pct_dem, geom = ggplot2::geom_violin, ref_geom = r_geom, ref_label = function() ggplot2::labs(color = 'Ref.'))
Plots a histogram of a statistic of a redist_plans
object,
with a reference line for each reference plan, if applicable.
redist.plot.hist(plans, qty, bins = NULL, ...) ## S3 method for class 'redist_plans' hist(x, qty, ...)
redist.plot.hist(plans, qty, bins = NULL, ...) ## S3 method for class 'redist_plans' hist(x, qty, ...)
plans |
the |
qty |
|
bins |
the number of bins to use in the histogram. Defaults to Freedman-Diaconis rule. |
... |
passed on to |
x |
|
A ggplot
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) group_by(plans, draw) %>% summarize(pop_dev = max(abs(total_pop/mean(total_pop) - 1))) %>% redist.plot.hist(pop_dev)
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) group_by(plans, draw) %>% summarize(pop_dev = max(abs(total_pop/mean(total_pop) - 1))) %>% redist.plot.hist(pop_dev)
Majority Minority Plots
redist.plot.majmin(grouppercent, type = "hist", title = "")
redist.plot.majmin(grouppercent, type = "hist", title = "")
grouppercent |
output from redist.group.percent |
type |
string in 'hist', 'toptwo', or 'box' |
title |
ggplot title |
ggplot
Create a ggplot map. It fills by plan or argument fill. If both are supplied, plan is used as the color and fill as the alpha parameter.
redist.plot.map( shp, adj, plan = NULL, fill = NULL, fill_label = "", zoom_to = NULL, boundaries = is.null(fill), title = "" )
redist.plot.map( shp, adj, plan = NULL, fill = NULL, fill_label = "", zoom_to = NULL, boundaries = is.null(fill), title = "" )
shp |
A SpatialPolygonsDataFrame, sf object, or redist_map. Required. |
adj |
A zero-indexed adjacency list. Created with redist.adjacency if not supplied and needed for coloring. Default is NULL. |
plan |
|
fill |
|
fill_label |
A string title of plot. Defaults to the empty string |
zoom_to |
|
boundaries |
A logical indicating if precinct boundaries should be plotted. |
title |
A string title of plot. Defaults to empty string. Optional. |
ggplot map
data(iowa) redist.plot.map(shp = iowa, plan = iowa$cd_2010) iowa_map <- redist_map(iowa, existing_plan = cd_2010) redist.plot.map(iowa_map, fill = dem_08/tot_08, zoom_to = (cd_2010 == 1))
data(iowa) redist.plot.map(shp = iowa, plan = iowa$cd_2010) iowa_map <- redist_map(iowa, existing_plan = cd_2010) redist.plot.map(iowa_map, fill = dem_08/tot_08, zoom_to = (cd_2010 == 1))
Plots the shape of the add_constr_grp_pow()
penalty.
redist.plot.penalty( tgt_min = 0.55, tgt_other = 0.25, strength_vra = 2500, pow_vra = 1.5, limits = TRUE )
redist.plot.penalty( tgt_min = 0.55, tgt_other = 0.25, strength_vra = 2500, pow_vra = 1.5, limits = TRUE )
tgt_min |
double, defaults to 0.55. The minority target percent. |
tgt_other |
double, defaults to 0.25. The other group target percent. |
strength_vra |
double, strength of the VRA constraint. |
pow_vra |
double, exponent of the VRA constraint. |
limits |
Whether to limit y axis to 0,500. Default is TRUE for comparability across values. |
This function allows you to plot the un-exponentiated penalty implemented as
add_constr_grp_pow()
. The function takes two key inputs,
tgt_min
and tgt_other
which center the minimum penalty spots. A higher y-value
indicates a higher penalty and incentivizes moving towards a spot with a lower y-value.
The x-axis indicates the group population proportion in a given district.
ggplot
Plot a district assignment
redist.plot.plans( plans, draws, shp, qty = NULL, interactive = FALSE, ..., geom = NULL )
redist.plot.plans( plans, draws, shp, qty = NULL, interactive = FALSE, ..., geom = NULL )
plans |
a |
draws |
the plan(s) to plot. Will match the |
qty |
the quantity to plot. Defaults to the district assignment. |
interactive |
if |
... |
additional arguments passed to the plotting functions. |
geom , shp
|
the |
A ggplot
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) redist.plot.plans(plans, c(1, 2, 3, 4), iowa)
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) redist.plot.plans(plans, c(1, 2, 3, 4), iowa)
Makes a scatterplot of two quantities of interest across districts or plans.
redist.plot.scatter(plans, x, y, ..., bigger = TRUE)
redist.plot.scatter(plans, x, y, ..., bigger = TRUE)
plans |
the |
x |
|
y |
|
... |
passed on to |
bigger |
if TRUE, make the point corresponding to the reference plan larger. |
A ggplot
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) plans %>% mutate(comp = distr_compactness(iowa)) %>% group_by(draw) %>% summarize(pop_dev = max(abs(total_pop/mean(total_pop) - 1)), comp = comp[1]) %>% redist.plot.scatter(pop_dev, comp)
library(dplyr) data(iowa) iowa <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) plans <- redist_smc(iowa, nsims = 100, silent = TRUE) plans %>% mutate(comp = distr_compactness(iowa)) %>% group_by(draw) %>% summarize(pop_dev = max(abs(total_pop/mean(total_pop) - 1)), comp = comp[1]) %>% redist.plot.scatter(pop_dev, comp)
For a statistic in a redist_plans
object,
make a traceplot showing the evolution of the statistic over MCMC iterations.
redist.plot.trace(plans, qty, district = 1L, ...)
redist.plot.trace(plans, qty, district = 1L, ...)
plans |
the |
qty |
|
district |
for |
... |
passed on to |
A ggplot
library(dplyr) data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_mergesplit_parallel(iowa_map, nsims = 200, chains = 2, silent = TRUE) %>% mutate(dem = group_frac(iowa_map, dem_08, dem_08 + rep_08)) %>% number_by(dem) redist.plot.trace(plans, dem, district = 1)
library(dplyr) data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05) plans <- redist_mergesplit_parallel(iowa_map, nsims = 200, chains = 2, silent = TRUE) %>% mutate(dem = group_frac(iowa_map, dem_08, dem_08 + rep_08)) %>% number_by(dem) redist.plot.trace(plans, dem, district = 1)
Static Variation of Information Plot
redist.plot.varinfo(plans, group_pop, total_pop, shp)
redist.plot.varinfo(plans, group_pop, total_pop, shp)
plans |
matrix of district assignments |
group_pop |
Required Population of subgroup being studied in each precinct. |
total_pop |
Required. Population of each precinct. |
shp |
sf dataframe |
patchworked ggplot
Plots the weighted adjacency graph by how often precincts coocur. If an argument to counties is provided, it subsets the edges to plot to those that cross over the county boundary.
redist.plot.wted.adj( shp, plans, counties = NULL, ref = TRUE, adj = NULL, plot_shp = TRUE )
redist.plot.wted.adj( shp, plans, counties = NULL, ref = TRUE, adj = NULL, plot_shp = TRUE )
shp |
A SpatialPolygonsDataFrame, sf object, or redist_map. Required. |
plans |
A |
counties |
unquoted name of a column in |
ref |
Plot reference map? Defaults to TRUE which gets the existing plan from |
adj |
A zero-indexed adjacency list. Extracted from |
plot_shp |
Should the shapes be plotted? Default is TRUE. |
ggplot
data(iowa) shp <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) plans <- redist_smc(shp, 100) redist.plot.wted.adj(shp, plans = plans, counties = region)
data(iowa) shp <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) plans <- redist_smc(shp, 100) redist.plot.wted.adj(shp, plans = plans, counties = region)
Compare the Population Overlap Across Plans at the Precinct Level
redist.prec.pop.overlap( plan_old, plan_new, total_pop, weighting = "s", normalize = TRUE, index_only = FALSE, return_mat = FALSE )
redist.prec.pop.overlap( plan_old, plan_new, total_pop, weighting = "s", normalize = TRUE, index_only = FALSE, return_mat = FALSE )
plan_old |
The reference plan to compare against |
plan_new |
The new plan to compare to the reference plan |
total_pop |
The total population by precinct This can also take a redist_map object and will use the population in that object. If nothing is provided, it weights all entries in plan equally. |
weighting |
Should weighting be done by sum of populations |
normalize |
Should entries be normalized by the total population |
index_only |
Default is FALSE. TRUE returns only one numeric index, the mean of the upper triangle of the matrix, under the weighting and normalization chosen. |
return_mat |
Defaults to FALSE, where it returns the summary by row. If TRUE returns matrix with length(plan_old) rows and columns. Ignored if index_only = TRUE. |
numeric vector with length(plan_old) entries
set.seed(5) data(iowa) iowa_map <- redist_map(iowa, total_pop = pop, pop_tol = 0.01, ndists = 4) plans <- redist_smc(iowa_map, 2, silent = TRUE) plans_mat <- get_plans_matrix(plans) ov_vec <- redist.prec.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map) redist.prec.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map, weighting = "s", normalize = FALSE, index_only = TRUE)
set.seed(5) data(iowa) iowa_map <- redist_map(iowa, total_pop = pop, pop_tol = 0.01, ndists = 4) plans <- redist_smc(iowa_map, 2, silent = TRUE) plans_mat <- get_plans_matrix(plans) ov_vec <- redist.prec.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map) redist.prec.pop.overlap(plans_mat[, 1], plans_mat[, 2], iowa_map, weighting = "s", normalize = FALSE, index_only = TRUE)
Prepares a run of the enumpart algorithm by ordering edges
redist.prep.enumpart( adj, unordered_path, ordered_path, weight_path = NULL, total_pop = NULL )
redist.prep.enumpart( adj, unordered_path, ordered_path, weight_path = NULL, total_pop = NULL )
adj |
zero indexed adjacency list |
unordered_path |
valid path to output the unordered adjacency map to |
ordered_path |
valid path to output the ordered adjacency map to |
weight_path |
A path (not including ".dat") to store a space-delimited file containing a vector of vertex weights. Only supply with total_pop. |
total_pop |
the vector of precinct populations. Only supply with weight_path |
0 on success
Benjamin Fifield, Kosuke Imai, Jun Kawahara, and Christopher T Kenny. "The Essential Role of Empirical Validation in Legislative Redistricting Simulation." Forthcoming, Statistics and Public Policy.
## Not run: temp <- tempdir() data(fl25) adj <- redist.adjacency(fl25) redist.prep.enumpart(adj = adj, unordered_path = paste0(temp, "/unordered"), ordered_path = paste0(temp, "/ordered")) ## End(Not run)
## Not run: temp <- tempdir() data(fl25) adj <- redist.adjacency(fl25) redist.prep.enumpart(adj = adj, unordered_path = paste0(temp, "/unordered"), ordered_path = paste0(temp, "/ordered")) ## End(Not run)
random.subgraph
returns a random subset of the shp provided
redist.random.subgraph(shp, n, adj = NULL)
redist.random.subgraph(shp, n, adj = NULL)
shp |
sf object or SpatialPolygonsDataFrame |
n |
number of edges to sample. n must be a positive integer. |
adj |
Optional. zero indexed adjacency list. |
Snowball sampling with backtracking
sf dataframe with n rows
Read Results from enumpart
redist.read.enumpart(out_path, skip = 0, n_max = -1L)
redist.read.enumpart(out_path, skip = 0, n_max = -1L)
out_path |
out_path specified in redist.run.enumpart |
skip |
number of lines to skip |
n_max |
max number of lines to read |
district_membership matrix
Benjamin Fifield, Kosuke Imai, Jun Kawahara, and Christopher T Kenny. "The Essential Role of Empirical Validation in Legislative Redistricting Simulation." Forthcoming, Statistics and Public Policy.
## Not run: temp <- tempdir() cds <- redist.read.enumpart(out_path = paste0(temp, "/enumerated")) ## End(Not run)
## Not run: temp <- tempdir() cds <- redist.read.enumpart(out_path = paste0(temp, "/enumerated")) ## End(Not run)
Tool to help reduce adjacency lists for analyzing subsets of maps.
redist.reduce.adjacency(adj, keep_rows)
redist.reduce.adjacency(adj, keep_rows)
adj |
A zero-indexed adjacency list. Required. |
keep_rows |
row numbers of precincts to keep |
zero indexed adjacency list with max value length(keep_rows) - 1
data(fl25_adj) redist.reduce.adjacency(fl25_adj, c(2, 3, 4, 6, 21))
data(fl25_adj) redist.reduce.adjacency(fl25_adj, c(2, 3, 4, 6, 21))
Ensures that for each column in the plans object, the first district listed is 1, the second is 2, up to n districts. Assumes that all columns have the same number of districts as the first.
redist.reorder(plans)
redist.reorder(plans)
plans |
A numeric vector (if only one map) or matrix with one row for each precinct and one column for each map. |
integer matrix
cds <- matrix(c(rep(c(4L, 5L, 2L, 1L, 3L), 5), rep(c(5L, 4L, 3L, 2L, 1L), 2), rep(c(4L, 5L, 2L, 1L, 3L), 3)), nrow = 25) redist.reorder(cds)
cds <- matrix(c(rep(c(4L, 5L, 2L, 1L, 3L), 5), rep(c(5L, 4L, 3L, 2L, 1L), 2), rep(c(4L, 5L, 2L, 1L, 3L), 3)), nrow = 25) redist.reorder(cds)
redist.rsg
generates redistricting plans using a random seed a grow
algorithm. This is the non-compact districting algorithm described in Chen and
Rodden (2013). The algorithm can provide start values for the other
redistricting routines in this package.
redist.rsg(adj, total_pop, ndists, pop_tol, verbose = TRUE, maxiter = 5000)
redist.rsg(adj, total_pop, ndists, pop_tol, verbose = TRUE, maxiter = 5000)
adj |
List of length N, where N is the number of precincts. Each list element is an integer vector indicating which precincts that precinct is adjacent to. It is assumed that precinct numbers start at 0. |
total_pop |
numeric vector of length N, where N is the number of precincts. Each element lists the population total of the corresponding precinct, and is used to enforce population constraints. |
ndists |
integer, the number of districts we want to partition the precincts into. |
pop_tol |
numeric, indicating how close district population targets have to be to the target population before algorithm converges. thresh=0.05 for example means that all districts must be between 0.95 and 1.05 times the size of target.pop in population size. |
verbose |
boolean, indicating whether the time to run the algorithm is printed. |
maxiter |
integer, indicating maximum number of iterations to attempt before convergence to population constraint fails. If it fails once, it will use a different set of start values and try again. If it fails again, redist.rsg() returns an object of all NAs, indicating that use of more iterations may be advised. |
list, containing three objects containing the completed redistricting plan.
plan
: A vector of length N, indicating the
district membership of each precinct.
district_list
A list of length Ndistrict. Each list contains a
vector of the precincts in the respective district.
district_pop
A vector of length Ndistrict, containing the
population totals of the respective districts.
Benjamin Fifield, Department of Politics, Princeton University [email protected], https://www.benfifield.com/
Michael Higgins, Department of Statistics, Kansas State University [email protected], https://www.k-state.edu/stats/about/people/HigginsMichael.html
Kosuke Imai, Department of Politics, Princeton University [email protected], https://imai.fas.harvard.edu
James Lo, [email protected]
Alexander Tarr, Department of Electrical Engineering, Princeton University [email protected]
Jowei Chen and Jonathan Rodden (2013) “Unintentional Gerrymandering: Political Geography and Electoral Bias in Legislatures.” Quarterly Journal of Political Science. 8(3): 239-269.
### Real data example from test set data(fl25) data(fl25_adj) res <- redist.rsg(adj = fl25_adj, total_pop = fl25$pop, ndists = 3, pop_tol = 0.05)
### Real data example from test set data(fl25) data(fl25_adj) res <- redist.rsg(adj = fl25_adj, total_pop = fl25$pop, ndists = 3, pop_tol = 0.05)
Runs the enumpart algorithm
redist.run.enumpart( ordered_path, out_path, ndists = 2, all = TRUE, n = NULL, weight_path = NULL, lower = NULL, upper = NULL, options = NULL )
redist.run.enumpart( ordered_path, out_path, ndists = 2, all = TRUE, n = NULL, weight_path = NULL, lower = NULL, upper = NULL, options = NULL )
ordered_path |
Path used in redist.prep.enumpart (not including ".dat") |
out_path |
Valid path to output the enumerated districts |
ndists |
number of districts to enumerate |
all |
boolean. TRUE outputs all districts. FALSE samples n districts. |
n |
integer. Number of districts to output if all is FALSE. Returns districts selected from uniform random distribution. |
weight_path |
A path (not including ".dat") to a space-delimited file containing a vector of
vertex weights, to be used along with |
lower |
A lower bound on each partition's total weight, implemented by rejection sampling. |
upper |
An upper bound on each partition's total weight. |
options |
Additional enumpart arguments. Not recommended for use. |
0 on success
Benjamin Fifield, Kosuke Imai, Jun Kawahara, and Christopher T Kenny. "The Essential Role of Empirical Validation in Legislative Redistricting Simulation." Forthcoming, Statistics and Public Policy.
## Not run: temp <- tempdir() redist.run.enumpart(ordered_path = paste0(temp, "/ordered"), out_path = paste0(temp, "/enumerated")) ## End(Not run)
## Not run: temp <- tempdir() redist.run.enumpart(ordered_path = paste0(temp, "/ordered"), out_path = paste0(temp, "/enumerated")) ## End(Not run)
Takes a plan and renumbers it to be from 1:ndists
redist.sink.plan(plan)
redist.sink.plan(plan)
plan |
vector of assignments, required. |
A vector with an ID that corresponds from 1:ndists, and attribute n
indicating the number of districts.
data(fl25_enum) plan <- fl25_enum$plans[, 5118] # Subset based on something: plan <- plan[plan != 2] plan <- vctrs::vec_group_id(plan) # Now plan can be used with redist_flip() plan
data(fl25_enum) plan <- fl25_enum$plans[, 5118] # Subset based on something: plan <- plan[plan != 2] plan <- vctrs::vec_group_id(plan) # Now plan can be used with redist_flip() plan
Builds a confidence interval for a quantity of interest, given importance sampling weights.
redist.smc_is_ci(x, wgt, conf = 0.99)
redist.smc_is_ci(x, wgt, conf = 0.99)
x |
A numeric vector containing the quantity of interest |
wgt |
A numeric vector containing the nonnegative importance weights. Will be normalized automatically. |
conf |
The confidence level for the interval. |
A two-element vector of the form [lower, upper]
containing
the importance sampling confidence interval.
Subsets a shp object along with its adjacency. Useful for running smaller analyses on pairs of districts. Provide population, ndists, pop_tol, and sub_ndists to get proper population parity constraints on subsets.
redist.subset(shp, adj, keep_rows, total_pop, ndists, pop_tol, sub_ndists)
redist.subset(shp, adj, keep_rows, total_pop, ndists, pop_tol, sub_ndists)
shp |
An sf object |
adj |
A zero-indexed adjacency list. Created with
|
keep_rows |
row numbers of precincts to keep. Random submap selected if not supplied. |
total_pop |
numeric vector with one entry for the population of each precinct. |
ndists |
integer, number of districts in whole map |
pop_tol |
The strength of the hard population constraint. |
sub_ndists |
integer, number of districts in subset map |
a list containing the following components:
shp |
The subsetted shp object |
adj |
The subsetted adjacency list for shp |
keep_rows |
The indices of the rows kept. |
sub_ndists |
The number of districts in the subset. |
sub_pop_tol |
The new parity constraint for a subset. |
After a cores analysis or other form of coarsening, sometimes you need to be at the original geography level to be comparable. This takes in a coarsened matrix and uncoarsens it to the original level
redist.uncoarsen(plans, group_index)
redist.uncoarsen(plans, group_index)
plans |
A coarsened matrix of plans. |
group_index |
The index used to coarsen the shape. |
matrix
Create Weighted Adjacency Data
redist.wted.adj(map = NULL, plans = NULL)
redist.wted.adj(map = NULL, plans = NULL)
map |
redist_map |
plans |
redist_plans |
tibble
data(iowa) shp <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) plans <- redist_smc(shp, 100) redist.wted.adj(shp, plans = plans)
data(iowa) shp <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.01) plans <- redist_smc(shp, 100) redist.wted.adj(shp, plans = plans)
redist_shortburst
The output of these functions may be passed into redist_shortburst()
as
score_fn
. Scoring functions have type redist_scorer
and may be combined
together using basic arithmetic operations.
scorer_group_pct(map, group_pop, total_pop, k = 1) scorer_pop_dev(map) scorer_splits(map, counties) scorer_multisplits(map, counties) scorer_frac_kept(map) scorer_polsby_popper(map, perim_df = NULL, areas = NULL, m = 1) scorer_status_quo(map, existing_plan = get_existing(map))
scorer_group_pct(map, group_pop, total_pop, k = 1) scorer_pop_dev(map) scorer_splits(map, counties) scorer_multisplits(map, counties) scorer_frac_kept(map) scorer_polsby_popper(map, perim_df = NULL, areas = NULL, m = 1) scorer_status_quo(map, existing_plan = get_existing(map))
map |
A |
group_pop |
A numeric vector with the population of the group for every precinct. |
total_pop |
A numeric vector with the population for every precinct. |
k |
the k-th from the top group fraction to return as the score. |
counties |
A numeric vector with an integer from 1:n_counties |
perim_df |
perimeter distance dataframe from |
areas |
area of each precinct (ie |
m |
the m-th from the bottom Polsby Popper to return as the score. Defaults to 1, the minimum Polsby Popper score |
existing_plan |
A vector containing the current plan. |
Function details:
scorer_group_pct
returns the k
-th top group percentage across districts.
For example, if the group is Democratic voters and k=3
, then the function
returns the 3rd-highest fraction of Democratic voters across all districts.
Can be used to target k
VRA districts or partisan gerrymanders.
scorer_pop_dev
returns the maximum population deviation within a plan.
Smaller values are closer to population parity, so use maximize=FALSE
with
this scorer.
scorer_splits
returns the fraction of counties that are split within a
plan. Higher values have more county splits, so use maximize=FALSE
with
this scorer.
scorer_frac_kept
returns the fraction of edges kept in each district.
Higher values mean more compactness.
scorer_polsby_popper
returns the m
-th Polsby Popper score within a plan.
Higher scores correspond to more compact districts. Use m=ndists/2
to
target the median compactness, m=1
to target the minimum compactness.
scorer_status_quo
returns 1 - the rescaled variation of information
distance between the plan and the existing_plan
. Larger values indicate the
plan is closer to the existing plan.
A scoring function of class redist_scorer
which returns a single numeric value per plan.
Larger values are generally better for frac_kept
, group_pct
, and polsby_popper
and smaller values are better for splits
and pop_dev
.
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) scorer_frac_kept(iowa_map) scorer_status_quo(iowa_map) scorer_group_pct(iowa_map, dem_08, tot_08, k = 2) 1.5*scorer_frac_kept(iowa_map) + 0.4*scorer_status_quo(iowa_map) 1.5*scorer_frac_kept(iowa_map) + scorer_frac_kept(iowa_map)*scorer_status_quo(iowa_map) cbind( comp = scorer_frac_kept(iowa_map), sq = scorer_status_quo(iowa_map) )
data(iowa) iowa_map <- redist_map(iowa, existing_plan = cd_2010, pop_tol = 0.05, total_pop = pop) scorer_frac_kept(iowa_map) scorer_status_quo(iowa_map) scorer_group_pct(iowa_map, dem_08, tot_08, k = 2) 1.5*scorer_frac_kept(iowa_map) + 0.4*scorer_status_quo(iowa_map) 1.5*scorer_frac_kept(iowa_map) + scorer_frac_kept(iowa_map)*scorer_status_quo(iowa_map) cbind( comp = scorer_frac_kept(iowa_map), sq = scorer_status_quo(iowa_map) )
redist_scorer
functions may be multiplied by constants and/or added
together to form linear combinations.
## S3 method for class 'redist_scorer' x * fn2 ## S3 method for class 'redist_scorer' fn1 + fn2 ## S3 method for class 'redist_scorer' fn1 - fn2
## S3 method for class 'redist_scorer' x * fn2 ## S3 method for class 'redist_scorer' fn1 + fn2 ## S3 method for class 'redist_scorer' fn1 - fn2
x |
a numeric or a |
fn2 |
a |
fn1 |
a |
function of class redist_scorer
redist_scorer
functions may be combined together to optimize along multiple
dimensions. Rather than linearly combining multiple scorers to form a single
objective as with scorer-arith, these functions allow analysts to approximate
the Pareto frontier for a set of scorers.
combine_scorers(...) ## S3 method for class 'redist_scorer' cbind(..., deparse.level = 1)
combine_scorers(...) ## S3 method for class 'redist_scorer' cbind(..., deparse.level = 1)
... |
a numeric or a |
deparse.level |
As in |
function of class redist_scorer. Will return a matrix with each column containing every plan's scores for a particular scoring function.
redist.segcalc
calculates the dissimilarity index of segregation (see
Massey & Denton 1987 for more details) for a specified subgroup under any
redistricting plan.
segregation_index( map, group_pop, total_pop = map[[attr(map, "pop_col")]], .data = cur_plans() ) redist.segcalc(plans, group_pop, total_pop)
segregation_index( map, group_pop, total_pop = map[[attr(map, "pop_col")]], .data = cur_plans() ) redist.segcalc(plans, group_pop, total_pop)
map |
a |
group_pop |
A vector of populations for some subgroup of interest. |
total_pop |
A vector containing the populations of each geographic unit. |
.data |
a |
plans |
A matrix of congressional district assignments or a redist object. |
redist.segcalc
returns a vector where each entry is the
dissimilarity index of segregation (Massey & Denton 1987) for each
redistricting plan in algout
.
Fifield, Benjamin, Michael Higgins, Kosuke Imai and Alexander Tarr. (2016) "A New Automated Redistricting Simulator Using Markov Chain Monte Carlo." Working Paper. Available at http://imai.princeton.edu/research/files/redist.pdf.
Massey, Douglas and Nancy Denton. (1987) "The Dimensions of Social Segregation". Social Forces.
data(fl25) data(fl25_enum) data(fl25_adj) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] fl25$init_plan <- init_plan ## 25 precinct, three districts - no pop constraint ## fl_map <- redist_map(fl25, existing_plan = 'init_plan', adj = fl25_adj) alg_253 <- redist_flip(fl_map, nsims = 10000) ## Get Republican Dissimilarity Index from simulations # old: rep_dmi_253 <- redist.segcalc(alg_253, fl25$mccain, fl25$pop) rep_dmi_253 <- seg_dissim(alg_253, fl25, mccain, pop) |> redistmetrics::by_plan(ndists = 3)
data(fl25) data(fl25_enum) data(fl25_adj) ## Get an initial partition init_plan <- fl25_enum$plans[, 5118] fl25$init_plan <- init_plan ## 25 precinct, three districts - no pop constraint ## fl_map <- redist_map(fl25, existing_plan = 'init_plan', adj = fl25_adj) alg_253 <- redist_flip(fl_map, nsims = 10000) ## Get Republican Dissimilarity Index from simulations # old: rep_dmi_253 <- redist.segcalc(alg_253, fl25$mccain, fl25$pop) rep_dmi_253 <- seg_dissim(alg_253, fl25, mccain, pop) |> redistmetrics::by_plan(ndists = 3)
Subset to sampled or reference draws
subset_sampled(plans, matrix = TRUE) subset_ref(plans, matrix = TRUE)
subset_sampled(plans, matrix = TRUE) subset_ref(plans, matrix = TRUE)
plans |
the |
matrix |
if |
a redist_plans
object, with only rows corresponding to
simulated (or reference) draws remaining.
Prints diagnostic information, which varies by algorithm. All algorithms
compute the plans_diversity()
of the samples.
## S3 method for class 'redist_plans' summary(object, district = 1L, all_runs = TRUE, vi_max = 100, ...)
## S3 method for class 'redist_plans' summary(object, district = 1L, all_runs = TRUE, vi_max = 100, ...)
object |
a redist_plans object |
district |
For R-hat values, which district to use for district-level
summary statistics. We strongly recommend calling |
all_runs |
When there are multiple SMC runs, show detailed summary statistics for all runs (the default), or only the first run? |
vi_max |
The maximum number of plans to sample in computing the pairwise variation of information distance (sample diversity). |
... |
additional arguments (ignored) |
For SMC and MCMC, if there are multiple runs/chains, R-hat values will be computed for each summary statistic. These values should be close to 1. If they are not, then there is too much between-chain variation, indicating that there are not enough samples. R-hat values are calculated after rank-normalization and folding. MCMC chains are split in half before R-hat is computed. For summary statistics that vary across districts, R-hat is calculated for the first district only.
For SMC, diagnostics statistics include:
Effective samples: the effective sample size at each iteration, computed using the SMC weights. Larger is better. The percentage in parentheses is the ratio of the effective samples to the total samples.
Acceptance rate: the fraction of drawn spanning trees which yield a valid redistricting plan within the population tolerance. Very small values (< 1%) can indicate a bottleneck and may lead to a lack of diversity.
Standard deviation of the log weights: More variable weights (larger s.d.) indicate less efficient sampling. Values greater than 3 are likely problematic.
Maximum unique plans: an upper bound on the number of unique redistricting plans that survive each stage. The percentage in parentheses is the ratio of this number to the total number of samples. Small values (< 100) indicate a bottleneck, which leads to a loss of sample diversity and a higher variance.
Estimated k
parameter: How many spanning tree edges were considered for
cutting at each split. Mostly informational, though large jumps may indicate
a need to increase adapt_k_thresh
.
Bottleneck: An asterisk will appear in the right column if a bottleneck appears likely, based on the values of the other statistics.
In the event of problematic diagnostics, the function will provide suggestions for improvement.
A data frame containing diagnostic information, invisibly.
data(iowa) iowa_map <- redist_map(iowa, ndists = 4, pop_tol = 0.1) plans <- redist_smc(iowa_map, 100) summary(plans)
data(iowa) iowa_map <- redist_map(iowa, ndists = 4, pop_tol = 0.1) plans <- redist_smc(iowa_map, 100) summary(plans)
Tally a variable by district
tally_var(map, x, .data = pl())
tally_var(map, x, .data = pl())
map |
a |
x |
a variable to tally. Tidy-evaluated. |
.data |
a |
a vector containing the tallied values by district and plan (column-major)