Replies to a) and b) below:

Andi Gutmans wrote:
Hi Stefan,

Thanks for writing such a good proposal.
In general I think traits is a very interesting idea. I can think of a few 
cases in my life as a programmer where it could have helped me out.

I think some comments re: the syntax/naming are good ones but I prefer not to 
get into naming in this email. I believe we can brainstorm re: naming when we 
figure out exactly what we want to do. I do support though finding a way which 
is simple for non-English speakers but not too simple like Perl:)

Thinking back at the times when something like this would have been useful to 
me I think there are a couple of main points I'd like to brainstorm about:

a)
I think Traits should be able to act as a self-contained behavior which can 
always be expected to work. For example if I want a Counter behavior I would 
like that not to depend on the properties in the containing class. While I 
don't think we should enable public nor protected properties in Traits I think 
allowing for private properties in Traits would come in very handy. It also is 
no problem when it comes to mangling as we can use the Trait name.

class Trait {
        private $counter = 0;
        function getNextSerialNumber() {
                return $this->counter++;
        }
}

I strongly recommend not to support protected/public and not to even get into 
the discussion of dealing with conflicts of such properties. But I think 
private is very useful.


If you read the stateful traits paper Stefan linked to, you can see a precedent for how this has been implemented. By default the properties are private to the scope of the trait. The client class can also make an alias to the property and have access to this property through this private alias property. The client class can also merge multiple properties into one common properties.

The problem with this is during flattening. When there is a conflicting property, all references have to be "alpha-renamed" in all the trait methods. This is a more costly operation as compared to the flattening of stateless traits. This may not be reason enough not to include it, but it should definitely be a consideration.

b)
I haven't thought this through completely but I think we may want to consider a different model 
from supporting "removing" and "aliasing" of functions. I think this can create 
a lot of complications especially in larger works and it'll be write-once code and hard for a 
newcomer to the code to really understand.

I think the main reason to support removing/aliasing of functions is in order to avoid 
conflicts with other Traits/Class methods. Maybe what we can do is to have only 
"aliasing" but what it would do is to create a public function with the new 
name and convert the old function into a private function.
Benefits:
- Keeps the old code from running predictably without breaking dependencies.
- Possibly even allowing some form of "is-a" relationship to continue to be 
valid (and therefore the interface discussion may even be resolved; at least to a certain 
level). In the case I faced an is-a relationship (i.e. working instanceof operator) would 
have been nice.

I need to put a bit more thought into this but it's an initial attempt to 
provide an additional point of view which may actually give some more structure 
to the proposal while still delivering the benefits of this idea.

I think if this kind of idea could be fully baked it would solve the problems I 
have had in the past possibly without the more dangerous renaming/removing of 
functions. I have not looked into feasibility of this idea yet.

Anyway, good ideas and thanks for the very professional way you have brought 
this idea to the community.
        
Let's try and nail down the behavior and then we can have the harder naming 
discussion :) Just saying harder because I've always considered naming the 
hardest part in software engineering...

Andi

Let me use a quote on aliasing from this paper <http://www.iam.unibe.ch/~scg/Archive/Papers/Scha03aTraits.pdf>: "The new name is used to obtain access to a method that would otherwise be unreachable because it has been overridden." The point of aliasing is to give the client class a way to call the old method after it has been overridden.

Generally, if two traits have the same method names, you would want to alias both of them, override the method, and use the aliases to compose the overridden method. The other option is to just choose one of the trait's method implementations and exclude all others.

And if you are talking about making the old method that is aliased into a method that is private to just that trait, flattening would then require some kind of "alpha-rename" for even stateless traits in some cases.

--
"Joshua Thompson" <[EMAIL PROTECTED]>
<http://www.schmalls.com>

--
PHP Internals - PHP Runtime Development Mailing List
To unsubscribe, visit: http://www.php.net/unsub.php

Reply via email to