Chapter 13 The Western Collaborative Group Study

13.1 The Western Collaborative Group Study (wcgs) data set

Vittinghoff et al. (2012) explore data from the Western Collaborative Group Study (WCGS) in great detail9. We’ll touch lightly on some key issues in this Chapter.

# A tibble: 3,154 x 22
      id   age agec  height weight lnwght wghtcat   bmi   sbp lnsbp   dbp
   <int> <int> <fct>  <int>  <int>  <dbl> <fct>   <dbl> <int> <dbl> <int>
 1  2343    50 46-50     67    200   5.30 170-200  31.3   132  4.88    90
 2  3656    51 51-55     73    192   5.26 170-200  25.3   120  4.79    74
 3  3526    59 56-60     70    200   5.30 170-200  28.7   158  5.06    94
 4 22057    51 51-55     69    150   5.01 140-170  22.1   126  4.84    80
 5 12927    44 41-45     71    160   5.08 140-170  22.3   126  4.84    80
 6 16029    47 46-50     64    158   5.06 140-170  27.1   116  4.75    76
 7  3894    40 35-40     70    162   5.09 140-170  23.2   122  4.80    78
 8 11389    41 41-45     70    160   5.08 140-170  23.0   130  4.87    84
 9 12681    50 46-50     71    195   5.27 170-200  27.2   112  4.72    70
10 10005    43 41-45     68    187   5.23 170-200  28.4   120  4.79    80
# ... with 3,144 more rows, and 11 more variables: chol <int>,
#   behpat <fct>, dibpat <fct>, smoke <fct>, ncigs <int>, arcus <int>,
#   chd69 <fct>, typchd69 <int>, time169 <int>, t1 <dbl>, uni <dbl>

Here, we have 3154 rows (subjects) and 22 columns (variables). Since I used read.csv to import the data, and then converted to a tibble, all variables containing character data will appear as factors.

13.1.1 Structure of wcgs

We can specify the (sometimes terrible) variable names, through the names function, or we can add other elements of the structure, so that we can identify elements of particular interest.

Classes 'tbl_df', 'tbl' and 'data.frame':   3154 obs. of  22 variables:
 $ id      : int  2343 3656 3526 22057 12927 16029 3894 11389 12681 10005 ...
 $ age     : int  50 51 59 51 44 47 40 41 50 43 ...
 $ agec    : Factor w/ 5 levels "35-40","41-45",..: 3 4 5 4 2 3 1 2 3 2 ...
 $ height  : int  67 73 70 69 71 64 70 70 71 68 ...
 $ weight  : int  200 192 200 150 160 158 162 160 195 187 ...
 $ lnwght  : num  5.3 5.26 5.3 5.01 5.08 ...
 $ wghtcat : Factor w/ 4 levels "< 140","> 200",..: 4 4 4 3 3 3 3 3 4 4 ...
 $ bmi     : num  31.3 25.3 28.7 22.1 22.3 ...
 $ sbp     : int  132 120 158 126 126 116 122 130 112 120 ...
 $ lnsbp   : num  4.88 4.79 5.06 4.84 4.84 ...
 $ dbp     : int  90 74 94 80 80 76 78 84 70 80 ...
 $ chol    : int  249 194 258 173 214 206 190 212 130 233 ...
 $ behpat  : Factor w/ 4 levels "A1","A2","B3",..: 1 1 1 1 1 1 1 1 1 1 ...
 $ dibpat  : Factor w/ 2 levels "Type A","Type B": 1 1 1 1 1 1 1 1 1 1 ...
 $ smoke   : Factor w/ 2 levels "No","Yes": 2 2 1 1 1 2 1 2 1 2 ...
 $ ncigs   : int  25 25 0 0 0 80 0 25 0 25 ...
 $ arcus   : int  1 0 1 1 0 0 0 0 1 0 ...
 $ chd69   : Factor w/ 2 levels "No","Yes": 1 1 1 1 1 1 1 1 1 1 ...
 $ typchd69: int  0 0 0 0 0 0 0 0 0 0 ...
 $ time169 : int  1367 2991 2960 3069 3081 2114 2929 3010 3104 2861 ...
 $ t1      : num  -1.63 -4.06 0.64 1.12 2.43 ...
 $ uni     : num  0.486 0.186 0.728 0.624 0.379 ...

