[Rd] Unexpected failure when calling new() with unnamed arg and

Hervé Pagès hpages at fredhutch.org
Thu May 14 00:13:27 CEST 2015


Thanks Martin for looking into this.  H.

On 05/13/2015 03:57 AM, Martin Maechler wrote:
>>>>>> Hervé Pagès <hpages at fredhutch.org>
>>>>>>      on Tue, 12 May 2015 15:18:42 -0700 writes:
>
>> Hi,
>
>> The man page for new() suggests that if 'a' is an object with slots
>> "slot1" and "slot2" and C is a class that extends the class of 'a',
>> then the 2 following calls should be equivalent:
>
>>     new("C", a, ...)
>>     new("C", slot1=a at slot1, slot2=a at slot2, ...)
>
>> This is generally the case but I just ran into a situation where it's
>> not. In the following example the former fails while the latter works:
>
>>     setClass("A", representation(slot1="numeric", slot2="logical"))
>>     setClass("B", contains="A", representation(design="formula"))
>>     setClass("C", contains="B")
>>     a <- new("A", slot1=77, slot2=TRUE)
>
>>     new("C", a, design=x ~ y)  # fails
>>     new("C", slot1=a at slot1, slot2=a at slot2, design=x ~ y)  # works
>
>> Note that new("B", a, design=x ~ y) works so the 3-level class
>> hierarchy is really needed in order to reproduce.
>
>> Probably related to this, I also noted that new("B") and/or new("C")
>> return invalid objects:
>
>>     c <- new("C")
>
>>     validObject(c)
>>     # Error in validObject(c) :
>>     #  invalid class “C” object: invalid object for slot "design"
>>     #  in class "C": got class "S4", should be or extend class "formula"
>
>>     is(c at design, "formula")
>>     # [1] FALSE
>
>>     class(c at design)
>>     # [1] "S4"
>
>> Note that 'c' can be fixed:
>
>>     c at design <- formula(NULL)
>
>>     validObject(c)
>>     # [1] TRUE
>
>> Maybe something that the default "initialize" method should take care
>> of?
>
>> Another singularity that is maybe at the root of all of this is that
>> the "formula" S4 class is virtual:
>
>>     showClass("formula")
>>     # Virtual Class "formula" [package "methods"]
>>     #
>>     # Slots:
>>     #
>>     # Name:   .S3Class
>>     # Class: character
>>     #
>>     # Extends: "oldClass"
>
>> so a bare call to new("formula") fails:
>
>>     new("formula")
>>     # Error in new("formula") :
>>     #   trying to generate an object from a virtual class ("formula")
>
>> Shouldn't new("formula") just return an "empty" S3 formula (like
>> formula(NULL) does), in the same way that new("integer") returns
>> an empty ordinary integer vector?
>
> Interesting .. and at least worth following.
>
> One problem and historical reason for the current setup seems
> that the "formula" S3 class is not from 'base' but 'stats' :
>
> R's source, src/library/methods/R/BasicClasses.R,
> lines 524 ff has the following comment block
>
> |  .OldClassesPrototypes is a list of S3 classes for which prototype
> |  objects are known & reasonable.  The classes will reappear in
> |  .OldClassesList, but will have been initialized first in
> |  .InitBasicClasses.  NB:  the methods package will NOT set up
> |  prototypes for S3 classes except those in package base and for "ts"
> |  (and would rather not do those either).  The package that owns the
> |  S3 class should have code to call setOldClass in its
> |  initialization.
>
> So, when John Chambers wrote this, he envisioned that the
> 'stats' package would do "the correct thing" for its own classes.
> OTOH, as history went, the stats package was never allowed to
> depend on methods.
> There are many other S3 classes from 'stats' which also end up
> similarly, being defined via  setOldClass() and that itself
> produces a VIRTUAL class.
> Also, another part of the (R source) comment above is no longer
> quite accurate, e.g., "data.frame" is in .OldClassesPrototypes
> but not in .OldClassesList ...
>
> As I do agree that "formula" is much more basic than these other classes,
> I'm currently looking at tweaks to the methods (and stats) code,
> to get this to work.... as indeed - you mentioned above -  we
> already allow "empty S3 formula" objects anyway.
>
> ... half an hour later : Indeed, I've been able to use the above information
> such that  new("formula") and new("formula", y ~ x)
> work.
>
> However, your code above now --- with my changes --- would  fail :
>
>   > setClass("A", representation(slot1="numeric", slot2="logical"))
>   > setClass("B", contains="A", representation(design="formula"))
>   > setClass("C", contains="B")
>   Error in reconcilePropertiesAndPrototype(name, slots, prototype, superClasses,  :
>     "B" is not eligible to be the data part of another class (must be a basic class or a virtual class with no slots)
>   >
>
> So, I am not yet committing my changes to R-devel.
> Hopefully more on this, later today.
>
> Martin Maechler,
> ETH Zurich
>
>
>> Thanks,
>> H.
>
>>   > sessionInfo()
>> R version 3.2.0 Patched (2015-04-17 r68202)
>> Platform: x86_64-unknown-linux-gnu (64-bit)
>> Running under: Ubuntu 14.04.2 LTS
>
>> --
>> Hervé Pagès
>> Fred Hutchinson Cancer Research Center
>
>   [..................]
>
>

-- 
Hervé Pagès

Program in Computational Biology
Division of Public Health Sciences
Fred Hutchinson Cancer Research Center
1100 Fairview Ave. N, M1-B514
P.O. Box 19024
Seattle, WA 98109-1024

E-mail: hpages at fredhutch.org
Phone:  (206) 667-5791
Fax:    (206) 667-1319



More information about the R-devel mailing list