首页 文章

R:对几个单独的图重新排序因子水平

提问于
浏览
6

我试图从相同的data.frame创建几个单独的图,每个图的y轴上有不同的因子级别顺序 . 每个图都应该按y顺序排列因子水平 .

我知道这可以为每个情节手动完成,但我正在寻找一种更有效和更优雅的方式,因为我将需要创建大量的情节 . 这不必包括使用facet_wrap,如果有另一种方式,可能有循环等?

library(ggplot2)
library(dplyr)
data("diamonds")

采用数据集和聚合两个因子水平(清晰度和切割):

means <- diamonds %>%
group_by(clarity, cut) %>%
summarise(carat = mean(carat))

在这里,我通过一个因素重新排序,但最终我想为每个图分别重新排序(通过降低清晰度的平均值) .

means$clarity <- reorder(means$clarity, means$carat, FUN = mean)

使用face_wrap创建单独的图 . 使用coord_flip可以更轻松地比较绘图 .

ggplot(means, aes(x = clarity, y = carat)) +
  geom_col() +
  facet_wrap(~cut, ncol = 1) +
  coord_flip()

您将看到,这会为每种类型的切割创建单独的图,但y轴上的因子水平的顺序对于每种情况都不正确 . 如何正确订购它们而不必为每种类型的切割手动完成?

1 回答

  • 6

    这可以通过使用两个函数在一个图中完成:

    reorder_within <- function(x, by, within, fun = mean, sep = "___", ...) {
      new_x <- paste(x, within, sep = sep)
      stats::reorder(new_x, by, FUN = fun)
    }
    
    
    scale_x_reordered <- function(..., sep = "___") {
      reg <- paste0(sep, ".+$")
      ggplot2::scale_x_discrete(labels = function(x) gsub(reg, "", x), ...)
    }
    

    可在github上获得dgrtwo/drlib

    ggplot(means, aes(x = reorder_within(clarity, carat, cut, mean), y = carat)) + 
      geom_col() + 
      scale_x_reordered() +
      facet_wrap(~cut,  scales = "free_y", ncol = 1) +
      coord_flip()
    

    enter image description here

相关问题