Copyright 2016 Allen B. Downey

MIT License: https://opensource.org/licenses/MIT

```
In [1]:
```from __future__ import print_function, division
%matplotlib inline
import numpy as np
import nsfg
import first

Given a list of values, there are several ways to count the frequency of each value.

```
In [2]:
```t = [1, 2, 2, 3, 5]

You can use a Python dictionary:

```
In [3]:
```hist = {}
for x in t:
hist[x] = hist.get(x, 0) + 1
hist

```
Out[3]:
```

You can use a `Counter`

(which is a dictionary with additional methods):

```
In [4]:
```from collections import Counter
counter = Counter(t)
counter

```
Out[4]:
```

Or you can use the `Hist`

object provided by `thinkstats2`

:

```
In [5]:
```import thinkstats2
hist = thinkstats2.Hist([1, 2, 2, 3, 5])
hist

```
Out[5]:
```

`Hist`

provides `Freq`

, which looks up the frequency of a value.

```
In [6]:
```hist.Freq(2)

```
Out[6]:
```

You can also use the bracket operator, which does the same thing.

```
In [7]:
```hist[2]

```
Out[7]:
```

If the value does not appear, it has frequency 0.

```
In [8]:
```hist[4]

```
Out[8]:
```

The `Values`

method returns the values:

```
In [9]:
```hist.Values()

```
Out[9]:
```

So you can iterate the values and their frequencies like this:

```
In [10]:
```for val in sorted(hist.Values()):
print(val, hist[val])

```
```

Or you can use the `Items`

method:

```
In [11]:
```for val, freq in hist.Items():
print(val, freq)

```
```

`thinkplot`

is a wrapper for `matplotlib`

that provides functions that work with the objects in `thinkstats2`

.

For example `Hist`

plots the values and their frequencies as a bar graph.

`Config`

takes parameters that label the x and y axes, among other things.

```
In [12]:
```import thinkplot
thinkplot.Hist(hist)
thinkplot.Config(xlabel='value', ylabel='frequency')

```
```

As an example, I'll replicate some of the figures from the book.

First, I'll load the data from the pregnancy file and select the records for live births.

```
In [13]:
```preg = nsfg.ReadFemPreg()
live = preg[preg.outcome == 1]

`Hist`

works with anything iterable, including a Pandas Series. The `label`

attribute appears in the legend when you plot the `Hist`

.

```
In [14]:
```hist = thinkstats2.Hist(live.birthwgt_lb, label='birthwgt_lb')
thinkplot.Hist(hist)
thinkplot.Config(xlabel='Birth weight (pounds)', ylabel='Count')

```
```

Before plotting the ages, I'll apply `floor`

to round down:

```
In [15]:
```ages = np.floor(live.agepreg)

```
In [16]:
```hist = thinkstats2.Hist(ages, label='agepreg')
thinkplot.Hist(hist)
thinkplot.Config(xlabel='years', ylabel='Count')

```
```

As an exercise, plot the histogram of pregnancy lengths (column `prglngth`

).

```
In [17]:
```# Solution goes here
length = live.prglngth

```
In [18]:
```hist = thinkstats2.Hist(length, label = 'preglngth')
thinkplot.Hist(hist)
thinkplot.Config(xlabel='Weeks', ylabel='Count')

```
```

`Hist`

provides smallest, which select the lowest values and their frequencies.

```
In [19]:
```for weeks, freq in hist.Smallest(10):
print(weeks, freq)

```
```

Use `Largest`

to display the longest pregnancy lengths.

```
In [20]:
```# Solution goes here
for weeks, freq in hist.Largest(10):
print(weeks, freq)

```
```

`birthord`

, then compute histograms of pregnancy length for the two groups.

```
In [21]:
```firsts = live[live.birthord == 1]
others = live[live.birthord != 1]
first_hist = thinkstats2.Hist(firsts.prglngth, label='first')
other_hist = thinkstats2.Hist(others.prglngth, label='other')

We can use `width`

and `align`

to plot two histograms side-by-side.

```
In [22]:
```width = 0.45
thinkplot.PrePlot(2)
thinkplot.Hist(first_hist, align='right', width=width)
thinkplot.Hist(other_hist, align='left', width=width)
thinkplot.Config(xlabel='weeks', ylabel='Count', xlim=[27, 46])

```
```

`Series`

provides methods to compute summary statistics:

```
In [23]:
```mean = live.prglngth.mean()
var = live.prglngth.var()
std = live.prglngth.std()

Here are the mean and standard deviation:

```
In [24]:
```mean, std

```
Out[24]:
```

As an exercise, confirm that `std`

is the square root of `var`

:

