library(tidyverse)
library(marginaleffects)
Worksheet 1
Two problems to work through.
Packages you’ll need:
Veggie burgers
You like hamburgers, but you are a vegetarian. What to do? Today, there are many brands of hamburgers without any meat in them. Some of these are designed to taste like meat, and some have their own flavour. A magazine rated the flavour and texture of 12 different (numbered) brands of meatless hamburgers (to give a rating score between 0 and 100), along with the price (in cents), the number of calories, the grams of fat, and the milligrams of sodium. These measurements are per burger. Is it possible to predict the rating score of a brand of meatless hamburger from the other measurements, and if so, how? The data are in http://ritsokiguess.site/datafiles/veggie-burgers.txt, in aligned columns.
(Some of this question appeared on the C32 Worksheet 11. There is some duplication, but I have tried to remove most of it.)
- Read in and display (most of) the data.
Solution
Aligned columns says read_table
:
<- "http://ritsokiguess.site/datafiles/veggie-burgers.txt"
my_url <- read_table(my_url) burgers
── Column specification ────────────────────────────────────────────────────────
cols(
brand = col_double(),
score = col_double(),
price = col_double(),
calories = col_double(),
fat = col_double(),
sodium = col_double()
)
burgers
There are 12 rows, one per brand, and the columns are as promised (and all quantitative, except for brand
, which is an identifier).
- Fit a suitable regression to predict score from the other measured variables (that is to say, not
brand
: why?). Display the results.
Solution
The brand is an identifier, so skip that:
.1 <- lm(score ~ price + calories + fat + sodium, data = burgers)
burgerssummary(burgers.1)
Call:
lm(formula = score ~ price + calories + fat + sodium, data = burgers)
Residuals:
Min 1Q Median 3Q Max
-19.376 -5.358 1.843 7.027 13.454
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 59.84879 35.67526 1.678 0.1373
price 0.12868 0.33907 0.380 0.7156
calories -0.58048 0.28876 -2.010 0.0843 .
fat 8.49825 3.47215 2.448 0.0443 *
sodium 0.04876 0.04062 1.200 0.2690
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 12.72 on 7 degrees of freedom
Multiple R-squared: 0.4991, Adjusted R-squared: 0.2128
F-statistic: 1.744 on 4 and 7 DF, p-value: 0.2443
My numbering scheme for models is based on the name of the dataframe. Base yours on the response variable score
if you prefer. But have a scheme, since there is going to be more than one model in this question.
- It looks as if both
price
andsodium
will be able to be removed from this regression. Do so, explain briefly why another test is necessary, and do that other test. What do you conclude? (Note: if you display your output to the second regression, something rather odd will appear. You can safely ignore that.)
Solution
There are several things to keep straight. The first thing is to fit a model without price
and sodium
. The easiest way to do this is to copy, paste and edit:
.2 <- lm(score ~ calories + fat, data = burgers)
burgerssummary(burgers.2)
Call:
lm(formula = score ~ calories + fat, data = burgers)
Residuals:
Min 1Q Median 3Q Max
-17.919 -6.786 -4.352 11.198 16.786
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 75.5907 24.4377 3.093 0.0129 *
calories -0.4600 0.2701 -1.703 0.1227
fat 7.7047 3.3703 2.286 0.0481 *
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 12.56 on 9 degrees of freedom
Multiple R-squared: 0.3725, Adjusted R-squared: 0.233
F-statistic: 2.671 on 2 and 9 DF, p-value: 0.1228
Aside: the odd thing is that calories
is no longer significant. This is confusing, because usually what happens is that explanatory variables become more significant when other non-significant variables are removed. So now you may be thinking that calories
should be removed as well. But see the Extra for what happens when you do that. This is why I said to ignore the odd thing. End of aside.
However, we removed two explanatory variables at once. This is not supported by the \(t\)-tests in the output from burgers.1
, because they only say what will happen if we remove one \(x\)-variable. Hence, we need a test that says whether removing those two \(x\)s at once was reasonable. That is this test:
anova(burgers.2, burgers.1, test = "F")
(That is F
in quotes, saying to do an \(F\)-test, not that anything is FALSE
.)
With a P-value of 0.45, this is saying that there is no significant difference in fit between the two models, and so we should prefer the smaller, simpler one burgers.2
, with just calories
and fat
in it, because it fits just as well as the bigger, more complicated one (and therefore we do not need the extra complication).
Extra: What happens if you do backward elimination from here, starting from the best model found so far?
The previous part told us that predicting score from just calories and fat was the best thing to do so far. That was the model I called burgers.2
:
summary(burgers.2)
Call:
lm(formula = score ~ calories + fat, data = burgers)
Residuals:
Min 1Q Median 3Q Max
-17.919 -6.786 -4.352 11.198 16.786
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 75.5907 24.4377 3.093 0.0129 *
calories -0.4600 0.2701 -1.703 0.1227
fat 7.7047 3.3703 2.286 0.0481 *
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 12.56 on 9 degrees of freedom
Multiple R-squared: 0.3725, Adjusted R-squared: 0.233
F-statistic: 2.671 on 2 and 9 DF, p-value: 0.1228
Evidently, calories
comes out now:
.3 <- lm(score ~ fat, data = burgers)
burgerssummary(burgers.3)
Call:
lm(formula = score ~ fat, data = burgers)
Residuals:
Min 1Q Median 3Q Max
-17.186 -8.538 -2.136 5.898 22.814
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 35.390 6.906 5.124 0.000448 ***
fat 2.949 2.059 1.432 0.182578
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 13.7 on 10 degrees of freedom
Multiple R-squared: 0.1702, Adjusted R-squared: 0.08724
F-statistic: 2.051 on 1 and 10 DF, p-value: 0.1826
Now, there is only one explanatory variable left, and it is no longer significant, so it too has to come out now! This seems to make no sense, since fat
was definitely significant before, and we would expect it still to be significant after removing something that was not significant. (Sometimes this happens; this is one of those cases.)
Another way of expressing your surprise is to look at the R-squared values (or the adjusted R-squared values) for the models we have fit so far:
Model | Explanatory | R-squared | Adj R-sq |
---|---|---|---|
1 | price + calories + fat + sodium |
0.50 | 0.21 |
2 | calories + fat |
0.37 | 0.23 |
3 | fat |
0.17 | 0.09 |
As we go through the models, R-squared goes dramatically down (it will go down because it always goes down when you take things out, but this seems too dramatic). Adjusted R-squared goes up when we take out price
and sodium
, but it too goes sharply down when we take out calories
, which doesn’t seem right.
There is no need to go any further than this, but if you want to take out fat
as well, leaving you with no explanatory variables at all, there are a couple of non-obvious ways to do it. One is to use update
:
.4 <- update(burgers.3, . ~ . -fat)
burgerssummary(burgers.4)
Call:
lm(formula = score ~ 1, data = burgers)
Residuals:
Min 1Q Median 3Q Max
-17.50 -10.50 -3.00 4.25 26.50
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 43.500 4.139 10.51 4.49e-07 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 14.34 on 11 degrees of freedom
This says that the predicted score
is 43.5, regardless of the values of anything else! There is no R-squared displayed, because that is zero for a model with no \(x\)-variables.
The other way is to find out that R understands 1
to mean a model with just an intercept:
.4a <- lm(score ~ 1, data = burgers)
burgerssummary(burgers.4a)
Call:
lm(formula = score ~ 1, data = burgers)
Residuals:
Min 1Q Median 3Q Max
-17.50 -10.50 -3.00 4.25 26.50
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 43.500 4.139 10.51 4.49e-07 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 14.34 on 11 degrees of freedom
Once again, the R-squared is zero.
- Another veggie burger (not in the original dataset) has the following values for the explanatory variables: price 91, calories 140, fat 5, sodium 450. What can you say about the likely score for a veggie burger with these values? Obtain a suitable interval, for each of your two models.
Solution
This is talking about the predicted response for an individual (this burger), not the mean response for all veggie burgers with those values for the explanatory variables, so it calls for a prediction interval in each case. This is the one that uses predict
, not the one that uses the marginaleffects
package (the one we did second in lecture).
The first step is to make a dataframe, by my tradition called new
, of values to predict for. Any way that produces you a one-row dataframe is good, for example:
<- tribble(
new ~price, ~calories, ~fat, ~sodium,
91, 140, 5, 450)
new
If you are stuck, type the values into a file (or even a spreadsheet) and read that in. But find a way.
For the model with all four explanatory variables:
predict(burgers.1, new, interval = "p")
fit lwr upr
1 54.72593 20.13798 89.31389
and for the model with only two:
predict(burgers.2, new, interval = "p")
fit lwr upr
1 49.71981 18.63013 80.80949
These intervals are distressingly wide, as is usually the way with prediction intervals (and we only have 12 observations to base the interval on). Also, it didn’t matter that in the second case, new
had some extra columns in it; these were just ignored.
Extra: these values are one SD above the mean in each case. How did I work them out? Like this:
%>%
burgers summarize(across(price:sodium, list(mean = \(x) mean(x), sd = \(x) sd(x)))) %>%
pivot_longer(everything(), names_to = c("variable", ".value"), names_sep = "_")
To work out summary statistics for a whole bunch of columns, use across
inside the summarize
. First is the columns you want to summarize, and then is how you want to summarize them. In this case, I wanted the mean and SD of each variable, two things, so I had to put them in a list
. Something possibly new here is that I “named” the elements of the list
(the mean =
and sd =
); what this does is to add mean
and sd
onto the name of each variable, so that I can tell which variable and which statistic I am looking at:
%>%
burgers summarize(across(price:sodium, list(mean = ~mean(.), sd = ~sd(.))))
This is all right, but I was hoping for something tidier. How about the names of the variables in the rows, and the names of the statistics in the columns? This is evidently some kind of pivot_longer
. It’s one of the fancy ones where the column names we have here encode two things, separated by an underscore. That means having two things in names_to
, and also having a names_sep
that says what those two things are separated by (an underscore).
To get the variable names in rows, we need to create a new column called something like variable
. This is the usual kind of pivot_longer
thing: put that first in names_to
, because the things that are going in variable
are the first part of the column names we have here. The second part of the column names we have so far, mean
or sd
, are going to make names of new columns, which is different from what would normally happen, which is this:
%>%
burgers summarize(across(price:sodium, list(mean = ~mean(.), sd = ~sd(.)))) %>%
pivot_longer(everything(), names_to = c("variable", "stat"), names_sep = "_", values_to = "value")
This has created a column called stat
, with the names of the statistics in it. This is all right, but is not as tidy as we1 would like.
To use the things in stat
as column names (and to fill the columns with the thing currently in value
), what you do is to replace the appropriate thing in names_to
with the special label .value
. When you do this, you can take out the values_to
, since pivot_longer
now knows where the values are going (into the new columns you are creating):
%>%
burgers summarize(across(price:sodium, list(mean = \(x) mean(x), sd = \(x) sd(x)))) %>%
pivot_longer(everything(), names_to = c("variable", ".value"), names_sep = "_")
Isn’t that pretty?
- Compare the lengths of your two intervals. Does it make sense that your shorter one should be shorter? Explain briefly.
Solution
After my long Extra, I need to display them again so that I can see them:
predict(burgers.1, new, interval = "p")
fit lwr upr
1 54.72593 20.13798 89.31389
predict(burgers.2, new, interval = "p")
fit lwr upr
1 49.71981 18.63013 80.80949
The first interval is about 69 points long and the second one is about 62 points long. Therefore the second interval is shorter.
Why is that? This is different from the other interval comparisons we have done, because this time we are comparing the same prediction from two different models.
Our previous work showed that the model burgers.2
was better than burgers.1
. This was because it had fewer explanatory variables in it, and we showed that the ones we removed from burgers.1
could safely be removed (the \(F\)-test in anova
). Or similar wording; you might have concluded that the extra explanatory variables in burgers.1
were not needed and could be taken out.
This is another reason for trying to find a good model: not only is a smaller model easier to explain, but it also gives better predictions, in the sense that the uncertainty around the prediction (as measured by the length of the interval) is smaller.
- Using our second model (the one with only
calories
andfat
in it), find a suitable interval for the mean score when (i) calories is 140 and fat is 5, (ii) calories is 120 and fat is 3. (You should have two intervals.)
Solution
This is the “mean of all possible scores” when the explanatory variables take the values shown, so it’s the confidence interval for the mean response rather than the prediction interval we had before. To start, make another new
with the two rows of values in it:
<- tribble(
new ~calories, ~fat,
140, 1,
120, 3
) new
Check. Then, predictions
(from the marginaleffects
package, which of course you remembered to install (once) and load first):
library(marginaleffects)
cbind(predictions(burgers.2, newdata = new)) %>%
select(calories, fat, estimate, conf.low, conf.high)
(if you use the wrong model, you’ll get an error, because the bigger model has some other things in it that we don’t have values for.)
- Explain briefly why the second interval is shorter than the first one. Make sure you justify your answer.
Solution
First, verify that the second interval really is shorter than the first one. (If, for some reason, it is not, then say that.) The first interval is of length about \(43 - (-5) = 48\), and the second one is of length about \(51-36 = 15\).
Aside: you might be thinking about working those out with R rather than by hand, and so you can:
cbind(predictions(burgers.2, newdata = new)) %>%
mutate(ci_length = conf.high - conf.low) %>%
select(calories, fat, conf.low, conf.high, ci_length)
Now we are in the familiar situation where we are comparing predictions for different values for the same model. So you might be suspecting that the second pair of values is closer to the mean (for the data as a whole) than the first pair is.2 But we should check that this is indeed the case.
Let’s look at the values we predicted for first:
new
To get the means, the easiest way is summary
, on the whole dataframe or the bit of it containing only calories
and fat
:
%>%
burgers select(calories, fat) %>%
summary()
calories fat
Min. : 80.0 Min. :0.00
1st Qu.: 97.5 1st Qu.:1.00
Median :115.0 Median :3.00
Mean :115.8 Mean :2.75
3rd Qu.:130.0 3rd Qu.:4.00
Max. :170.0 Max. :6.00
The values for both variables are both above their means, but the second values for both calories
and fat
are closer to their means.
There is a tidyverse way to do this, which uses across
again, but it’s a bit simpler than the other one I did (in an Extra above):
%>%
burgers summarize(across(c(calories, fat), \(x) mean(x)))
(“for each of calories
and fat
, work out the mean of it”).
Somehow, make the assertion that the second values for calories
and fat
are closer to their means than the first values of each of them, and then demonstrate that this is indeed true. Or, think to yourself “this probably depends on the means somehow”, find the means, and then say that the second values are closer to the means, so the prediction for them should be better (in the sense of having a shorter confidence interval).
Blood pressure
Twenty people with high blood pressure had various other measurements taken. The aim was to see which of these were associated with blood pressure, with the aim of understanding what causes high blood pressure. The variables observed were:
Pt
: patient number (ignore)BP
: (diastolic) blood pressure, in mmHgAge
in yearsWeight
in kgBSA
: body surface area, in m\(^2\)Dur
: length of time since diagnosis of high blood pressure, in yearsPulse
: pulse rate, in beats per minuteStress
: score on a questionnaire about stress levels (higher score is more stressed)
The data values, separated by tabs, are in https://ritsokiguess.site/datafiles/bloodpress.txt.
- Read in and display (some of) the data.
Solution
The data values are separated by tabs, so read_tsv
is what you need:
<- "https://ritsokiguess.site/datafiles/bloodpress.txt"
my_url <- read_tsv(my_url) bp
Rows: 20 Columns: 8
── Column specification ────────────────────────────────────────────────────────
Delimiter: "\t"
dbl (8): Pt, BP, Age, Weight, BSA, Dur, Pulse, Stress
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
bp
20 patients, and all the columns as listed. Remember that in R, uppercase and lowercase are different, so there is no problem in using (lowercase) bp
for the name of a dataframe that has a column called (uppercase) BP
in it. But if that confuses you, feel free to give your dataframe a more descriptive name like blood_pressure
or even something like health_stats
.
Extra: you can also use read_delim
for this, but you have to do it right. If you do it as read_delim(my_url, " ")
or similar, it won’t work, because the data values are not separated by single spaces:
<- read_delim(my_url, " ") bpxx
Rows: 20 Columns: 1
── Column specification ────────────────────────────────────────────────────────
Delimiter: " "
chr (1): Pt BP Age Weight BSA Dur Pulse Stress
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
bpxx
All of the data values have been smooshed together into a single column with a rather amusing name. The rather odd \t
is the key to understanding what has happened. The simplest way to make it work is to use read_delim
, but without saying what the delimiter character is:
<- read_delim(my_url) bpx
Rows: 20 Columns: 8
── Column specification ────────────────────────────────────────────────────────
Delimiter: "\t"
dbl (8): Pt, BP, Age, Weight, BSA, Dur, Pulse, Stress
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
bpx
If you look at the R Console now, you’ll see a line in the file-reading message that says
Delimiter: "\t"
What happened is that read_delim
successfully guessed what was separating the data values, by looking at the first few lines of the data file and seeing that there were a lot of tabs apparently separating data values. This behaviour is new in the latest version of readr
;3 if you use this idea, you should be able to explain why it worked. If you cannot, it looks like a lucky guess.
By this point, you might have guessed that the mysterious \t
is the way R represents a tab (you would be right), and that therefore you should be able to do this:4
<- read_delim(my_url, "\t") bpxxxx
Rows: 20 Columns: 8
── Column specification ────────────────────────────────────────────────────────
Delimiter: "\t"
dbl (8): Pt, BP, Age, Weight, BSA, Dur, Pulse, Stress
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
bpxxxx
and you would once again be right.
\(\blacksquare\)
- Make a plot of the blood pressure against each of the measured explanatory variables. Hint: use the idea from C32 of making a suitable long dataframe and using facets in your graph.
Solution
Use pivot_longer
with all the columns containing \(x\)-variables:
%>%
bp pivot_longer(Age:Stress, names_to = "xname", values_to = "x")
and then plot BP
against x
facetting by xname
:
%>%
bp pivot_longer(Age:Stress, names_to = "xname", values_to = "x") %>%
ggplot(aes(x = x, y = BP)) + geom_point() +
facet_wrap(~xname, scales = "free")
Use scales = "free"
to have each scatterplot fill its facet.
If you don’t get to this, you are faced with making six scatterplots one by one, which will be a lot of work in comparison (and it’s easy to mess up the copy/paste and miss one of them out).
\(\blacksquare\)
- Which explanatory variables seem to have a moderate or strong linear relationship with blood pressure?
Solution
I would say BSA
, Weight
and maybe Pulse
. For me, the Age
relationship is not quite strong enough, and there is basically no relationship with Dur
or Stress
.
I don’t mind precisely where you draw the line. You could include Pulse
and Age
or not, but I think you should definitely include BSA
and Weight
and definitely exclude Dur
and Stress
.
Extra: Stress is often considered to be a cause of high blood pressure, but it seems from this dataset that it is not.
\(\blacksquare\)
- Run a regression predicting blood pressure from
BSA
andWeight
, and display the output. Does the significance or lack of significance of each of your explanatory variables surprise you? Explain briefly.
Solution
.1 <- lm(BP ~ BSA + Weight, data = bp)
bpsummary(bp.1)
Call:
lm(formula = BP ~ BSA + Weight, data = bp)
Residuals:
Min 1Q Median 3Q Max
-1.8932 -1.1961 -0.4061 1.0764 4.7524
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 5.6534 9.3925 0.602 0.555
BSA 5.8313 6.0627 0.962 0.350
Weight 1.0387 0.1927 5.392 4.87e-05 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 1.744 on 17 degrees of freedom
Multiple R-squared: 0.9077, Adjusted R-squared: 0.8968
F-statistic: 83.54 on 2 and 17 DF, p-value: 1.607e-09
Both explanatory variables have a strong relationship with blood pressure according to the scatterplots, so we would expect them both to be significant. Weight
is, but BSA
is not. This I find surprising.
\(\blacksquare\)
- Explain briefly why it does in fact make sense that the regression results came out as they did. You may wish to draw another graph to support your explanation.
Solution
The way we learned this in C32 is that BSA
has nothing to add to a regression that also contains Weight
, when it comes to predicting blood pressure. That is to say, the way BSA
is related to blood pressure is similar to the way Weight
is .
To gain a bit more insight, we might be suffering from multicollinearity here, and the reason that BSA
is not significant as we expected could be that it and Weight
are related to each other. To find out about that, make a scatterplot of the two explanatory variables:5
ggplot(bp, aes(y = BSA, x = Weight)) + geom_point()
These are clearly related. To be more precise about “not needing BSA
as well”: if we know a person’s weight, we already know something about their body surface area (if the weight is bigger, so is their body surface area). So there is less information than you would otherwise expect in BSA
if you already know their weight.
This is a less dramatic example than the one in lecture (with the football punters), but the point is the same: if the \(x\)-variables are correlated, one or more of them could be less significant than you would expect, because the information it contains is mostly already contained in other \(x\)-variables.
Extra: another way of looking at all this is via pairwise correlations:
cor(bp)
Pt BP Age Weight BSA Dur
Pt 1.00000000 0.03113499 0.04269354 0.02485650 -0.03128800 0.1762455
BP 0.03113499 1.00000000 0.65909298 0.95006765 0.86587887 0.2928336
Age 0.04269354 0.65909298 1.00000000 0.40734926 0.37845460 0.3437921
Weight 0.02485650 0.95006765 0.40734926 1.00000000 0.87530481 0.2006496
BSA -0.03128800 0.86587887 0.37845460 0.87530481 1.00000000 0.1305400
Dur 0.17624551 0.29283363 0.34379206 0.20064959 0.13054001 1.0000000
Pulse 0.11228508 0.72141316 0.61876426 0.65933987 0.46481881 0.4015144
Stress 0.34315169 0.16390139 0.36822369 0.03435475 0.01844634 0.3116398
Pulse Stress
Pt 0.1122851 0.34315169
BP 0.7214132 0.16390139
Age 0.6187643 0.36822369
Weight 0.6593399 0.03435475
BSA 0.4648188 0.01844634
Dur 0.4015144 0.31163982
Pulse 1.0000000 0.50631008
Stress 0.5063101 1.00000000
The correlations with BP
(second column, or second row) are mostly as you would expect from the scatterplots: very high with Weight
and BSA
, moderately high for Age
and Pulse
, and low for the others. I usually find that the correlation suggests a stronger relationship than the scatterplot does.
The other thing is that the correlation between Weight
and BSA
is the highest of all the correlations between variables that are not the response. So if you know a person’s weight, you can already make a good guess at their body surface area, even without having the actual values. Hence, including BSA
in the regression when you already have Weight
is not very helpful.
The data for this problem came from here.
\(\blacksquare\)
Footnotes
Well, I.↩︎
There is one pair of values in each case, hence a singular verb.↩︎
The part of the
tidyverse
where the file-reading functions live.↩︎This looks like two characters, but it is actually only one. The backslash is called an “escape character” and doesn’t count as a character for these purposes. It’s used as a way to represent some characters that you cannot normally print. Another one is
\n
, which is the “newline” at the end of each line of a file. This is how R Studio or anything else knows to start a new line at this point and not somewhere else. Otherwise the file would be all one (very long) line on the screen. To convince yourself of that, run the codecat("Hello\nWorld")
and see what it displays.↩︎Either axis for either variable is good, since neither of these is a response.↩︎