Графики рядом с ggplot2

r visualization ggplot2

297170 просмотра

12 ответа

Я хотел бы разместить два графика рядом, используя пакет ggplot2 , то есть сделать эквивалент par(mfrow=c(1,2)).

Например, я хотел бы, чтобы следующие два графика показывали бок о бок с одинаковым масштабом.

x <- rnorm(100)
eps <- rnorm(100,0,.2)
qplot(x,3*x+eps)
qplot(x,2*x+eps)

Нужно ли помещать их в один и тот же data.frame?

qplot(displ, hwy, data=mpg, facets = . ~ year) + geom_smooth()
Автор: Christopher DuBois Источник Размещён: 24.07.2019 11:47

Ответы (12)


439 плюса

Решение

Любые ggplots рядом (или n графиков на сетке)

Функция grid.arrange()в gridExtraпакете объединит несколько графиков; это то, как вы положили два рядом.

require(gridExtra)
plot1 <- qplot(1)
plot2 <- qplot(1)
grid.arrange(plot1, plot2, ncol=2)

Это полезно, когда два графика не основаны на одних и тех же данных, например, если вы хотите построить разные переменные без использования reshape ().

Это будет выводить результат как побочный эффект. Чтобы напечатать побочный эффект в файл, укажите драйвер устройства (например pdf, pngи т. Д.), Например

pdf("foo.pdf")
grid.arrange(plot1, plot2)
dev.off()

или использовать arrangeGrob()в сочетании с ggsave(),

ggsave("foo.pdf", arrangeGrob(plot1, plot2))

Это эквивалентно использованию двух разных графиков par(mfrow = c(1,2)). Это не только экономит время на организацию данных, это необходимо, когда вы хотите два разных графика.


Приложение: Использование граней

Аспекты полезны для создания похожих сюжетов для разных групп. Это указано ниже во многих ответах ниже, но я хочу выделить этот подход примерами, эквивалентными приведенным выше графикам.

mydata <- data.frame(myGroup = c('a', 'b'), myX = c(1,1))

qplot(data = mydata, 
    x = myX, 
    facets = ~myGroup)

ggplot(data = mydata) + 
    geom_bar(aes(myX)) + 
    facet_wrap(~myGroup)

Обновить

plot_gridфункция в cowplotстоит проверить в качестве альтернативы grid.arrange. Смотрите ответ по @ claus-wilke ниже и эту виньетку для эквивалентного подхода; но функция позволяет более точно контролировать местоположение и размер графика, основываясь на этой виньетке .

Автор: David LeBauer Размещён: 14.10.2010 04:52

126 плюса

Один из недостатков решений, основанных на grid.arrangeтом, что они затрудняют маркировку графиков буквами (A, B и т. Д.), Как того требует большинство журналов.

Я написал пакет cowplot для решения этой (и нескольких других) проблем, в частности функции plot_grid():

library(cowplot)

iris1 <- ggplot(iris, aes(x = Species, y = Sepal.Length)) +
  geom_boxplot() + theme_bw()

iris2 <- ggplot(iris, aes(x = Sepal.Length, fill = Species)) +
  geom_density(alpha = 0.7) + theme_bw() +
  theme(legend.position = c(0.8, 0.8))

plot_grid(iris1, iris2, labels = "AUTO")

введите описание изображения здесь

plot_grid()Возвращаемый объект - это другой объект ggplot2, и вы можете сохранить его ggsave()как обычно:

p <- plot_grid(iris1, iris2, labels = "AUTO")
ggsave("plot.pdf", p)

В качестве альтернативы вы можете использовать функцию cowplot save_plot(), которая представляет собой тонкую обертку вокруг, ggsave()которая позволяет легко получить правильные размеры для комбинированных участков, например:

p <- plot_grid(iris1, iris2, labels = "AUTO")
save_plot("plot.pdf", p, ncol = 2)

( ncol = 2Аргумент говорит, save_plot()что есть два графика рядом, и save_plot()делает сохраненное изображение в два раза шире.)