13.1.2 Codebook for wcgs

This table was lovingly hand-crafted, and involved a lot of typing. We’ll look for better ways in 432.

Name Stored As Type Details (units, levels, etc.)
id integer (nominal) ID #, nominal and uninteresting
age integer quantitative age, in years - no decimal places
agec factor (5) (ordinal) age: 35-40, 41-45, 46-50, 51-55, 56-60
height integer quantitative height, in inches
weight integer quantitative weight, in pounds
lnwght number quantitative natural logarithm of weight
wghtcat factor (4) (ordinal) wt: < 140, 140-170, 170-200, > 200
bmi number quantitative body-mass index:
703 * weight in lb / (height in in)2
sbp integer quantitative systolic blood pressure, in mm Hg
lnsbp number quantitative natural logarithm of sbp
dbp integer quantitative diastolic blood pressure, mm Hg
chol integer quantitative total cholesterol, mg/dL
behpat factor (4) (nominal) behavioral pattern: A1, A2, B3 or B4
dibpat factor (2) (binary) behavioral pattern: A or B
smoke factor (2) (binary) cigarette smoker: Yes or No
ncigs integer quantitative number of cigarettes smoked per day
arcus integer (nominal) arcus senilis present (1) or absent (0)
chd69 factor (2) (binary) CHD event: Yes or No
typchd69 integer (4 levels) event: 0 = no CHD, 1 = MI or SD,
2 = silent MI, 3 = angina
time169 integer quantitative follow-up time in days
t1 number quantitative heavy-tailed (random draws)
uni number quantitative light-tailed (random draws)

13.1.3 Quick Summary

       id             age          agec          height         weight   
 Min.   : 2001   Min.   :39.0   35-40: 543   Min.   :60.0   Min.   : 78  
 1st Qu.: 3741   1st Qu.:42.0   41-45:1091   1st Qu.:68.0   1st Qu.:155  
 Median :11406   Median :45.0   46-50: 750   Median :70.0   Median :170  
 Mean   :10478   Mean   :46.3   51-55: 528   Mean   :69.8   Mean   :170  
 3rd Qu.:13115   3rd Qu.:50.0   56-60: 242   3rd Qu.:72.0   3rd Qu.:182  
 Max.   :22101   Max.   :59.0                Max.   :78.0   Max.   :320  
                                                                         
     lnwght        wghtcat          bmi            sbp          lnsbp     
 Min.   :4.36   < 140  : 232   Min.   :11.2   Min.   : 98   Min.   :4.58  
 1st Qu.:5.04   > 200  : 213   1st Qu.:23.0   1st Qu.:120   1st Qu.:4.79  
 Median :5.14   140-170:1538   Median :24.4   Median :126   Median :4.84  
 Mean   :5.13   170-200:1171   Mean   :24.5   Mean   :129   Mean   :4.85  
 3rd Qu.:5.20                  3rd Qu.:25.8   3rd Qu.:136   3rd Qu.:4.91  
 Max.   :5.77                  Max.   :38.9   Max.   :230   Max.   :5.44  
                                                                          
      dbp           chol     behpat       dibpat     smoke     
 Min.   : 58   Min.   :103   A1: 264   Type A:1589   No :1652  
 1st Qu.: 76   1st Qu.:197   A2:1325   Type B:1565   Yes:1502  
 Median : 80   Median :223   B3:1216                           
 Mean   : 82   Mean   :226   B4: 349                           
 3rd Qu.: 86   3rd Qu.:253                                     
 Max.   :150   Max.   :645                                     
               NA's   :12                                      
     ncigs          arcus       chd69         typchd69        time169    
 Min.   : 0.0   Min.   :0.000   No :2897   Min.   :0.000   Min.   :  18  
 1st Qu.: 0.0   1st Qu.:0.000   Yes: 257   1st Qu.:0.000   1st Qu.:2842  
 Median : 0.0   Median :0.000              Median :0.000   Median :2942  
 Mean   :11.6   Mean   :0.299              Mean   :0.136   Mean   :2684  
 3rd Qu.:20.0   3rd Qu.:1.000              3rd Qu.:0.000   3rd Qu.:3037  
 Max.   :99.0   Max.   :1.000              Max.   :3.000   Max.   :3430  
                NA's   :2                                                
       t1             uni       
 Min.   :-47.4   Min.   :0.001  
 1st Qu.: -1.0   1st Qu.:0.257  
 Median :  0.0   Median :0.516  
 Mean   :  0.0   Mean   :0.505  
 3rd Qu.:  1.0   3rd Qu.:0.756  
 Max.   : 47.0   Max.   :0.999  
 NA's   :39                     

