简体   繁体   中英

Plot two time series with different y-axes: one as a dot plot (or a bar plot) and the other as a line

I have two time series of data, each with a different range of values. I would like to plot one as a dotplot and the other as a line over the dotplot. (I would settle for a decent-looking barplot and a line over the barplot, but my preference is a dotplot.)

#make some data
require(lubridate)
require(ggplot)

x1 <- sample(1990:2010, 10, replace=F)
x1 <- paste(x1, "-01-01", sep="")
x1 <- as.Date(x1)
y1 <- sample(1:10, 10, replace=T)
data1 <- cbind.data.frame(x1, y1)

year <- sample(1990:2010, 10, replace=F)
month <- sample(1:9, 10, replace=T)
day <- sample(1:28, 10, replace=T)

x2 <- paste(year, month, day, sep="-")
x2 <- as.Date(x2)
y2 <- sample(100:200, 10, replace=T)
data2 <- cbind.data.frame(x2, y2)
data2 <- data2[with(data2, order(x2)), ]

# frequency data for dot plot
x3 <- sample(1990:2010, 25, replace=T)
data2 <- as.data.frame(x3)

I can make a dotplot or barplot with one data set in ggplot:

ggplot() + geom_dotplot(data=data2, aes(x=x3))

散点图

ggplot() + geom_bar(data=data, aes(x=x1, y=y1), stat="identity")

barplot

But I can't overlay the second data set because ggplot doesn't permit a second y-axis.

I can't figure out how to plot a time series using barplot().

I can plot the first set of data as an "h" type plot, using plot(), and add the second set of data as a line, but I can't make the bars any thicker because each one corresponds to a single day over a stretch of many years, and I think it's ugly.

plot(data$x1, data$y1, type="h")
par(new = T)
plot(data2$x2, data2$y2, type="l", axes=F, xlab=NA, ylab=NA)
axis(side=4)

plotplot

Any ideas? My only remaining idea is to make two separate plots and overlay them in a graphics program. :/

An easy workaround is to follow your base plotting instinct and beef up lwd for type='h' . Be sure to set lend=1 to prevent rounded lines:

par(mar=c(5, 4, 2, 5) + 0.1)
plot(data1, type='h', lwd=20, lend=1, las=1, xlab='Date', col='gray',
     xlim=range(data1$x1, data2$x2))
par(new=TRUE)
plot(data2, axes=FALSE, type='o', pch=20, xlab='', ylab='', lwd=2,
     xlim=range(data1$x1, data2$x2))
axis(4, las=1)
mtext('y2', 4, 3.5)

在此处输入图片说明

I removed the original answer.


To answer your question about making a dot plot, you can rearrange your data so that you can use the base plotting function. An example:

use the chron package for plotting:

library(chron)

dummy data:

count.data <- data.frame("dates" = c("1/27/2000", "3/27/2000", "6/27/2000", "10/27/2000"), "counts" = c(3, 10, 5, 1), stringsAsFactors = F) 

replicate the dates in a list:

rep.dates <- sapply(1:nrow(count.data), function(x) rep(count.data$dates[x], count.data$counts[x]))

turn the counts into a sequence:

seq.counts <- sapply(1:nrow(count.data), function(x) seq(1, count.data$counts[x], 1))

plot it up:

plot(as.chron(rep.dates[[1]]), seq.counts[[1]], xlim = c(as.chron("1/1/2000"), as.chron("12/31/2000")),
 ylim = c(0, 20), pch = 20, cex = 2)
for(i in 2:length(rep.dates)){
   points(as.chron(rep.dates[[i]]), seq.counts[[i]], pch = 20, cex = 2)
}

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM