R/first.R

Defines functions `first.xts` `first.default` `first`

#
#   xts: eXtensible time-series
#
#   Copyright (C) 2008  Jeffrey A. Ryan jeff.a.ryan @ gmail.com
#
#   Contributions from Joshua M. Ulrich
#
#   This program is free software: you can redistribute it and/or modify
#   it under the terms of the GNU General Public License as published by
#   the Free Software Foundation, either version 2 of the License, or
#   (at your option) any later version.
#
#   This program is distributed in the hope that it will be useful,
#   but WITHOUT ANY WARRANTY; without even the implied warranty of
#   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
#   GNU General Public License for more details.
#
#   You should have received a copy of the GNU General Public License
#   along with this program.  If not, see <http://www.gnu.org/licenses/>.


#' Return First or Last n Elements of A Data Object
#' 
#' Generic functions to return the first or last elements or rows of a vector
#' or two-dimensional data object.
#' 
#' A more advanced subsetting is available for zoo objects with indexes
#' inheriting from POSIXt or Date classes.
#' 
#' Quickly and easily extract the first or last `n` observations of an object.
#' When `n` is a number, these functions are similar to [`head()`] and
#' [`tail()`], but only return the *first* or *last* observation by default.
#' 
#' `n` can be a character string if `x` is an xts object or coerceable to xts.
#' It must be of the form \sQuote{n period}, where 'n' is a numeric value
#' (1 if not provided) describing the number of periods to return. Valid
#' periods are: secs, seconds, mins, minutes, hours, days, weeks, months,
#' quarters, and years.
#'
#' The 'period' portion can be any frequency greater than or equal to the
#' frequency of the object's time index. For example, `first(x, "2 months")`
#' will return the first 2 months of data even if `x` is hourly frequency.
#' Attempts to set 'period' to a frequency less than the object's frequency
#' will throw an error.
#' 
#' `n` may be positive or negative, whether it's a number or character string.
#' When `n` is positive, the functions return the obvious result. For example,
#' `first(x, "1 month")` returns the first month's data. When `n` is negative,
#' all data *except* first month's is returned.
#' 
#' Requesting more data than is in `x` will throw a warning and simply return
#' `x`.
#' 
#' @param x An object.
#' @param n Number of observations to return.
#' @param keep Should removed values be kept as an attribute on the result?
#' @param \dots Arguments passed to other methods.
#' 
#' @return A subset of elements/rows of the original data.
#' 
#' @author Jeffrey A. Ryan
#' 
#' @keywords utilities
#' @examples
#' 
#' first(1:100)
#' last(1:100)
#' 
#' data(LakeHuron)
#' first(LakeHuron,10)
#' last(LakeHuron)
#' 
#' x <- xts(1:100, Sys.Date()+1:100)
#' first(x, 10)
#' first(x, '1 day')
#' first(x, '4 days')
#' first(x, 'month')
#' last(x, '2 months')
#' last(x, '6 weeks')
#' 
`first` <-
function(x,...)
{
  UseMethod("first")
}

#' @rdname first
`first.default` <-
function(x,n=1,keep=FALSE,...)
{
  if(length(x) == 0)
    return(x)
  if(is.character(n)) {
    xx <- try.xts(x, error=FALSE)
    if(is.xts(xx)) {
      xx <- first.xts(x, n=n, keep=keep, ...)
      return(reclass(xx))
    }
  }
  if(is.null(dim(x))) {
    if(n > 0) {
      sub <- seq_len(min(n, length(x)))
      xx <- x[sub]
      if(keep) xx <- structure(xx,keep=x[(-(-n)+1):NROW(x)])
      xx
    } else if(n < 0) {
      sub <- seq.int(to = length(x), length.out = max(length(x)-(-n), 0L))
      xx <- x[sub]
      if(keep) xx <- structure(xx,keep=x[1:(-n)])
      xx
    } else {
      xx <- x[0]
      if(keep) xx <- structure(xx,keep=x[0])
      xx
    }
  } else {
    if(n > 0) {
      sub <- seq_len(min(n, NROW(x)))
      xx <- x[sub,,drop=FALSE]
      if(keep) xx <- structure(xx,keep=x[(-(-n)+1):NROW(x),])
      xx
    } else if(n < 0) {
      sub <- seq.int(to = NROW(x), length.out = max(NROW(x)-(-n), 0L))
      xx <- x[sub,,drop=FALSE]
      if(keep) xx <- structure(xx,keep=x[1:(-n),])
      xx
    } else {
      xx <- x[0,,drop=FALSE]
      if(keep) xx <- structure(xx,keep=x[0,])
      xx
    }
  }
}

