798

我有一个图,其中 x 轴是一个标签很长的因素。虽然可能不是一个理想的可视化,但现在我想简单地将这些标签旋转为垂直。我已经用下面的代码弄清楚了这部分,但正如你所看到的,标签并不完全可见。

data(diamonds)
diamonds$cut <- paste("Super Dee-Duper",as.character(diamonds$cut))
q <- qplot(cut,carat,data=diamonds,geom="boxplot")
q + opts(axis.text.x=theme_text(angle=-90))

在此处输入图像描述

4

8 回答 8

1325

将最后一行更改为

q + theme(axis.text.x = element_text(angle = 90, vjust = 0.5, hjust=1))

默认情况下,轴在文本的中心对齐,即使在旋转时也是如此。当您旋转 +/- 90 度时,您通常希望它在边缘对齐:

替代文字

上图来自这篇博文。

于 2009-08-25T22:36:04.047 回答
118

利用coord_flip()

data(diamonds)
diamonds$cut <- paste("Super Dee-Duper",as.character(diamonds$cut))

qplot(cut, carat, data = diamonds, geom = "boxplot") +
  coord_flip()

在此处输入图像描述


添加str_wrap()

# wrap text to no more than 15 spaces
library(stringr)
diamonds$cut2 <- str_wrap(diamonds$cut, width = 15)
qplot(cut2, carat, data = diamonds, geom = "boxplot") +
  coord_flip()

在此处输入图像描述


R for Data Science的第 3.9 章中,Wickham 和 Grolemund 谈到了这个确切的问题:

coord_flip()切换 x 和 y 轴。如果您想要水平箱线图,这很有用(例如)。它对于长标签也很有用:很难在不重叠 x 轴的情况下让它们适合。

于 2017-08-01T02:29:00.983 回答
98

To make the text on the tick labels fully visible and read in the same direction as the y-axis label, change the last line to

q + theme(axis.text.x=element_text(angle=90, hjust=1))
于 2010-12-20T04:24:57.717 回答
78

ggplot 3.3.0通过提供guide_axis(angle = 90)(作为guide参数scale_..或作为x参数guides)来解决这个问题:

library(ggplot2)
data(diamonds)
diamonds$cut <- paste("Super Dee-Duper", as.character(diamonds$cut))

ggplot(diamonds, aes(cut, carat)) +
  geom_boxplot() +
  scale_x_discrete(guide = guide_axis(angle = 90)) +
  # ... or, equivalently:
  # guides(x =  guide_axis(angle = 90)) +
  NULL

论点的文档中angle

与在 theme() / element_text() 中设置角度相比,这还使用了一些启发式方法来自动选择您可能想要的 hjust 和 vjust。


或者,它还提供 guide_axis(n.dodge = 2)(作为 的guide参数scale_..或作为 的x参数guides)通过垂直躲避标签来克服过度绘图问题。在这种情况下它工作得很好:

library(ggplot2)
data(diamonds)
diamonds$cut <- paste("Super Dee-Duper",as.character(diamonds$cut))

ggplot(diamonds, aes(cut, carat)) + 
  geom_boxplot() +
  scale_x_discrete(guide = guide_axis(n.dodge = 2)) +
  NULL

于 2020-03-12T08:58:37.770 回答
31

我想提供一个替代解决方案,因为引入了画布旋转功能,所以在最新版本的ggtern中需要一个类似于我即将提出的强大解决方案。

基本上,您需要使用三角函数确定相对位置,方法是构建一个返回element_text对象、给定角度(即度数)和定位(即 x、y、顶部或右侧之一)信息的函数。

#Load Required Libraries
library(ggplot2)
library(gridExtra)