For a more detailed description, we might consider Hmisc::describe, psych::describe, mosaic::favstats, etc.

13.2 Are the SBPs Normally Distributed?

Consider the question of whether the distribution of the systolic blood pressure results is well-approximated by the Normal.

Since the data contain both sbp and lnsbp (its natural logarithm), let’s compare them. Note that in preparing the graph, we’ll need to change the location for the text annotation.

We can also look at Normal Q-Q plots, for instance…

There’s at best a small improvement from sbp to lnsbp in terms of approximation by a Normal distribution.

13.3 Describing Outlying Values with Z Scores

It looks like there’s an outlier (or a series of them) in the SBP data.

sbp 
       n  missing distinct     Info     Mean      Gmd      .05      .10 
    3154        0       62    0.996    128.6    16.25      110      112 
     .25      .50      .75      .90      .95 
     120      126      136      148      156 

lowest :  98 100 102 104 106, highest: 200 208 210 212 230

The maximum value here is 230, and is clearly the most extreme value in the data set. One way to gauge this is to describe that observation’s Z score, the number of standard deviations away from the mean that the observation falls. Here, the maximum value, 230 is 6.71 standard deviations above the mean, and thus has a Z score of 6.7.

A negative Z score would indicate a point below the mean, while a positive Z score indicates, as we’ve seen, a point above the mean. The minimum systolic blood pressure, 98 is 2.03 standard deviations below the mean, so it has a Z score of -2.

Recall that the Empirical Rule suggests that if a variable follows a Normal distribution, it would have approximately 95% of its observations falling inside a Z score of (-2, 2), and 99.74% falling inside a Z score range of (-3, 3). Do the systolic blood pressures appear Normally distributed?

13.5 Re-Leveling a Factor

Well, that’s not so good. We really want those weight categories (the levels) to be ordered more sensibly.

 wghtcat    n percent
   < 140  232  0.0736
   > 200  213  0.0675
 140-170 1538  0.4876
 170-200 1171  0.3713

Like all factor variables in R, the categories are specified as levels.

[1] "< 140"   "> 200"   "140-170" "170-200"

We want to change the order of the levels in a new version of this factor variable so they make sense. There are multiple ways to do this, but I prefer the fct_relevel function from the forcats package. Which order is more appropriate?

I’ll add a new variable to the wcgs data called weight_f that relevels the wghtcat data.

 weight_f    n percent
    < 140  232  0.0736
  140-170 1538  0.4876
  170-200 1171  0.3713
    > 200  213  0.0675

For more on the forcats package, check out Grolemund and Wickham (2019), especially the Section on Factors.

13.5.1 SBP by Weight Category

We might see some details well with a ridgeline plot, too.

Picking joint bandwidth of 3.74

As the plots suggest, patients in the heavier groups generally had higher systolic blood pressures.

  weight_f min  Q1 median  Q3 max mean   sd    n missing
1    < 140  98 112    120 130 196  123 14.7  232       0
2  140-170 100 118    124 134 192  126 13.7 1538       0
3  170-200 100 120    130 140 230  131 15.6 1171       0
4    > 200 110 126    132 150 212  138 16.8  213       0

13.6 Are Weight and SBP Linked?

Let’s build a scatter plot of SBP (Outcome) by Weight (Predictor), rather than breaking down into categories.

  • The mass of the data is hidden from us - showing 3154 points in one plot can produce little more than a blur where there are lots of points on top of each other.
  • Here the least squares regression line (in red), and loess scatterplot smoother, (in blue) can help.

The relationship between systolic blood pressure and weight appears to be very close to linear, but of course there is considerable scatter around that generally linear relationship. It turns out that the Pearson correlation of these two variables is 0.253.

13.7 SBP and Weight by Arcus Senilis groups?

An issue of interest to us will be to assess whether the SBP-Weight relationship we see above is similar among subjects who have arcus senilis and those who do not.

Arcus senilis is an old age syndrome where there is a white, grey, or blue opaque ring in the corneal margin (peripheral corneal opacity), or white ring in front of the periphery of the iris. It is present at birth but then fades; however, it is quite commonly present in the elderly. It can also appear earlier in life as a result of hypercholesterolemia.

Wikipedia article on Arcus Senilis, retrieved 2017-08-15

Let’s start with a quick look at the arcus data.

 arcus    n  percent valid_percent
     0 2211 0.701015         0.701
     1  941 0.298351         0.299
    NA    2 0.000634            NA

We have 2 missing values, so we probably want to do something about that before plotting the data, and we may also want to create a factor variable with more meaningful labels than 1 (which means yes, arcus senilis is present) and 0 (which means no, it isn’t.)

          arcus_f    0   1 NA_
    Arcus senilis    0 941   0
 No arcus senilis 2211   0   0
             <NA>    0   0   2

Let’s build a version of the wcgs data that eliminates all missing data in the variables of immediate interest, and then plot the SBP-weight relationship in groups of patients with and without arcus senilis.

13.8 Linear Model for SBP-Weight Relationship: subjects without Arcus Senilis


Call:
lm(formula = sbp ~ weight, data = filter(wcgs, arcus == 0))

Residuals:
   Min     1Q Median     3Q    Max 
-29.01 -10.25  -2.45   7.55  99.85 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept)  95.9219     2.5552    37.5   <2e-16 ***
weight        0.1902     0.0149    12.8   <2e-16 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 14.8 on 2209 degrees of freedom
Multiple R-squared:  0.0687,    Adjusted R-squared:  0.0683 
F-statistic:  163 on 1 and 2209 DF,  p-value: <2e-16

The linear model for the 2211 patients without Arcus Senilis has R2 = 6.87%.

  • The regression equation is 95.92 - 0.19 weight, for those patients without Arcus Senilis.

13.9 Linear Model for SBP-Weight Relationship: subjects with Arcus Senilis


Call:
lm(formula = sbp ~ weight, data = filter(wcgs, arcus == 1))

Residuals:
   Min     1Q Median     3Q    Max 
-30.34  -9.64  -1.96   7.97  76.74 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept)  101.879      3.756   27.13  < 2e-16 ***
weight         0.163      0.022    7.39  3.3e-13 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 14.2 on 939 degrees of freedom
Multiple R-squared:  0.0549,    Adjusted R-squared:  0.0539 
F-statistic: 54.6 on 1 and 939 DF,  p-value: 3.29e-13

The linear model for the 941 patients with Arcus Senilis has R2 = 5.49%.

  • The regression equation is 101.88 - 0.163 weight, for those patients with Arcus Senilis.

13.10 Including Arcus Status in the model