Более подробное описание порядка расположения графиков в сетке см. В этой виньетке. Существует также виньетка, объясняющая, как создавать сюжеты с общей легендой.

Частая путаница заключается в том, что пакет cowplot меняет тему ggplot2 по умолчанию. Пакет ведет себя так, потому что изначально был написан для внутреннего использования в лаборатории, и мы никогда не используем тему по умолчанию. Если это вызывает проблемы, вы можете использовать один из следующих трех подходов, чтобы обойти их:

1. Установите тему вручную для каждого сюжета. Я думаю, что это хорошая практика - всегда указывать определенную тему для каждого сюжета, как я это делал + theme_bw()в примере выше. Если вы указываете конкретную тему, тема по умолчанию не имеет значения.

2. Верните тему по умолчанию обратно к ggplot2 default. Вы можете сделать это с помощью одной строки кода:

theme_set(theme_gray())

3. Вызовите функции cowplot, не подключая пакет. Вы также не можете вызывать library(cowplot)или require(cowplot)и вместо этого вызывать функции cowplot, предварительно добавив cowplot::. Например, приведенный выше пример использования темы по умолчанию ggplot2 будет выглядеть так:

## Commented out, we don't call this
# library(cowplot)

iris1 <- ggplot(iris, aes(x = Species, y = Sepal.Length)) +
  geom_boxplot()

iris2 <- ggplot(iris, aes(x = Sepal.Length, fill = Species)) +
  geom_density(alpha = 0.7) +
  theme(legend.position = c(0.8, 0.8))

cowplot::plot_grid(iris1, iris2, labels = "AUTO")

введите описание изображения здесь

Обновление: Начиная с ggplot2 3.0.0, участки могут быть помечены напрямую, см., Например, здесь.

Автор: Claus Wilke Размещён: 04.07.2015 05:53

45 плюса

Вы можете использовать следующую multiplotфункцию из книги рецептов Винстона Чанга R

multiplot(plot1, plot2, cols=2)

multiplot <- function(..., plotlist=NULL, cols) {
    require(grid)

    # Make a list from the ... arguments and plotlist
    plots <- c(list(...), plotlist)

    numPlots = length(plots)

    # Make the panel
    plotCols = cols                          # Number of columns of plots
    plotRows = ceiling(numPlots/plotCols) # Number of rows needed, calculated from # of cols

    # Set up the page
    grid.newpage()
    pushViewport(viewport(layout = grid.layout(plotRows, plotCols)))
    vplayout <- function(x, y)
        viewport(layout.pos.row = x, layout.pos.col = y)

    # Make each plot, in the correct location
    for (i in 1:numPlots) {
        curRow = ceiling(i/plotCols)
        curCol = (i-1) %% plotCols + 1
        print(plots[[i]], vp = vplayout(curRow, curCol ))
    }

}
Автор: David LeBauer Размещён: 05.12.2011 09:17

18 плюса

Используя пакет заплаток , вы можете просто использовать +оператор:

# install.packages("devtools")
devtools::install_github("thomasp85/patchwork")

library(ggplot2)
p1 <- ggplot(mtcars) + geom_point(aes(mpg, disp))
p2 <- ggplot(mtcars) + geom_boxplot(aes(gear, disp, group = gear))

library(patchwork)
p1 + p2

пэчворк

Автор: Deena Размещён: 23.01.2018 10:24

17 плюса

Да, я думаю, что вы должны правильно организовать свои данные. Одним из способов было бы это:

X <- data.frame(x=rep(x,2),
                y=c(3*x+eps, 2*x+eps),
                case=rep(c("first","second"), each=100))

qplot(x, y, data=X, facets = . ~ case) + geom_smooth()

Я уверен, что есть лучшие трюки в plyr или rehape - я все еще не совсем в курсе всех этих мощных пакетов Хэдли.

Автор: Dirk Eddelbuettel Размещён: 08.08.2009 06:56

15 плюса

Используя пакет изменения формы, вы можете сделать что-то вроде этого.

