Extended Type Design.

Andrei Alexandrescu (See Website For Email) SeeWebsiteForEmail at erdani.org
Sat Mar 17 00:04:35 PDT 2007


Reiner Pope wrote:
> Andrei Alexandrescu (See Website For Email) wrote:
>> Reiner Pope wrote:
>>> Andrei Alexandrescu (See Website For Email) Wrote:
>>>
>>>> We have talked about a design. In short, the intent is to define 
>>>> three flavors of immutability:
>>>>
>>>> a) final - a simple storage class controlling the immutability of 
>>>> the bits allocated for the symbol per se;
>>>>
>>>> b) const - type qualifier meaning an immutable view of an otherwise 
>>>> modifiable data. const does not control the bits of the object, only 
>>>> the storage addressed indirectly by it (transitively);
>>>>
>>>> c) "superconst" - denoted as "const!" or "super const": type 
>>>> qualifier meaning that the data is genuinely unmodifiable.
>>>>
>>> Is there anything in your discussion about compile-time constants 
>>> (the sort that you need for 'static if', etc) ? You spoke a while ago 
>>> about improving compile-time evaluation syntax so that there would be 
>>> no difference at the call  site between runtime and compile-time 
>>> evaluation (like for regexps).
>>
>> I understand this is high on Walter's list.
> Good to hear that.
> 
>  The syntax discussed is:
>>
>> // dispatched when the pattern is known at compile time
>> RegExpMatch match(const char[] p)(const char[] p, char[] text) {...}
>> // dispatched when the pattern is a runtime string
>> RegExpMatch match()(const char[] p, char[] text) {...}
> That seems fine, but I can't help wondering, "why the double-declaration 
> of p in the first instance?" and also, "why does the second instance of 
> match even need to be a template?".

Good question. It must be a template by the pigeonhole principle: each 
compile-time regex generates in theory different code, so you must have 
as many distinct functions as strings.

>>> It seems like you could get this quite easily by adding just one more 
>>> parameter modifier (is this a storage class? I'm not sure) to denote, 
>>> "any time this function is called, this parameter must be known at 
>>> compile-time". Effectively, it would just be a conversion from this:
>>>
>>> RegExpMatch match(static const char[] pattern, char[] text) {...}
>>> ...
>>> main()
>>> {
>>>      match("gr[ae]y", din.read());
>>> }
>>>
>>> to this:
>>>
>>> RegExpMatch match(char[] pattern)(char[] text) {...}
>>> ...
>>> main()
>>> {
>>>      match!("gr[ae]y")(din.read());
>>> }
>>>
>>> Any plans this way?
>>
>> The above forces only compile-time constant regexps. What we need is 
>> to dispatch depending on compile-time constant vs. run-time string.
> Well, I kind of took it for granted that the above suggestion would 
> allow for overloading by compile-time-constness (with a preference to 
> the compile-time version when there's a choice). So that you could write:
> 
> RegExpMatch match(static const char[] pattern, char[] text){...}
> RegExpMatch match(char[] pattern, char[] text) {...}
> 
>>
>> Combined with macros, this will essentially allow a call to writefln 
>> to morph, depending on its arguments, into putch(), fputs(), write(), 
>> fprintf() - or execute today's dynamically-checked writefln.
> 
> True. I was also interested in another use of this, though: allowing a 
> library-time implementation of tuples (since what kind of feature is one 
> that can't be implemented in the library?). There are basically two 
> reasons that this can't be done at the moment:
> 
> alias Tuple!(int, double) myTuple;
> myTuple[0] = 5; // First reason
> foreach (val; myTuple) // Second reason
> {
>     writef(val);
> }
> 
> I would imagine that you could implement the first feature as:
> 
> struct MyTupleImpl(TypeList...)
> {
>    ...
>    TypeList[i] opIndex(static const int i) {...}
> // We grab the return type based on the parameter, so we must know it at 
> compile time. It makes no sense to have a runtime version of this.
> }
> 
> and at a stretch, you could imagine the second feature as working in a 
> similar way: your compile-time representation of the foreach delegate is 
> either a char[] to be mixed in, or an expression alias, when we get that:
> 
> struct MyTupleImpl(TypeList...)
> {
>     int opApply(static const char[] code)
>     {
>         // iterate through tuple; this iteration would actually have to 
> be generated with CTFE, but you get the idea
>         {
>             TypeList[i] __special_foreach_key = whatever;
>             mixin(code);
>         }
>     }
> }

That's not on the list yet. One thing at a time :o).


Andrei



More information about the Digitalmars-d mailing list