Call:
lm(formula = sbp ~ weight * arcus, data = filter(wcgs, !is.na(arcus)))

Residuals:
   Min     1Q Median     3Q    Max 
-30.34 -10.15  -2.35   7.67  99.85 

Coefficients:
             Estimate Std. Error t value Pr(>|t|)    
(Intercept)   95.9219     2.5244   38.00   <2e-16 ***
weight         0.1902     0.0147   12.92   <2e-16 ***
arcus          5.9566     4.6197    1.29     0.20    
weight:arcus  -0.0276     0.0270   -1.02     0.31    
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 14.6 on 3148 degrees of freedom
Multiple R-squared:  0.066, Adjusted R-squared:  0.0651 
F-statistic: 74.1 on 3 and 3148 DF,  p-value: <2e-16

The actual regression equation in this setting includes both weight, and an indicator variable (1 = yes, 0 = no) for arcus senilis status, and the product of weight and that 1/0 indicator.

  • Note the use of the product term weight*arcus in the setup of the model to allow both the slope of weight and the intercept term in the model to change depending on arcus senilis status.
    • For a patient who has arcus, the regression equation is SBP = 95.92 + 0.19 weight + 5.96 (1) - 0.028 weight (1) = 101.88 + 0.162 weight.
    • For a patient without arcus senilis, the regression equation is SBP = 95.92 + 0.19 weight + 5.96 (0) - 0.028 weight (0) = 95.92 + 0.19 weight.

The linear model including the interaction of weight and arcus to predict sbp for the 3152 patients with known Arcus Senilis status has R2 = 6.6%.

13.11 Predictions from these Linear Models

What is our predicted SBP for a subject weighing 175 pounds?

How does that change if our subject weighs 200 pounds?

Recall that

  • Without Arcus Senilis, linear model for SBP = 95.9 + 0.19 x weight
  • With Arcus Senilis, linear model for SBP = 101.9 + 0.16 x weight

So the predictions for a 175 pound subject are:

  • 95.9 + 0.19 x 175 = 129 mm Hg without Arcus Senilis, and

  • 101.9 + 0.16 x 175 = 130 mm Hg with Arcus Senilis.

And thus, the predictions for a 200 pound subject are:

  • 95.9 + 0.19 x 200 = 134 mm Hg without Arcus Senilis, and

  • 101.9 + 0.16 x 200 = 134.4 mm Hg with Arcus Senilis.

13.13 Scatterplot and Correlation Matrices

A scatterplot matrix can be very helpful in understanding relationships between multiple variables simultaneously. There are several ways to build such a thing, including the pairs function…

13.13.1 Using the car package

Or, we can use the scatterplotMatrix function from the car package, which adds some detail and fitting to the plots, and places density estimates (with rug plots) on the diagonals.

13.13.2 Displaying a Correlation Matrix

          sbp     age  weight  height
sbp    1.0000  0.1660  0.2530  0.0184
age    0.1660  1.0000 -0.0344 -0.0954
weight 0.2530 -0.0344  1.0000  0.5330
height 0.0184 -0.0954  0.5330  1.0000

13.13.3 Using the GGally package

The ggplot2 system doesn’t have a built-in scatterplot system. There are some nice add-ins in the world, though. One option I sort of like is in the GGally package, which can produce both correlation matrices and scatterplot matrices.

The ggpairs function provides a density plot on each diagonal, Pearson correlations on the upper right and scatterplots on the lower left of the matrix.

References

Grolemund, Garrett, and Hadley Wickham. 2019. R for Data Science. O’Reilly. http://r4ds.had.co.nz/.

Vittinghoff, Eric, David V. Glidden, Stephen C. Shiboski, and Charles E. McCulloch. 2012. Regression Methods in Biostatistics: Linear, Logistic, Survival, and Repeated Measures Models. Second. Springer-Verlag, Inc. http://www.biostat.ucsf.edu/vgsm/.