library(ggplot2)
wide <- data.frame(x = rnorm(100), eps = rnorm(100, 0, .2))
wide$first <- with(wide, 3 * x + eps)
wide$second <- with(wide, 2 * x + eps)
long <- melt(wide, id.vars = c("x", "eps"))
ggplot(long, aes(x = x, y = value)) + geom_smooth() + geom_point() + facet_grid(.~ variable)
Автор: Thierry Размещён: 10.08.2009 09:52

10 плюса

Обновление: этот ответ очень старый. gridExtra::grid.arrange()сейчас рекомендуемый подход. Я оставляю это здесь на случай, если это будет полезно.


Стивен Тернер опубликовал эту arrange()функцию в блоге Getting Genetics Done (инструкции см. В посте)

vp.layout <- function(x, y) viewport(layout.pos.row=x, layout.pos.col=y)
arrange <- function(..., nrow=NULL, ncol=NULL, as.table=FALSE) {
 dots <- list(...)
 n <- length(dots)
 if(is.null(nrow) & is.null(ncol)) { nrow = floor(n/2) ; ncol = ceiling(n/nrow)}
 if(is.null(nrow)) { nrow = ceiling(n/ncol)}
 if(is.null(ncol)) { ncol = ceiling(n/nrow)}
        ## NOTE see n2mfrow in grDevices for possible alternative
grid.newpage()
pushViewport(viewport(layout=grid.layout(nrow,ncol) ) )
 ii.p <- 1
 for(ii.row in seq(1, nrow)){
 ii.table.row <- ii.row 
 if(as.table) {ii.table.row <- nrow - ii.table.row + 1}
  for(ii.col in seq(1, ncol)){
   ii.table <- ii.p
   if(ii.p > n) break
   print(dots[[ii.table]], vp=vp.layout(ii.table.row, ii.col))
   ii.p <- ii.p + 1
  }
 }
}
Автор: Jeromy Anglim Размещён: 28.04.2010 04:49

9 плюса

ggplot2 основан на сеточной графике, которая предоставляет другую систему для размещения графиков на странице. Команда par(mfrow...)не имеет прямого эквивалента, так как объекты сетки (называемые гробами ) не обязательно рисуются сразу, но могут быть сохранены и обработаны как обычные объекты R перед преобразованием в графический вывод. Это обеспечивает большую гибкость, чем рисование этой ныне модели базовой графики, но стратегия обязательно немного другая.

Я написал, grid.arrange()чтобы обеспечить простой интерфейс как можно ближе к par(mfrow). В простейшем виде код будет выглядеть так:

library(ggplot2)
x <- rnorm(100)
eps <- rnorm(100,0,.2)
p1 <- qplot(x,3*x+eps)
p2 <- qplot(x,2*x+eps)

library(gridExtra)
grid.arrange(p1, p2, ncol = 2)

введите описание изображения здесь

Дополнительные параметры подробно описаны в этой виньетке .

Одна распространенная жалоба заключается в том, что графики не обязательно выровнены, например, когда они имеют метки осей разного размера, но это grid.arrangeсделано по замыслу: не предпринимает попыток обращаться к объектам ggplot2 в особом случае и обрабатывает их в равной степени с другими гробами (например, решетчатые графики). ). Это просто помещает гробы в прямоугольную схему.

Для особого случая объектов ggplot2 я написал другую функцию ggarrangeс похожим интерфейсом, которая пытается выровнять панели графиков (включая фасетированные графики) и пытается соблюдать пропорции, когда они определены пользователем.

library(egg)
ggarrange(p1, p2, ncol = 2)

Обе функции совместимы с ggsave(). Для общего обзора различных вариантов и некоторого исторического контекста эта виньетка предлагает дополнительную информацию .

Автор: baptiste Размещён: 02.12.2017 04:20

8 плюса

Также стоит упомянуть пакет multipanelfigure . Смотрите также этот ответ .

library(ggplot2)
theme_set(theme_bw())