#' @rdname first
`first.xts` <-
function(x,n=1,keep=FALSE,...)
{
  if(length(x) == 0)
    return(x)
  if(is.character(n)) {
    # n period set
    np <- strsplit(n," ",fixed=TRUE)[[1]]
    if(length(np) > 2 || length(np) < 1)
      stop(paste("incorrectly specified",sQuote("n"),sep=" "))
    # series periodicity
    sp <- periodicity(x)
    # requested periodicity$units
    sp.units <- sp[["units"]]
    rpu <- np[length(np)]
    rpf <- ifelse(length(np) > 1, as.numeric(np[1]), 1)
    if(rpu == sp.units) {
      n <- rpf
    } else {
      # if singular - add an s to make it work
      if(substr(rpu,length(strsplit(rpu,'')[[1]]),length(strsplit(rpu,'')[[1]])) != 's')
        rpu <- paste(rpu,'s',sep='')
      u.list <- list(secs=4,seconds=4,mins=3,minutes=3,hours=2,days=1,
                     weeks=1,months=1,quarters=1,years=1)
      dt.options <- c('seconds','secs','minutes','mins','hours','days',
                      'weeks','months','quarters','years')
      if(!rpu %in% dt.options)
        stop(paste("n must be numeric or use",paste(dt.options,collapse=',')))
      dt <- dt.options[pmatch(rpu,dt.options)]
      if(u.list[[dt]] > u.list[[sp.units]]) {
        #  req is for higher freq data period e.g. 100 mins of daily data
        stop(paste("At present, without some sort of magic, it isn't possible",
             "to resolve",rpu,"from",sp$scale,"data"))
      }
      ep <- endpoints(x,dt)
      if(rpf > length(ep)-1) {
        rpf <- length(ep)-1
        warning("requested length is greater than original")
      }
      if(rpf > 0) {
        n <- ep[rpf+1]
        if(is.null(dim(x))) {
          xx <- x[1:n]
        } else {
          xx <- x[1:n,,drop=FALSE]
        }
        if(keep) xx <- structure(xx,keep=x[(ep[-(-rpf)+1]+1):NROW(x)])
        return(xx)
      } else if(rpf < 0) {
        n <- ep[-rpf+1]+1
        if(is.null(dim(x))) {
          xx <- x[n:NROW(x)]
        } else {
          xx <- x[n:NROW(x),,drop=FALSE]
        }
        if(keep) xx <- structure(xx,keep=x[1:(ep[-rpf+1])])
        return(xx)
      } else {
        if(is.null(dim(x))) {
          xx <- x[0]
        } else {
          xx <- x[0,,drop=FALSE]
        }
        if(keep) xx <- structure(xx,keep=x[0])
        return(xx)
      }
    }
  }
  if(length(n) != 1) stop("n must be of length 1")
  if(n > 0) {
    n <- min(n, NROW(x))
    if(is.null(dim(x))) {
      xx <- x[1:n]
    } else {
      xx <- x[1:n,,drop=FALSE]
    }
    if(keep) xx <- structure(xx,keep=x[(-(-n)+1):NROW(x)])
    xx
  } else if(n < 0) {
    if(abs(n) >= NROW(x))
      return(x[0])
    if(is.null(dim(x))) {
      xx <- x[(-n+1):NROW(x)]
    } else {
      xx <- x[(-n+1):NROW(x),,drop=FALSE]
    }
    if(keep) xx <- structure(xx,keep=x[1:(-n)])
    xx
  } else {
    if(is.null(dim(x))) {
      xx <- x[0]
    } else {
      xx <- x[0,,drop=FALSE]
    }
    if(keep) xx <- structure(xx,keep=x[0])
    xx
  }
}

Try the xts package in your browser

Any scripts or data that you put into this service are public.

xts documentation built on June 22, 2024, 9:56 a.m.