Please include the context of the discussion in your responses. See inline below.

On 1/24/2012 11:33 PM, Ajay Askoolum wrote:
Thanks you,

I can get the length of aa with length(unlist(aa)). If aa has 4
dimensions, I imagine I'd need to do

max(sapply(aa,sapply,sapply,length)

Correct.

How can I do this in a generic way? That is in a loop. I am clear
about the exit condition for the loop.

By generic, I assume that you mean without knowing the depth of the lists (that is, "dimensions") to begin with?

d<-1


start loop

if d = length(unlist(aa)) then exit loop

Note that length(unlist(aa)) will only equal the product of the dimensions if the data is "regular." That is, there is an entry for every combination of indices (within the dimensions).

else
     d<-d *<expr>


How do I construct<expr>  such that it does

d<- d * length(aa)                      # first pass

d<- d * max(sapply(aa, length))         # second pass

d<- d * max(sapply(aa, sapply, length)) # third pass


# ? #                                    # fourth path etc



(Apologies for the pseudo code describing the loop; I am not near a
machine with R)

I don't really understand this.

One way I can thing of is to create a loop counter variable, say
lc<-1 and to increment it within the loop and then use a switch
statement to execute the appropriate expression. This sems like a
kludge to me.

Is there a neater way?

If you are trying to get back a vector of the dimensions, then this would work:

dimRecursiveList <- function(l) {
        if (class(l) == "list") {
                c(length(l), dimRecursiveList(l[[1]]))
        } else {
                NULL
        }
}

From previous context:

aa <- list(list(list(37531.52, 62787.32, 5503.184, 33832.8),
                list(20469.60, 27057.27, 51160.25, 45165.24),
                list(957.932,  21902.94, 37531.52, 62787.32)),
           list(list(5503.184, 33832.8,  20469.6,  27057.27),
                list(51160.25, 45165.24, 957.932,  21902.94),
                list(37531.52, 62787.32, 5503.184, 33832.8)))

Which then gives

> dimRecursiveList(aa)
[1] 2 3 4

In this case, the data is regular, so

> Reduce(`*`, dimRecursiveList(aa)) == length(unlist(aa))
[1] TRUE

If the data are not regular, and you want the dimension to be the largest, then it is more complicated (due to bookkeeping)

dimRecursiveListIrregular <- function(l) {
  if (class(l) == "list") {
    dims <- sapply(l, dimRecursiveListIrregular)
    if (is.null(dim(dims))) {
      if (all(is.na(dims))) {
        length(l)
      } else {
        c(length(l), max(dims, na.rm=TRUE))
      }
    } else {
      c(length(l), apply(dims, 1, max, na.rm=TRUE))
    }
  } else {
    NA
  }
}

If the data are regular, then it is better to convert this to an array. This function will do it for arbitrary depth

RecursiveListToArray <- function(l) {
  if(class(l) == "list") {
    laply(l, RecursiveListToArray)
  } else {
    l
  }
}

> aaa <- RecursiveListToArray(aa)
> dim(aaa)
[1] 2 3 4

--
Brian S. Diggs, PhD
Senior Research Associate, Department of Surgery
Oregon Health & Science University

______________________________________________
R-help@r-project.org mailing list
https://stat.ethz.ch/mailman/listinfo/r-help
PLEASE do read the posting guide http://www.R-project.org/posting-guide.html
and provide commented, minimal, self-contained, reproducible code.

Reply via email to