#Build Function to Return Element Text Object
rotatedAxisElementText = function(angle,position='x'){
  angle     = angle[1]; 
  position  = position[1]
  positions = list(x=0,y=90,top=180,right=270)
  if(!position %in% names(positions))
    stop(sprintf("'position' must be one of [%s]",paste(names(positions),collapse=", ")),call.=FALSE)
  if(!is.numeric(angle))
    stop("'angle' must be numeric",call.=FALSE)
  rads  = (angle - positions[[ position ]])*pi/180
  hjust = 0.5*(1 - sin(rads))
  vjust = 0.5*(1 + cos(rads))
  element_text(angle=angle,vjust=vjust,hjust=hjust)
}

坦率地说,在我看来,我认为应该ggplot2hjustandvjust参数提供一个“自动”选项,无论如何,在指定角度时,让我们演示一下上面的工作原理。

#Demonstrate Usage for a Variety of Rotations
df    = data.frame(x=0.5,y=0.5)
plots = lapply(seq(0,90,length.out=4),function(a){
  ggplot(df,aes(x,y)) + 
    geom_point() + 
    theme(axis.text.x = rotatedAxisElementText(a,'x'),
          axis.text.y = rotatedAxisElementText(a,'y')) +
    labs(title = sprintf("Rotated %s",a))
})
grid.arrange(grobs=plots)

产生以下内容:

例子

于 2016-05-02T09:32:35.303 回答
11

The ggpubr package offers a shortcut that does the right thing by default (right align text, middle align text box to tick):

library(ggplot2)
diamonds$cut <- paste("Super Dee-Duper", as.character(diamonds$cut))
q <- qplot(cut, carat, data = diamonds, geom = "boxplot")
q + ggpubr::rotate_x_text()

Created on 2018-11-06 by the reprex package (v0.2.1)

Found with a GitHub search for the relevant argument names: https://github.com/search?l=R&q=element_text+angle+90+vjust+org%3Acran&type=Code

于 2018-11-06T22:23:20.540 回答
6

已过时 - 请参阅此答案以获取更简单的方法


要在没有其他依赖项的情况下获得可读的 x 刻度标签,您需要使用:

  ... +
  theme(axis.text.x = element_text(angle = 90, hjust = 1, vjust = 0.5)) +
  ...

这会将刻度标签逆时针旋转 90°,并将它们的末端 ( hjust = 1) 垂直对齐,并将它们的中心与相应的刻度标记 ( vjust = 0.5) 水平对齐。

完整示例:

library(ggplot2)
data(diamonds)
diamonds$cut <- paste("Super Dee-Duper",as.character(diamonds$cut))
q <- qplot(cut,carat,data=diamonds,geom="boxplot")
q + theme(axis.text.x = element_text(angle = 90, hjust = 1, vjust = 0.5))


请注意,垂直/水平对齐参数vjust/ hjustofelement_text与文本相关。因此,vjust负责水平对齐。

没有vjust = 0.5它会是这样的:

q + theme(axis.text.x = element_text(angle = 90, hjust = 1))

没有hjust = 1它会是这样的:

q + theme(axis.text.x = element_text(angle = 90, vjust = 0.5))

如果出于某种(有线)原因您想将刻度标签顺时针旋转 90°(以便可以从左侧读取它们),您需要使用:q + theme(axis.text.x = element_text(angle = -90, vjust = 0.5, hjust = -1)).

所有这些已经在这个答案的评论中讨论过,但我经常回到这个问题,我想要一个答案,我可以从中复制而不阅读评论。

于 2020-03-11T16:25:04.913 回答
0

另一种方法coord_flip()是使用ggstance包。优点是它可以更轻松地将图形与其他图形类型结合起来,而且您可能更重要的是,可以为您的坐标系设置固定比例

library(ggplot2)
library(ggstance)

diamonds$cut <- paste("Super Dee-Duper", as.character(diamonds$cut))

ggplot(data=diamonds, aes(carat, cut)) + geom_boxploth()

reprex 包于 2020-03-11 创建(v0.3.0)

于 2020-03-11T12:34:10.003 回答