q1 <- ggplot(mtcars) + geom_point(aes(mpg, disp))
q2 <- ggplot(mtcars) + geom_boxplot(aes(gear, disp, group = gear))
q3 <- ggplot(mtcars) + geom_smooth(aes(disp, qsec))
q4 <- ggplot(mtcars) + geom_bar(aes(carb))

library(magrittr)
library(multipanelfigure)
figure1 <- multi_panel_figure(columns = 2, rows = 2, panel_label_type = "none")
# show the layout
figure1

figure1 %<>%
  fill_panel(q1, column = 1, row = 1) %<>%
  fill_panel(q2, column = 2, row = 1) %<>%
  fill_panel(q3, column = 1, row = 2) %<>%
  fill_panel(q4, column = 2, row = 2)
figure1

# complex layout
figure2 <- multi_panel_figure(columns = 3, rows = 3, panel_label_type = "upper-roman")
figure2

figure2 %<>%
  fill_panel(q1, column = 1:2, row = 1) %<>%
  fill_panel(q2, column = 3, row = 1) %<>%
  fill_panel(q3, column = 1, row = 2) %<>%
  fill_panel(q4, column = 2:3, row = 2:3)
figure2

Создано в 2018-07-06 пакетом представлением (v0.2.0.9000).

Автор: Tung Размещён: 07.07.2018 06:29

4 плюса

Использование tidyverse:

x <- rnorm(100)
eps <- rnorm(100,0,.2)
df <- data.frame(x, eps) %>% 
  mutate(p1 = 3*x+eps, p2 = 2*x+eps) %>% 
  tidyr::gather("plot", "value", 3:4) %>% 
  ggplot(aes(x = x , y = value)) + 
    geom_point() + 
    geom_smooth() + 
    facet_wrap(~plot, ncol =2)

df

введите описание изображения здесь

Автор: aelwan Размещён: 30.01.2017 05:09

1 плюс

Приведенные выше решения могут быть неэффективными, если вы хотите построить несколько графиков ggplot с использованием цикла (например, как здесь предлагается: создание нескольких графиков в ggplot с различными значениями оси Y с использованием цикла ), что является желательным шагом для анализа неизвестного ( или большие) наборы данных (например, когда вы хотите построить график всех переменных в наборе данных).

Приведенный ниже код показывает, как это сделать, используя упомянутый выше метод multiplot (), источник которого находится здесь: http://www.cookbook-r.com/Graphs/Multiple_graphs_on_one_page_(ggplot2) :

plotAllCounts <- function (dt){   
  plots <- list();
  for(i in 1:ncol(dt)) {
    strX = names(dt)[i]
    print(sprintf("%i: strX = %s", i, strX))
    plots[[i]] <- ggplot(dt) + xlab(strX) +
      geom_point(aes_string(strX),stat="count")
  }

  columnsToPlot <- floor(sqrt(ncol(dt)))
  multiplot(plotlist = plots, cols = columnsToPlot)
}

Теперь запустите функцию - чтобы получить Count для всех переменных, напечатанных с использованием ggplot на одной странице

dt = ggplot2::diamonds
plotAllCounts(dt)

Следует отметить следующее:
использование aes(get(strX)), которое вы обычно используете в циклах при работе ggplot, в приведенном выше коде вместо того, aes_string(strX)чтобы НЕ рисовать желаемые графики. Вместо этого он построит последний сюжет много раз. Я не выяснил, почему - это, возможно, придется сделать aesи aes_stringвызвали ggplot.

В противном случае, надеюсь, вы найдете функцию полезной.

Автор: IVIM Размещён: 19.04.2017 09:34

-1 плюса

cowplotПакет дает вам хороший способ сделать это, таким образом , который соответствует публикации.

x <- rnorm(100)
eps <- rnorm(100,0,.2)
A = qplot(x,3*x+eps, geom = c("point", "smooth"))+theme_gray()
B = qplot(x,2*x+eps, geom = c("point", "smooth"))+theme_gray()
cowplot::plot_grid(A, B, labels = c("A", "B"), align = "v")

введите описание изображения здесь

Автор: tim Размещён: 29.08.2017 03:40
Вопросы из категории :
32x32