Red Code, Green Code (nwcpp video)
janderson
askme at me.com
Tue May 8 19:14:58 PDT 2007
Pragma wrote:
> janderson wrote:
>> sergk wrote:
>>> I've just noticed a Northwest C++ Users Group presentation
>>> "Red Code, Green Code: Generalizing const" by Scott Meyers, and
>>> found it quite interesting.
>>>
>>> So here it is:
>>> http://nwcpp.org/Meetings/2007/04.html
>>>
>>>
>>> PS./OT: seems Scott do not really enjoying C++ template meta
>>> programming =)
>>>
>>> -- serg.
>>
>> The next time some tries to argue library over language, or that
>> templates in C++ are perfect, we should point them to that video.
>>
>> PS - What would be the nicest way to do this in D using current syntax?
>
> Well, we can trans-literate from the C++ code to get the same results.
> Here I have an example that replaces the functionality from the vector
> template with tuples, and some additional template namespaces and
> operations to make it all come together. The actual example is in the
> bottom third of this snippet.
>
> (Mucking with code from
> http://herbsutter.spaces.live.com/blog/cns!2D4327CC297151BB!207.entry)
>
> // This is placed in the Public Domain
>
> template Tuple(V...){
> alias V Tuple;
> }
>
> template InTuple(Item){
> const bool InTuple = false;
> }
>
> template InTuple(Item,First,V...){
> static if(is(Item == First)){
> const bool InTuple = true;
> }
> else alias InTuple!(Item,V) InTuple;
> }
>
> template TupleRemoveAll(Item){
> alias Tuple!() TupleRemoveAll;
> }
>
> template TupleRemoveAll(Item,First,V...){
> static if(is(Item == First)){
> alias TupleRemoveAll!(Item,V) TupleRemoveAll;
> }
> else alias Tuple!(First,TupleRemoveAll!(Item,V)) TupleRemoveAll;
> }
>
> // use template 'struct' to prevent tuple flattening when passing
> constraints around
> template Constraints(V...){
> alias V list;
> }
>
> template ConstraintRemove(alias Constraints){
> alias Constraints.list list;
> }
>
> template ConstraintRemove(alias Constraints,First,V...){
> alias Cosntraints!(TupleRemoveAll!(First,Constraints.list))
> newConstraints;
> alias Tuple!(newConstraints,ConstraintRemove!(newConstraints,V)) list;
> }
>
> struct IgnoreConstraints {}
>
> template ConstraintCheck(alias Local,Caller:IgnoreConstraints){
> const bool ConstraintCheck = true;
> }
>
> // ensure that all local constraints are satisfied by caller
> template ConstraintCheck(alias Local,alias Caller){
> static if(Local.list.length > 0){
> static if(Caller.list.length > 0){
> alias Local.list[0] first;
> static if(InTuple!(Local.list[0],Caller.list)){
> alias Constraints!(TupleRemoveAll!(first,Local.list))
> newLocalConstraints;
> alias ConstraintCheck!(newLocalConstraints,Caller)
> ConstraintCheck;
> }
> else{
> const bool ConstraintCheck = false;
> }
> }
> else{ // nothing left to check
> const bool ConstraintCheck = false;
> }
> }
> else{ // nothing left to check
> const bool ConstraintCheck = true;
> }
> }
>
> // Here's the useful part:
> struct ExceptionSafe {}
> struct LGPLed {}
> struct Reviewed {}
>
> void f(alias CallerConstraints)(Params params){
> alias Constraints!(ExceptionSafe) MyConstraints;
> static assert(ConstraintCheck!(MyConstraints,CallerConstraints));
> /* do something */
> }
>
> void g(alias CallerConstraints)(){
> alias Constraints!(ExceptionSafe,Reviewed) MyConstraints;
> static assert(ConstraintCheck!(MyConstraints,CallerConstraints));
>
> // try to call the other function
> f!(MyConstraints)(); // error, trying to call unreviewed code from
> reviewed code
> f!(ConstraintRemove!(MyConstraints,Reviewed))(); // ok, ignore
> Reviewed constraint
> f!(IgnoreConstraints)(); // ok, explicitly ignore constraints entirely
> }
>
> void main(){
> alias Constraints!(ExceptionSafe,Reviewed) MyConstraints;
> g!(MyConstraints)();
> }
>
> This won't compile under DMD 0.012 and later - Bug 1196 keeps operations
> like InTuple!() from operating correctly. So this is only half-tested.
> There are also a few different ways one could implement this. I opted
> for this particular definition of Constraints!() to allow for the
> ConstraintCheck!() operation to work for comparing two tuple lists
> (otherwise, they'd collapse into a single tuple).
>
> Aside from the advantages that tuples offer, I'm having a hard time
> seeing a clear advantage over the C++ version. We still get template
> bloat, and namespace concerns, plus it's not much more readable. At a
> minimum, the lack of a need for a heavy amount of compile-time code is a
> clear advantage and I think (not sure here) the concerns about virtual
> methods are also addressed thanks to D's design. So I think this edges
> out the C++ version rather invisibly.
>
Maybe something like C# user-definable attributes would help. (I'm sure
someone has asked for this before).
More information about the Digitalmars-d
mailing list