Sealed Subclasses of Dynamic Classes

So, what’s the deal with sealed subclasses of dynamic classes? First of all, if you have a class like this:

package {
    import flash.display.*;
    public class SealedClip extends MovieClip {
        public function method():void {
            this.q = 5; // compiler error, of course
            trace(this === super); // true
            super.q = 5; // runtime error, ReferenceError: Error #1056: Cannot create property q on flash.display.MovieClip.
            this['q'] = 5; // ReferenceError: Error #1056: Cannot create property q on SealedClip.
        }
    }
}

// elsewhere
var clip:MovieClip = new SealedClip();
clip.property = '10'; // ReferenceError: Error #1056: Cannot create property property on SealedClip.
clip.method();

…you run into issues since the compiler can’t catch the errors, which is expected, but still fairly silly. The weird thing is that this === super, but the error claims that super is a MovieClip. If it really was a regular MovieClip, though, then an error wouldn’t have been generated. I’m guessing that the SealedClip instance doesn’t actually contain a full MovieClip instance, because a regular MovieClip would have a hashtable of properties. Instead, only the MovieClip’s trait object is being passed on to the SealedClip, so the SealedClip instance won’t have a property hashtable either. That still doesn’t really explain why super thinks that it is this. The AS3 documentation doesn’t really clarify anything, either.

The super statement provides such a mechanism, in that it contains a reference to the immediate superclass.

Since I’ve tried to establish that an actual reference to the superclass doesn’t exist (or at least not a reference similar to one that you would get when instantiating the superclass by itself), the documentation’s statement can’t be quite correct. So what’s really going on with super here?

In a somewhat related matter, I find it somewhat amusing that sealed subclasses can break their parents:

package {
    public dynamic class DynamicBase {
        public function DynamicBase(){
            dynamicMethod();
        }
        public function dynamicMethod():void {
            this['dynProp'] = 'dynValue';
        }
    }
}

// ...

package {
    public class SealedSubclass extends DynamicBase {

    }
}

// later
var klass:DynamicBase = new SealedSubclass();

/*
ReferenceError: Error #1056: Cannot create property dynProp on SealedSubclass.
    at DynamicBase/dynamicMethod()
    at DynamicBase()
    at SealedSubclass()
*/

It makes sense that that error occurs, since the methods are being called in the context of the new, sealed object that is being created.

My question is, was this done purely for performance and to maneuver around difficulties created with the dynamic Object class? It seems that in another AS3 implementation, one could have had a hashtable made for an object only if one or more of its defining classes was dynamic. That way, the penalty of adding the hashtable would always exist as long as there was one dynamic class in the inheritance chain, but then one wouldn’t run into trouble with breaking existing classes simply by sealing a subclass. Of course, such an implementation would cause problems due to the Object class being dynamic, since all instances would then need a hashtable and all of the performance gains gained from allowing sealed classes would go away. One then wonders whether or not it was a good idea to make the Object class dynamic. It probably would have been too big of a compatibility issue to seal the Object class for AS3, so I can see why it is still dynamic and why sealed classes eliminate the hashtables that would otherwise exist… at least in the actual implementation of AS3. Perhaps making the object class dynamic was just the result of a lack of ability to see where ActionScript would eventually be headed, which is more than understandable. To me, it seems that it would make more sense to have some sort of HashTable or similar class be the base class for dynamic classes, and then leave Object as the base for all other classes (since it seems that most classes don’t need to be dynamic), if one was going to allow sealed classes at all. I’m pretty sure that that’s what’s done in languages like Java.