```
In [25]:
```# Solution goes here
std == np.sqrt(var)

```
Out[25]:
```

Here's are the mean pregnancy lengths for first babies and others:

```
In [26]:
```firsts.prglngth.mean(), others.prglngth.mean()

```
Out[26]:
```

And here's the difference (in weeks):

```
In [27]:
```firsts.prglngth.mean() - others.prglngth.mean()

```
Out[27]:
```

```
In [28]:
```def CohenEffectSize(group1, group2):
"""Computes Cohen's effect size for two groups.
group1: Series or DataFrame
group2: Series or DataFrame
returns: float if the arguments are Series;
Series if the arguments are DataFrames
"""
diff = group1.mean() - group2.mean()
var1 = group1.var()
var2 = group2.var()
n1, n2 = len(group1), len(group2)
pooled_var = (n1 * var1 + n2 * var2) / (n1 + n2)
d = diff / np.sqrt(pooled_var)
return d

Compute the Cohen effect size for the difference in pregnancy length for first babies and others.

```
In [29]:
```# Solution goes here
CohenEffectSize(firsts.prglngth, others.prglngth)

```
Out[29]:
```

Using the variable `totalwgt_lb`

, investigate whether first babies are lighter or heavier than others.

Compute Cohen’s effect size to quantify the difference between the groups. How does it compare to the difference in pregnancy length?

```
In [30]:
```# Solution goes here
firsts_hist=thinkstats2.Hist(np.round(firsts.totalwgt_lb), label='firsts')
others_hist=thinkstats2.Hist(np.round(others.totalwgt_lb), label='others')
print('mean')
print('firsts:', firsts.totalwgt_lb.mean())
print('others:', others.totalwgt_lb.mean())
print('')
print('stdev')
print('firsts:', firsts.totalwgt_lb.mean())
print('others:', others.totalwgt_lb.mean())
print('')
print('median')
print('firsts:', firsts.totalwgt_lb.median())
print('others:', others.totalwgt_lb.median())
width=0.45
thinkplot.PrePlot(2)
thinkplot.Hist(firsts_hist, align='left', width=width)
thinkplot.Hist(others_hist, align='right', width=width)
thinkplot.Config(xlabel='pounds', ylabel='Count', xlim=(0, 15))

```
```

```
In [58]:
```# Solution goes here
CohenEffectSize(firsts.totalwgt_lb, others.totalwgt_lb)

```
Out[58]:
```

For the next few exercises, we'll load the respondent file:

```
In [31]:
```resp = nsfg.ReadFemResp()

`totincr` the total income for the respondent's family. To interpret the codes see the codebook.

```
In [34]:
```# Solution goes here
inc_hist = thinkstats2.Hist(resp.totincr)
thinkplot.Hist(inc_hist)

```
```

Make a histogram of `age_r`, the respondent's age at the time of interview.

```
In [35]:
```# Solution goes here
age_hist = thinkstats2.Hist(resp.age_r)
thinkplot.Hist(age_hist)

```
```

Make a histogram of `numfmhh`, the number of people in the respondent's household.

```
In [36]:
```# Solution goes here
fmhh_hist = thinkstats2.Hist(resp.numfmhh)
thinkplot.Hist(fmhh_hist)

```
```

`parity`, the number of children borne by the respondent. How would you describe this distribution?

```
In [37]:
```# Solution goes here
parity_hist = thinkstats2.Hist(resp.parity)
thinkplot.Hist(parity_hist)

```
```

Use Hist.Largest to find the largest values of `parity`.

```
In [39]:
```# Solution goes here
parity_hist.Largest(10)

```
Out[39]:
```

Let's investigate whether people with higher income have higher parity. Keep in mind that in this study, we are observing different people at different times during their lives, so this data is not the best choice for answering this question. But for now let's take it at face value.

Use `totincr` to select the respondents with the highest income (level 14). Plot the histogram of `parity` for just the high income respondents.

```
In [44]:
```# Solution goes here
hinc = resp[resp['totincr'] == 14]
other = resp[resp['totincr'] < 14]
hinc_hist = thinkstats2.Hist(hinc.parity)
thinkplot.Hist(hinc_hist)

```
```

Find the largest parities for high income respondents.

```
In [45]:
```# Solution goes here
hinc_hist.Largest(5)

```
Out[45]:
```

Compare the mean `parity` for high income respondents and others.

```
In [49]:
```# Solution goes here
print('mean parity, high income:', hinc.parity.mean())
print('mean parity, other:', other.parity.mean())

```
```

```
In [50]:
```# Solution goes here
CohenEffectSize(hinc.parity, other.parity)

```
Out[50]:
```