problems with Rebindable

ag0aep6g via Digitalmars-d-learn digitalmars-d-learn at puremagic.com
Sat May 21 06:47:08 PDT 2016


On 05/21/2016 03:36 PM, chmike wrote:
> Note however that it doesn't work with immutable. It only works with
> constant. I guess this is because immutable is "stronger" than const. I
> determined that only const was supported by looking at Rebindable's code.
>
> Here is the code that finally works as I want. The flyweight pattern is
> thus well supported with the exception that switch can't be used.
>
> using static functions to get the Infos.one also allow to implement lazy
> object instantiation.
[...]
> I wasn't indeed using Rebindable correctly and it support only const
> objects, not immutable objects.

I think your conclusion is wrong. Works fine if you add a couple 
`immutable`s (and change one `IInfo` to `Info`):

----
import std.stdio;
import std.typecons;


interface IInfo {
     string toString() const;
}

alias Rebindable!(immutable IInfo) Info; // Compiles just fine.

class Infos {

     static class Obj : IInfo
     {
         this(string msg) immutable { this.msg = msg; }
         private string msg;
         override string toString() const { return msg; }
     }

     static Info one()
     {
         static auto x = Info(new immutable Obj("I'm one"));
         return x;
     }
     static Info two()
     {
         static auto x = Info(new immutable Obj("I'm two"));
         return x;
     }
}

void main()
{
     Info x1;
     Info x2 = Infos.one;

     assert(x1 is null);
     assert(x2 !is null);
     assert(x2 is Infos.one);
     assert(x2 == Infos.one);

     x1 = x2;
     assert(x1 is x2);
     assert(x1 == x2);

     assert(x1 is Infos.one);
     assert(x1 == Infos.one);

     writeln(x1);

     Info x3 = Info(new immutable Infos.Obj("I'm one"));
     assert(x1 !is x3);
     assert(x1 != x3); // Because there is no opEqual for deep equality test

     Info
         o1 = new immutable Infos.Obj("I'm one"),
         o2 = new immutable Infos.Obj("I'm one");
     assert(o1 !is o2);
     assert(o1 != o2); // What I need for the flyweight pattern

     /* -- Doesn't compile : x1 is not a string or integral value
     switch(x1)
     {
     case Infos.one: writeln("case Infos.one"); break;
     default: writeln("default"); break;
     }
     */
}
----

I thought marking the constructor `pure` would make it possible to 
implicitly convert a mutable `new` expression to immutable, but I 
couldn't get that to work. That would avoid all those `immutable`s. I'm 
probably forgetting something here.


More information about the Digitalmars-d-learn mailing list