Does the above analysis seem correct? (-: This is the sort of stuff that I am actually interested in, so I would appreciate comments.

“this” returns the memory address of the instance of the class, and since “super” is also the instance of the class (only down-casted) they both return the same value. That way “this” is always equal to “super” and it’s “super” and so on.
When you create a sealed class, all classes it extends that were previously dynamic will become sealed too (only one dynamic class would needed to make the instance dynamic if it didn’t work that way, so MovieClip and Object will cease to be dynamic).

This is why you can’t take any class and down-cast it to object to add dynamic variables to it. It may be down-casted, but it is still an instance of the class and is therefore still sealed.
For example, you can’t take a Sprite, downcast it to Object, add variables to it and cast it to Sprite again.
[AS]var mySprite:Sprite = new Sprite();
var spriteObject:Object = Object(mySprite);
spriteObject.dynamicProperty = “foo”; // Throws an error, it may be downcasted to an Object, but it is still a Sprite
return mySprite;[/AS]
That wouldn’t make sense, you could add dynamic variables to any sealed class :slight_smile:

You need :sen:! :lol:

super is supposed to be the same object. The point of super is so you can access methods and properties of the parent class in case they’ve been overridded so you can’t access them through this. As for the rest. You aren’t declaring any of the variables so I don’t understand how you expect it to work. You always need to declare every variable as in var [identifier]:[class] or var q:int = 5

Regarding Super:
I’m not quite sure if and how this relates, however, go down to the title:
**"**The ActionScript 3.0 class object" on this link here
The Traits Object, and the prototype property (to some extent) seem to explain what is happening with super.

flassari: I think you’re missing the point. There would be no point to the super reference if it referenced the same thing as this. Casting without using one of the special global functions never makes new methods or properties visible or available, it only shuts the compiler up, so something special must be happening with super. To use Wikipedia’s example, if you have an Animals that speak (with the default string “Speak”), you can never make a Dog instance’s speak method print the string “Speak” by casting if the Dog class overrides speak so that it instead prints “Woof”. That’s why super is a special construct, not just a casted form of this. No?

I think that you must have missed my discussion of this:

Again, I think that you missed parts of my post, since I discussed that in the last paragraph (at length):

If super is precisely the same object as this, then there would be no reason to include super in AS3 at all. Obviously something is going on behind the scenes, and I was attempting to ask whether or not anyone here knew any implementation details.

You’re missing the point of dynamic classes. Declaring a class as a dynamic class allows you to declare properties as I did in my post. As you can see, I commented in the errors that would occur if you were to run any of those lines. However, the reason I used the syntax that I did was because it would be valid syntax if the implicit reference in the accessor call was a dynamic instance. When I said “fairly silly,” I was referring to the compiler’s inability to catch the error. I realize that it would be difficult or impossible for a compiler to catch such errors, but I was using my power to have an opinion to call the difficulty silly.

creatify: I actually had that page open as I was making my first post (especially when I mentioned traits in my first full paragraph). I agree that it kind of hints at what’s probably going on, but it doesn’t really provide the sort of detail that I was looking for, which is part of the reason why I made my post. Since I’m asking about some internal details, I am basically looking for documents like that one, or, like Anogar suggested, someone like senocular to explain what may not be documented anywhere yet.

I can’t speak to the inner workings of [FONT=“Courier New”]super()[/FONT] - but an observation is that strict equality ([FONT=“Courier New”]===[/FONT]), from my understanding, compares not the contents of the two expressions but whether or not the two pointers end up at the same address in the memory.

Since classes that extend a superclass are instances also of that superclass, as evidenced by something like this:


var mySprite:Sprite = new Sprite();

trace(mySprite instanceof EventDispatcher); //traces true

… it occurs to me that perhaps [FONT=“Courier New”]this()[/FONT] refers to the specific link in the prototype chain that is specifically a [FONT=“Courier New”]Sprite[/FONT], but that as a whole object, both [FONT=“Courier New”]Sprite [/FONT]and [FONT=“Courier New”]EventDispatcher [/FONT]are contained within the memory pointer that is [FONT=“Courier New”]mySprite[/FONT].

Does that make any sense? Perhaps [FONT=“Courier New”]super()[/FONT] works exactly how you think it does, it’s [FONT=“Courier New”]===[/FONT] that is a little different. :2c:

Look, if ClassB extends ClassA, ClassB is still a ClassA. Lets say ClassA has a method Met1, and ClassB overrides it with a new version of Met1. If you cal Met1 from ClassB you will use ClassB’s version of Met1. But what if you want to use ClassA’s version of Met1? Then you call super.Met1 so you can use the parent class’s version of it.

We all understand that, I’m not sure what it’s in response to.

Well he wrote: If super is precisely the same object as this, then there would be no reason to include super in AS3 at all.

Which gave me the impression he didn’t know what super did.

He’s curious why *this *strictly evaluates to super even though you can’t add dynamic properties as you might expect of a MovieClip, he knows exactly how to use them both. I think you’ve misunderstood his question.

… it occurs to me that perhaps this() refers to the specific link in the prototype chain that is specifically a Sprite, but that as a whole object, both Sprite and EventDispatcher are contained within the memory pointer that is mySprite.

Does that make any sense? Perhaps super() works exactly how you think it does, it’s === that is a little different.
Right, we know that object instances contain pointers to the trait objects from their parent classes and defining class, like EventDispatcher and Sprite. Your idea that “perhaps this() refers to the specific link in the prototype chain that is specifically a Sprite” is pretty much the kind of idea that I wanted to hear. I mean, do you think that it’s likely that certain operators’ responses to this and super were just essentially hard-coded into AS? Going back to one case that I mentioned before:

super['prop'] = 5; // Cannot create property prop on DynamicBase.
this['prop'] = 5;    // Cannot create property prop on SealedSubclass. 

Unlike the === operator, the operator reports that there’s a difference between this and super. I like the way that you said “strict equality noparse[/noparse], from my understanding, compares not the contents of the two expressions but whether or not the two pointers end up at the same address in the memory.” To me, it seems like the AS3 implementors would have had to think, “Hmm, which object do we want super to resolve to when comparing with other objects?” and then separately think “Hmm, which object do we want super to resolve to when accessing properties or methods?” because, for example:

super.overrideMethod == this.overrideMethod // false 

Basically, the only difference between all of these uses of this and super is that a different operator is being used, and the results are only sometimes the same between the two keywords or special objects or whatever you want to call them. So, it could be that something happens to super at compile time that differentiates it from this, it could be that references (in AS) are more complicated than super simple pointer abstractions, and they contain additional info that would be able to help operators differentiate between the two in some cases at runtime, or it could be that the super reference references an object that isn’t quite this, but that is resolved to this by some operators (which reminds me of your chain idea, since the special behavior of this that you were talking about would imply that there is some extra information passed when using this that is used internally, and that makes a lot of sense).

Thanks Anogar, I’m definitely getting some useful thought from you out of this thread. :slight_smile: Of course I would still appreciate more thoughts from you or others, since I don’t think we’ve quite reached the bottom of the mystery, so to speak.

Edit: Fidodo: My response was directed at flassari’s:

I was arguing that if super was just like *this *except that that it was casted to some other type (which is what flassari was asserting), then you could just do something like ParentClass(subclassInstance).overrideMethod to replace all uses of super, which isn’t the case.

Glad to give you some things to think about. I agree that we’re not to the bottom of this - my idea was total conjecture, I have no idea if that’s the case or not. Bump in hopes of Senocular noticing it. :slight_smile:

Do you think that looking through the Tamarin source code would help?

Okay, I think I’ve found the answer to the super question:

I found most of the following information from the ECMAScript Fourth Edition Language Specification Working Draft 1.0 and the Standard ECMA-262 3rd Edition ECMAScript Language Specification.

First of all:

So it seems like we were right about traits having something to do with the discrepancy, and with some more information about references in AS3, one can figure out exactly why:

The first bold line almost explains what’s going on, but it’s not very explicit and doesn’t explain super expressions. The next lines that are bold are so just to point out that the Reference structs have a member named limit that holds the traits object. Also, it initially wasn’t clear to me that the “references resulting from super expressions” were LimitedBases (which is relevant in the evaluation of super expressions). Here are relevant parts of the section that describes super:

For comparison, here is the evaluation portion of the this expression (the important part is just to note that they are evaluated differently):

Looking at the SuperExpression evaluation, one can see that the end result, at least in the case that I was concerned with, is that a new reference is generated that contains the same object reference but a different limit member, which means that the traits object would be the superclass’ traits object. This is what caused the different runtime error text when I attempted to access properties of super and this. When the super expression was evaluated, the reference that was returned had a different traits object to look through for the property, and since it was established earlier that traits are used to identify type, it makes sense that the super error would report that it was a DynamicBase, and the this error would report that it was a SealedSubclass.

The evaluation description above also helps to answer the strict equality question. Although the references that result from the evaluation of this and super are different, they still pass the equality test due to the way the algorithm is implemented: (The following is from the ES3 specification because the ES4 specification draft noted that the implementation were the same)

To “refer to the same object” doesn’t involve comparing the Reference struct instances (since those would be different, but the “this === super” test returned true), but instead the base member of the structs that result from the evaluation of this and super. Those two references would be the same, due to step 5.1 of super’s evaluation, which gives the eventual base the same value as this. :beam:

[whisper]I thought that a double-post would be fine given that this post isn’t really an ‘edit’ of the earlier post.[/whisper]

Wow, great research. Very interesting stuff. :slight_smile:

[whisper]Best double post evar.[/whisper]