library( dplyr )
library( knitr )
library( pander )
library( stargazer )
library( scales )
set.seed( 1234 )
# set stargazer type to text for
# previewing in RMD docs but
# convert to type HTML when knitting
# (next code chunk)
"text" s.type <-
###################################
#
# STARGAZER SETTINGS
#
###################################
# DO NOT RUN CHUNK UNLESS KNITTING:
# changes table formats to html
# before rendering RMD docs
"html" s.type <-
Helper functions for the pairs() correlation table:
function(x, y, digits=2, prefix="", cex.cor)
panel.cor <-
{ par("usr"); on.exit(par(usr))
usr <-par(usr = c(0, 1, 0, 1))
abs(cor(x, y))
r <- format(c(r, 0.123456789), digits=digits)[1]
txt <- paste(prefix, txt, sep="")
txt <-if(missing(cex.cor)) cex <- 0.8/strwidth(txt)
cor.test(x,y)
test <-# borrowed from printCoefmat
symnum(test$p.value, corr = FALSE, na = FALSE,
Signif <-cutpoints = c(0, 0.001, 0.01, 0.05, 0.1, 1),
symbols = c("***", "**", "*", ".", " "))
text(0.5, 0.5, txt, cex = 1.5 )
text(.7, .8, Signif, cex=cex, col=2)
}
function (x, y, col = par("col"), bg = NA, pch = par("pch"),
panel.smooth <-cex = 0.5, col.smooth = "red", span = 2/3, iter = 3, ...)
{points(x, y, pch = 19, col = gray(0.7,0.2), bg = bg, cex = cex)
is.finite(x) & is.finite(y)
ok <-if (any(ok))
lines(stats::lowess(x[ok], y[ok], f = span, iter = iter),
col = col.smooth, lwd=2, ...)
}
# custom plot
function( x1, x2, lab1="", lab2="", draw.line=T, ... )
jplot <-
{
plot( x1, x2,
pch=19,
col=gray(0.6, alpha = 0.2),
cex=0.5,
bty = "n",
xlab=lab1,
ylab=lab2, cex.lab=1.5,
... )
if( draw.line==T ){
is.finite(x1) & is.finite(x2)
ok <-lines( lowess(x2[ok]~x1[ok]), col="red", lwd=3 ) }
}
The dependent variable in our model is the median home value. The choice draws on a large body of work on “hedonic pricing models” which assert that home values are a good price mechanism to capture the state of a neighborhood because they “price in” all of the characteristics of the house as well as the features of the neighborhood and surrounding city.
Anyone who has ever paid $2,500 to rent a 400-square foot apartment in New York City know that when you acquire a dwelling you are paying for location as much as for the space to store your things and sleep at night. And location really means proximity to amenities, assurance of safety, and access to things like low taxes or high-performing schools.
Hedonic pricing models use home value as the dependent variable, typically include features of the house (lot size, square footage, year built, number of bedrooms and bathrooms, etc.) to establish a baseline prediction of the value of the home based upon the house itself. They then incorporate many additional attributes of the neighborod to determine how people value things. Using this approach you can answer questions like, what is the value of having a park withing walking distance, i.e. what is the premium people will pay for homes that are near a park? What is the value of being in a good school district? What is the real-estate cost of proximity to a factory that produces toxic smoke? Does crime reduce the value of a home? Do new nearby bike lanes increase the value of homes?
We don’t have data on individual houses for this project, thus we can’t run formal hedonic models that account for the features of a home. Rather, we have a census estimate of the median home value in a census tract.
We also don’t have recent sales data that would tell us exactly how much people are paying for homes. Rather, the Census estimates the Median Home Value by asking people how much their homes are worth. There are definite limitations to this methodology (people who are attached to their homes might over-value them, people who don’t follow real estate might under-value them). But there is good work on “the wisdom of crowds” that shows that many independent guesses on topics like these, once aggregated, are pretty accurate.
This fact was established partly by watching people try to guess the weight of a pig at the state fair to win a prize. The typical person was not very good at the task and thus their estimate was off by several dozen pounds. But people that do not have experience guessing the weight of pigs will get close enough, and they are also equally as likely to guess too high as they are to guess to low. So if you take all of the guesses from the crowd and average them you typically come within a pound or two of the actual weight! See: Surowiecki, J. (2005). The wisdom of crowds.
Value is the respondent’s estimate of how much the property (house and lot) would sell for if it were for sale.
This tabulation includes only specified owner-occupied housing units–one-family houses on less than 10 acres without a business or medical office on the property. These data exclude mobile homes, houses with a business or medical office, houses on 10 or more acres, and housing units in multi-unit structures. Certain tabulations elsewhere include the value of all owner-occupied housing units and vacant-for-sale housing units. Also available are data on mortgage status and selected monthly owner costs.
The median divides the value distribution into two equal parts: one-half of the cases falling below the median value of the property (house and lot) and one-half above the median. Median value calculations are rounded to the nearest hundred dollars.
Owner-Occupied - A housing unit is owner-occupied if the owner or co-owner lives in the unit, even if it is mortgaged or not fully paid for. The owner or co-owner must live in the unit and usually is Person 1 on the questionnaire. The unit is “Owned by you or someone in this household with a mortgage or loan” if it is being purchased with a mortgage or some other debt arrangement such as a deed of trust, trust deed, contract to purchase, land contract, or purchase agreement.
Note that a census tract could feasibly contain a large number of businesses or rental units and a small number of houses or condos owned by citizens. These cases are conceptually challenging because the tract-level MHV is determined by a handful of homes. You will see below that we drop cases from the data with MHV less than $10,000 in 2000 and high home values in 2010. These are likely cases where the tract was a dirt lot that turned into a new development, or they were lofts or apartments in dense business districts converted to condos. In a few instances there were large public housing projects that were torn down and replaced with mixed use developments. As a result, many of the large percentage changes we see like a one thousand percent increase in home value are likely caused by small-sample problems and are not necessarily meaningful measures of change. The percent change variable is especially sensitive to small-sample problems because as the denominator approaches zero the value blows up to infinity. If you see a growth rate of 150,000 percent, it likely means that the median home value measure in 2000 was close to zero, not that the tract has actually changed in any meaningful way suggested by that magnitude.
There are many ways to deal with outliers like these, but for now we will simply filter out values above and below thresholds that we feel are unrealistic and focus on the 95 percent of the data within these bounds. If there was time we would do an in-depth analysis of those data points to ensure we are not omitting meaningful cases.
Similar to the hedonic housing models, the major assumption that our analysis relies upon is increases in the median home value in a neighbhorhood (Census tract) suggest the community is improving. We want to be careful here, because even the worst neighborhods in superstar cities will see home values increase, but at a much slower rate than all other neighborhods in the city. So “increase” needs to be contextualized.
When home values increase in a community, after controlling for metro-level “secular” market trends, the change suggests that the neighborhood is more desirable for citizens of the city. That can be driven by improvements in homes. Or it can be driven by improvements in amenities near the community such as new restaurants or light rail, quality of life through lower crime or better parks, or through gentrification processes that might displace disadvantaged populations that have lower income and savings and thus real estate markets were less competitive.
Median home value is going to capture all of these things, so at the very least it helps us measure change. Whether or not that change is a good thing depends on your perspective. People value good public transit, so they are willing to pay for proximity to light rail stations and metro stops. Even when a city can’t charge high rates for fares, property values increase as a result of new public infrastructure, and thus the city can recover some of the construction costs over time. These are largely synergistic changes that can incentivize cities to invest in transit.
Change diven by real estate speculation or by driving middle class families out of a neighborhood are still positive increases, but can be perceived as “bad” progress.
In short, we will pay attention to the nuances of gentrification, but generally speaking we will interpret an increase in home value being a signal that both the housing stock and the quality of the neighborhood have improved.
Since the US Census tries to ensure census tracts have populations between 2,000 and 4,000 individuals, regions that experience growth in population density require that tracts be split over time and regions that are losing population need to combine tracts. As a result, comparing tracts over time can be tricky since the geographies change.
Harmonization of census data is the process of translating data from multiple decades onto a single geography so that changes over time are meaningful. This requires a weighted sampling process called apportionment. Typically you use the most recent map because it will have the most geographic units (you are more likely to split tracts over time than to combine them), and thus supports the most granular analysis. Note that tracts are only redrawn each 10 years, so 2012 and 2017 data would come from the same geographies. Data from 2008 and 2012, however, would not (tracts were re-drawn in 2010).
For this lab we will use data from the Longitudinal Tract Database (LTDB) Project, a project that has harmonized a set of variables from 1970 to 2010.
Logan, John R., Zengwang Xu, and Brian J. Stults. 2014. “Interpolating US Decennial Census Tract Data from as Early as 1970 to 2010: A Longitudinal Tract Database” The Professional Geographer 66(3): 412–420. (Click here to download). Logan, John R., Brian J. Stults, and Zengwang Xu. 2016. “Validating Population Estimates for Harmonized Census Tract Data, 2000–2010” Annals of the American Association of Geographers.
"https://github.com/DS4PS/cpp-529-fall-2020/raw/main/LABS/data/rodeo/LTDB-2000.rds"
URL1 <- readRDS( gzcon( url( URL1 ) ) )
d1 <-
"https://github.com/DS4PS/cpp-529-fall-2020/raw/main/LABS/data/rodeo/LTDB-2010.rds"
URL2 <- readRDS( gzcon( url( URL2 ) ) )
d2 <-
"https://github.com/DS4PS/cpp-529-fall-2020/raw/main/LABS/data/rodeo/LTDB-META-DATA.rds"
URLmd <- readRDS( gzcon( url( URLmd ) ) )
md <-
select( d1, - year )
d1 <- select( d2, - year )
d2 <-
merge( d1, d2, by="tractid" )
d <- merge( d, md, by="tractid" ) d <-
Filter Rural Districts
table( d$urban )
##
## rural urban
## 12971 59722
filter( d, urban == "urban" ) d <-
Create New Variables
You will want to create more variables, but these examples will get you started.
select( d, tractid,
d <-
mhmval00, mhmval12,
hinc00,
hu00, vac00, own00, rent00, h30old00,
empclf00, clf00, unemp00, prof00,
dpov00, npov00,
ag25up00, hs00, col00,
pop00.x, nhwht00, nhblk00, hisp00, asian00,
cbsa, cbsaname )
d <- d %>%
mutate( # percent white in 2000
p.white = 100 * nhwht00 / pop00.x,
# percent black in 2000
p.black = 100 * nhblk00 / pop00.x,
# percent hispanic in 2000
p.hisp = 100 * hisp00 / pop00.x,
# percent asian in 2000
p.asian = 100 * asian00 / pop00.x,
# percent high school grads by age 25 in 2000
p.hs = 100 * (hs00+col00) / ag25up00,
# percent pop with college degree in 2000
p.col = 100 * col00 / ag25up00,
# percent employed in professional fields in 2000
p.prof = 100 * prof00 / empclf00,
# percent unemployment in 2000
p.unemp = 100 * unemp00 / clf00,
# percent of housing lots in tract that are vacant in 2000
p.vacant = 100 * vac00 / hu00,
# dollar change in median home value 2000 to 2010
pov.rate = 100 * npov00 / dpov00 )
# adjust 2000 home values for inflation
.00 <- d$mhmval00 * 1.28855
mhv.10 <- d$mhmval12
mhv
# change in MHV in dollars
mhv.10 - mhv.00
mhv.change <-
# drop low 2000 median home values
# to avoid unrealistic growth rates.
#
# tracts with homes that cost less than
# $1,000 are outliers
.00[ mhv.00 < 1000 ] <- NA
mhv
# change in MHV in percent
100 * ( mhv.change / mhv.00 )
mhv.growth <-
$mhv.00 <- mhv.00
d$mhv.10 <- mhv.10
d$mhv.change <- mhv.change
d$mhv.growth <- mhv.growth d
tractid <chr> | mhmval00 <dbl> | mhmval12 <dbl> | hinc00 <dbl> | hu00 <dbl> | ||
---|---|---|---|---|---|---|
1 | fips-01-001-020100 | 76600.0 | 121500 | 36685.00 | 768.9899 | |
2 | fips-01-001-020200 | 72900.0 | 130500 | 30298.00 | 731.0000 | |
3 | fips-01-001-020300 | 79900.0 | 118700 | 46731.00 | 1263.0000 | |
4 | fips-01-001-020400 | 89800.0 | 133500 | 46142.00 | 1871.0000 | |
5 | fips-01-001-020500 | 116593.5 | 174500 | 58886.27 | 2282.3435 | |
6 | fips-01-001-020600 | 70400.0 | 129600 | 33699.00 | 1309.5748 |
Median Home value
hist( mhv.00, breaks=200, xlim=c(0,500000),
col="gray20", border="white",
axes=F,
xlab="MHV (median = $138k)",
ylab="",
main="Median Home Value in 2000 (2010 US dollars)" )
axis( side=1, at=seq(0,500000,100000),
labels=c("$0","$100k","$200k","$300k","$400k","$500k") )
abline( v=median( mhv.00, na.rm=T ), col="orange", lwd=3 )
Descriptives
data.frame( MedianHomeValue2000=mhv.00,
df <-MedianHomeValue2010=mhv.10,
MHV.Change.00.to.10=mhv.change,
MHV.Growth.00.to.12=mhv.growth )
stargazer( df,
type=s.type,
digits=0,
summary.stat = c("min", "p25","median","mean","p75","max") )
Statistic | Min | Pctl(25) | Median | Mean | Pctl(75) | Max |
MedianHomeValue2000 | 1,280 | 105,661 | 154,884 | 187,119 | 224,337 | 1,288,551 |
MedianHomeValue2010 | 9,999 | 123,200 | 193,200 | 246,570 | 312,000 | 1,000,001 |
MHV.Change.00.to.10 | -1,228,651 | 7,187 | 36,268 | 60,047 | 94,881 | 1,000,001 |
MHV.Growth.00.to.12 | -97 | 6 | 25 | 34 | 50 | 17,494 |
Change in MHV 2000-2010
If a home worth $10 million increased in value by $100k over ten years it would not be that surprising. If a home worth $50k increased by $100k over the same period that is a growth of 200% and is notable.
The change in value variable only reports absolute change, but does not provide a sense of whether that is a big or small amount for the census tract.
hist( mhv.change/1000, breaks=500,
xlim=c(-100,500), yaxt="n", xaxt="n",
xlab="Thousand of US Dollars (adjusted to 2010)", cex.lab=1.5,
ylab="", main="Change in Median Home Value 2000 to 2010",
col="gray20", border="white" )
axis( side=1, at=seq( from=-100, to=500, by=100 ),
labels=paste0( "$", seq( from=-100, to=500, by=100 ), "k" ) )
mean( mhv.change/1000, na.rm=T )
mean.x <-abline( v=mean.x, col="darkorange", lwd=2, lty=2 )
text( x=200, y=1500,
labels=paste0( "Mean = ", dollar( round(1000*mean.x,0)) ),
col="darkorange", cex=1.8, pos=3 )
median( mhv.change/1000, na.rm=T )
median.x <-abline( v=median.x, col="dodgerblue", lwd=2, lty=2 )
text( x=200, y=2000,
labels=paste0( "Median = ", dollar( round(1000*median.x,0)) ),
col="dodgerblue", cex=1.8, pos=3 )
Percent Change in MHV 2000 to 2010
The percent change variable provides some context for growth rates of value in census tracts.
Plot the percent change variable:
hg <-hist( mhv.growth, breaks=5000,
xlim=c(-100,200), yaxt="n", xaxt="n",
xlab="", cex.main=1.5,
ylab="", main="Growth in Home Value by Census Tract 2000 to 2010",
col="gray40", border="white" )
axis( side=1, at=seq( from=-100, to=200, by=50 ),
labels=paste0( seq( from=-100, to=200, by=50 ), "%" ) )
max( hg$count )
ymax <-
mean( mhv.growth, na.rm=T )
mean.x <-abline( v=mean.x, col="darkorange", lwd=2, lty=2 )
text( x=100, y=(0.5*ymax),
labels=paste0( "Mean = ", round(mean.x,0), "%"),
col="darkorange", cex=1.8, pos=4 )
median( mhv.growth, na.rm=T )
median.x <-abline( v=median.x, col="dodgerblue", lwd=2, lty=2 )
text( x=100, y=(0.6*ymax),
labels=paste0( "Median = ", round(median.x,0), "%"),
col="dodgerblue", cex=1.8, pos=4 )
Metro Level Statistics
Both changes within a census tract and changes within a city can affect the price of a home. Since our policy of interest focuses on changes made within a tract (new business and new housing created through NMTC and LIHTC), we want to control for things happening at the metro level to ensure we are not inferring that programs that target the tract are driving outcomes when it is actually broad metro-level trends.
You might want to calculate several metro-level statistics for your model (growth in population in the city, changes in demographics, changes in industry structure and wealth, for example). You can calculate a metro level statistic by aggregating up (population count) or averaging across census tracts. For example:
# average growth in median home value for the city
d <- d %>%
group_by( cbsaname ) %>%
mutate( metro.mhv.change = median( mhv.change, na.rm=T ),
metro.mhv.growth = 100 * median( mhv.growth, na.rm=T ) ) %>%
ungroup()
We are going to build our policy inferences in two stages. This lab will start by modeling changes in median home values that are predicted by tract characteristics in 2000 and changes in the city between 2000 and 2010.
In the following lab we will then add our policy variables - measures of government programs.
Typically you will start with a simple model of the relationship of the policy and the outcome and add controls to improve the model. In some cases, especially when you have a large amount of data like this, it can be helpful building your models from the ground up starting with the controls. This ensures you understand all the specification considerations of your model before adding the one variable you care about. If your baseline model does not improve when adding the policy variable chances are the policy is not generating much impact.
Variable selection is the process of picking the variables and their functional form that you will include in the model.
It is one of the most important steps in the process, but also it is an art, not a science.
The goal in evaluation models is to achieve strong model fit while accounting for competing hypothses (alternative explanations to the assertion that the program or policy caused the change we see in the data).
Note that we have two types of variables in these models: metrics from 2000 and metrics of change between 2000 and 2010. We have two units of analysis - census tract and metro area. And we have two ways to represent quantitative variables - as counts and as proportions.
Sometimes only one type is meaningful - the population of a census tract is not very meaningful because tracts are designed to hold 2000 to 4000 people, and they split when they grow.
Population density, on the other hand, might be meaningful. The growth of the city between 2000 and 2010 is likely meaningful.
The number of white people in a tract might not be meaningful, but the proportion of white people in a tract (a measure of diversity) likely is.
So pay attention to how you are constructing variables, and whether the way you are presenting a variable (unit of analysis, time period, and count versus proportion) makes sense.
Let’s say we have 10 indendent variables to choose from, and we want a simple model with 3 independent variables.
Each variable can be represented at the community (tract) or the metro (cbsa) level (unit of analysis).
And each variable can be measured as counts or proportions.
So each variable turns into 4 potential variables:
So 10 variables x 2 units of anlysis x 2 metrics translates to approximately 40 distinct variables that we need to choose from:
So here is the real challanege: with 40 variables we can construct almost 10,000 unique 3-variable models!
choose( 40, 3 ) %>% format( big.mark="," )
## [1] "9,880"
We actually have close to 100 census variables in our dataset and more than 4 different ways to represent each, so we have more than 10 million possible models to explore.
choose( (4*100), 3 ) %>% format( big.mark="," )
## [1] "10,586,800"
The good news is that you should never construct models by trying random combinations of lots of variables.
Importance of Theory:
Use theory to guide your variables. One of the reasons you have been asked to read a lot of background material on the topic is to develop some sensitivity to which variables matter. We know that white flight drove housing prices down in central city neighborhoods from 1980 to 2000, and the re-discovery of city life has been driving the rise in home values from 2000 to 2010. Young professionals especially want to live in diverse urban communities near transit and amenities, i.e. not suburbs.
School quality has been a huge factor in driving the migration of the upper middle class to suburbs, so the rise in households that do not have children has removed one of the main barriers to urban living. The growth of cities has resulted in long and stressful commutes, making urban neighborhoods located near business districts more desirable. Baby boomers that fled cities when they started having kids are now selling their large suburban homes and buying smaller townhomes and condos that do not require as much maintenance and provide access to restaurants and cultural amenities that retired people have time to enjoy.
Understanding some of the basic factors driving change can help you identify variables that can serve as proxies for some of these factors.
It is helpful to test your understanding of your policy domain by trying to explain the main drivers of change to a hypothetical undergraduate student. Is the idea clear enough in your own mind to explain it to another person? How confident do you feel in your story? If you don’t feel comfortable with this exercise, perhaps you should do some more background reading and reflection before you try to model anything!
Specification is the process of selecting variables and their functional form for your model.
Typically you will go through the process of selecting variables that you believe capture your theory and allow you to test for important competing hypotheses.
After creating your baseline model you will likely want to explore the functional form of variables in your model to see if there are transformations that improve your model.
Throughout the process and in the final stages you will conduct sensitivity analysis and robustness checks to ensure that your results are not being biased or entirely driven by an outlier or the wrong functional form. There are many subtle ways that bias can sneak into models other than omitted variables. Recall the Anscombe examples where four datasets produced identical regression models (coefficients and significance levels) but have the following form:
One clear lesson is never rely entirely on model fit statistics reported by your statistical software. You need to look at your data, check summary statistics to make sure variables seem plausible, and visualize your data as much as possible.
Many problems are subtle. For example, maybe you are very comfortable with the functional form of your policy variable and your model fit. What happens if you transform a control variable? Change the measure from counts to proportions or add a quadratic term? Do your inferences remain the same, or do they vary widely when you alter controls?
A good data scientist is always paranoid about problems in the data that you are hard to see, especially when you want to use observational data to make consequential decisions, like whether or not to continue a multi-billion dollar federal program or mandate state-wide reductions in class size.
In CPP 523 you were introduced to the Seven Sins of Regression - issues that commonly plague cross-sectional models and prevent us from using observational data for causal inference.
This lab will highlight some elements of specification that can be harmful to the ability to use the models for inference.
We will introduce approaches to:
It is common to have data that is not normally distributed, especially in the social sciences. Variables that have a high level of skew or kurtosis can cause problems with inferential statistics by biasing slopes or inflating standard errors.
There are formal tests for skew, but the easiest way to diagnose the problem is by visualizing distributions using histograms or box and whisker plots.
Let’s start by looking at two of the census variables before and after log transformations:
par( mfrow=c(1,2) )
hist( d$pov.rate, breaks=50, col="gray20", border="white",
yaxt="n", xlab="", ylab="", main="Poverty Rate")
hist( log(d$pov.rate+1), breaks=50, col="gray20", border="white",
yaxt="n", xlab="", ylab="", main="Poverty Rate (logged)")
hist( d$p.vacant, breaks=50, col="gray20", border="white",
yaxt="n", xlab="", ylab="", main="Vacancy Rate")
hist( log(d$p.vacant+1), breaks=50, col="gray20", border="white",
yaxt="n", xlab="", ylab="", main="Vacancy Rate (logged)")
We know that bell-shaped curves are typically preferable, but it’s not clear how that helps when looking at a single variable. The issues are clearer when looking at the relationship between two skewed variables:
log10( d$p.vacant + 1 )
log.p.vacant <- log10( d$pov.rate + 1 )
log.pov.rate <- sample( 1:length(log.pov.rate), 5000 )
these <-
par( mfrow=c(1,2) )
jplot( d$pov.rate[these], d$p.vacant[these],
lab1="Poverty Rate", lab2="Vacant Homes (percent)",
main="Raw Measures" )
jplot( log.pov.rate[these], log.p.vacant[these],
lab1="Poverty Rate", lab2="Vacant Homes (percent)",
main="Log Transformed" )
We have now greatly strenghtened the relationship between these variables:
cor( d$p.vacant, d$pov.rate, use="pairwise.complete" )
[1] 0.2771027
cor( log.p.vacant, log.pov.rate, use="pairwise.complete" )
[1] 0.4373242
What exactly does a log transformation do again? Technically, they convert a one-unit change in a variable from a level to a rate of growth. You can read the more precise technical definition here.
But the real intuition comes from understanding the origins of skew. There are many processes in natural and social systems that are governed by exponential, not linear, growth. As the systems progress and differences compound, each additional unit of time results in larger changes in the system.
We can think of another example that originates from diminishing marginal returns. How much does it cost to increase the quality of a bottle of wine? Vinyards can use things like better grapes, better fertilizer, better viticulture, and better land to improve the quality of their product. Producers will do the cheapest things first, maybe buy new equipment to harvest and press grapes. Planting better grapes is expensive because it costs money for the vine stocks and grapes from newly-planted vines are not used in wine-making for at least a year. But they are fixed cost investments that can be amortized over time. Hiring a better vintner to oversee the production of wine will have a big impact on quality, but requires the ongoing cost of paying a large salary to a highly-paid specialist. When all of these options have been exhausted, the last step to improving wine quality is acquiring better land in a region amenable to grapes. This land, however, is very expensive.
As a result, if you ever pay attention to the price of wine at the store you will see that there are what appear to be several jumps in price. The cheapest table wine you can find costs around $3. It’s not great wine, so if you want to upgrade to a decent table wine the prices cluster around $6. If you are bringing a bottle to dinner at a friend’s house and you want something slightly nicer the price is about $12. If you want to splurge for a holiday celebration you will find the next step up in class clusters around $25, and after that $50.
Do you see the patern here? For each additional unit increase in quality the price doubles. It is an exponential scale. When you are producing cheap wine you can spend a small amount of money to get better equipment to improve quality by one point on this five point scale. The next step up in quality is more costly. And the step after that requires even more capital.
Consequently, if you look at data on wine prices you will find a skewed distribution with the market dominated by inexpensive wine and more limited selection in each additional quality traunch (Source-AC-Nielsen):
If we wanted to conduct a study on the relationship between wine price and customer satisfaction (spoiler alert, the relationship is nebulous) we would introduce specification bias by trying to model quality as a linear function of price.
Alternatively, if we take into account that the relationship between price and quality might not be linear and knowing that price is a skewed variable we can adjust our hypothesis and specification accordingly. It’s not that price doesn’t matter (as the article in the previous paragraph suggests) but we need a better specification to model the relationship. If we transform price using a log scale the relationship becomes clear.
In the regular levels data the relationship between price and satisfaction is very strong up until $20, then it weakens considerably. In the model where price has been converted to a log scale the relationship is a lot cleaner and the model is a better fit, which will results in smaller standard errors and a higher likelihood of finding results that support our theorized relationship between price and quality.
par( mfrow=c(1,2) )
jplot( price, quality,
lab1="Price of Wine (USD)",
lab2="Quality Rating")
abline( h=3.1, col="steelblue", lty=3 )
abline( v=20, col="steelblue", lty=3 )
jplot( log2(price), quality, xlim=c(1,6),
lab1="Price of Wine (log2)",
lab2="Quality Rating" )
Addressing Skew in Census Data:
Returning to our census variables, when we look at the relationship between variables in the scatterplot we see that a lot of the data clusters in the lower left hand corner of the scatterplots, which is a sign of skew:
# create subset to visualize in correlation matrix
select( d, mhv.growth, p.vacant, p.prof, pov.rate, p.unemp )
d2 <-
# reduce data density for visualization
set.seed( 1234 )
sample_n( d2, 10000 ) %>% na.omit()
d3 <-
# correlation plots
pairs( d3, upper.panel=panel.cor, lower.panel=panel.smooth )
After applying log transformations note that the bivariate correlations increase except for relationship between MHV and (mhv.growth) vacancy rates (p.vacant):
set.seed( 1234 )
select( d, mhv.growth, p.vacant, p.prof, pov.rate, p.unemp )
d2 <-
# recode some vars to remove outliers and skew
$mhv.growth[ d2$mhv.growth > 200 ] <- NA
d2$p.unemp <- log10( d2$p.unemp + 1 )
d2$p.vacant <- log10( d2$p.vacant + 1 )
d2$p.prof <- log10( d2$p.prof + 1 )
d2$pov.rate <- log10( d2$pov.rate + 1 )
d2
sample_n( d2, 5000 ) %>% na.omit()
d4 <-pairs( d4, upper.panel=panel.cor, lower.panel=panel.smooth )
The correlation between the transformed version of median home value change and vacancy rates falls, but is that a bad thing? The non-transformed version contains a statistically-significant correlation, but take a look at the scatterplot:
jplot( d3$p.vacant, d3$mhv.growth, ylim=c(-50,100),
lab1="Vacancy Rates", lab2="MHV Growth" )
It is likely an inflated correlation because of the handful of outliers. Does the graph look familiar? Recall the plot on the right from Anscombe’s Quartet:
So fixing skew made the correlation weaker, but represents a case of mitigating a spurious correlation that was caused by outliers in the data.
Recall that one way to think about a control variable that serves as a competing hypothesis is a variable that deletes “problematic” variance that might be the true explanation for the program change, reducing the variance available for the policy variable to the components that are independent (uncorrelated) with the controls.
Multicollinearity occurs when two variables in the model are highly correlated. If both are included in the model at the same time they can essentially cancel each other out. Standard errors will increase and slopes typically shift toward the null. As a result, even if they pair of variables are individually significant and have large effect sizes, they will have small slopes and lack statistical significance when included toghether.
Multicollinearity is common when there are mutiple variables that represent different measures of the same underlying construct in the same dataset. In studies of firms, for example, revenue, firm expenses, and number of employees are all measures of firm size and thus will be highly correlated. Including them all in the model will results in none of them having large or significant slopes.
This is important specifically when you need to interpret a regression coefficient. In the firm example, perhaps you are interested in whether or not companies that use a specific HR practice are more profitable, controlling for size and industry. In this case including all of the size controls together would not hurt because you are not interpretting the control in your analysis.
In our case, perhaps we are interested in whether economically distressed tracts are likely to grow faster than tracts that have already gentrified because of rent gap theory (urban core neighborhoods with lots condemned houses and vacant lots are easier to use for large-scale projects since land is cheap, so developers would target these communities instead of stable neighborhoods where land is expensive and zoning more complicated). If we include multiple measures of distress (unemployment rate, poverty rate, high school drop-out rate) and multiple measures of stability (number of people working in professional industries, percentage with college degrees, rates of divorce) then the high correlation between any set of variables can mute the slopes, making it hard to interpret the results.
Take this example where we try to include both poverty rate and unemployment. A sign of multicollinarity is when both coefficients get smaller and the standard errors get bigger when the two variables are included in the same model (model 3 here).
d
reg.data <-
$mhv.growth[ reg.data$mhv.growth > 200 ] <- NA
reg.data$p.unemp <- log10( reg.data$p.unemp + 1 )
reg.data$p.vacant <- log10( reg.data$p.vacant + 1 )
reg.data$pov.rate <- log10( reg.data$pov.rate + 1 )
reg.data
lm( mhv.growth ~ pov.rate, data=reg.data )
m1 <- lm( mhv.growth ~ p.unemp, data=reg.data )
m2 <- lm( mhv.growth ~ pov.rate + p.unemp, data=reg.data )
m3 <-
stargazer( m1, m2, m3,
type=s.type, digits=2,
omit.stat = c("rsq","f") )
Dependent variable: | |||
mhv.growth | |||
(1) | (2) | (3) | |
pov.rate | 12.93*** | 9.67*** | |
(0.42) | (0.61) | ||
p.unemp | 15.57*** | 6.04*** | |
(0.56) | (0.83) | ||
Constant | 16.69*** | 17.43*** | 15.24*** |
(0.44) | (0.46) | (0.48) | |
Observations | 58,798 | 58,801 | 58,791 |
Adjusted R2 | 0.02 | 0.01 | 0.02 |
Residual Std. Error | 34.89 (df = 58796) | 34.94 (df = 58799) | 34.87 (df = 58788) |
Note: | p<0.1; p<0.05; p<0.01 |
Note that:
*The standard error of the regression (SER) is the “average distance” of each data point to the regression line, so a measure of how accurate the model is when used for prediction. Similar to R-square, it is a measure of model fit."
This suggests that these variables contain redundant information and including them both together causes them to cancel each other out.
If these were simply control variables it would not harm your model, per se. But if these are competing hypotheses and thus you are interpretting the magnitude of effects relative to the policy variable you have now muted part of their effect, making them look less important. If they are highly correlated they can cancel each other out almost completely in your model.
When you are worried about multicollinearity:
If you have high multicollinearity and you want to interpret the variables then: