digitalmars.D - using opAssign or having my own assign function
- Martin d Anjou (23/23) Sep 04 2007 Hi,
- Bill Baxter (19/51) Sep 04 2007 The only other thing that comes to mind is that you could use the
- Reiner Pope (53/85) Sep 04 2007 If the size of your BitVector is fixed at compile-time, then you can
- Martin d Anjou (4/14) Sep 08 2007 Nice. I got it to work too. Can the declaration look more like a built-i...
- Reiner Pope (7/23) Sep 08 2007 Isn't that exactly what it is?
- Matti Niemenmaa (6/20) Sep 08 2007 Since BitVector's opCall does practically nothing, you might as well rem...
Hi, I am trying to build a data type where a key characteristic is that the data holding capacity is fixed and never changing, and the declaration looks like this: auto a = BitVector(4); // "a" can hold unsigned numbers from 0 to 15 auto b = BitVector(10); // 10 bits of data auto c = BitVector(79); // etc. Say I now want to add b and c, and store the result in a, while preserving the property that "a" has to have a capacity of 4 bits only. Adding and truncating the upper bits to fit in "a" is the easy part. The problematic part is stating that the result of the addition goes to variable "a". I was first tempted to simply say "a=b+c", but b+c will return a new instance of BitVector, scrap the original "a", and loose the holding capacity of "a". I thought of overloading opAssign(), but I can't do "BitVector opAssign(BitVector other) { ... }". So I came to the conclusion the least non-intuitive thing to do would be "a.assign(b+c)", which preserves the storage capacity declared for variable "a", and does not create a new instance for "a". Question: Is there a better way to do this? Question: Is there a chance opAssign could be used for this kind of thing at all in the future? Thanks, Martin
Sep 04 2007
Martin d Anjou wrote:Hi, I am trying to build a data type where a key characteristic is that the data holding capacity is fixed and never changing, and the declaration looks like this: auto a = BitVector(4); // "a" can hold unsigned numbers from 0 to 15 auto b = BitVector(10); // 10 bits of data auto c = BitVector(79); // etc. Say I now want to add b and c, and store the result in a, while preserving the property that "a" has to have a capacity of 4 bits only. Adding and truncating the upper bits to fit in "a" is the easy part. The problematic part is stating that the result of the addition goes to variable "a". I was first tempted to simply say "a=b+c", but b+c will return a new instance of BitVector, scrap the original "a", and loose the holding capacity of "a". I thought of overloading opAssign(), but I can't do "BitVector opAssign(BitVector other) { ... }". So I came to the conclusion the least non-intuitive thing to do would be "a.assign(b+c)", which preserves the storage capacity declared for variable "a", and does not create a new instance for "a". Question: Is there a better way to do this? Question: Is there a chance opAssign could be used for this kind of thing at all in the future? Thanks, MartinThe only other thing that comes to mind is that you could use the expression object trick as in Blitz++ et al. Have opAdd return a BitVectorAddExpr, and then give BitVector an opAssign from BitVectorAddExpr that will just trigger a call to a.assign(b+c). struct BitVector { ... BitVectorAddExpr opAdd(ref BitVector b, ref BitVector c) { return BitVectorAddExpr(&b,&c); } void opAssign( BitVectorAddExpr expr ) { a.assign(*expr.L+*expr.R); } } struct BitVectorAddExpr { static typeof(*this) opCall( . . . ) {. . . } BitVector *L; BitVector *R; } --bb
Sep 04 2007
Martin d Anjou wrote:Hi, I am trying to build a data type where a key characteristic is that the data holding capacity is fixed and never changing, and the declaration looks like this: auto a = BitVector(4); // "a" can hold unsigned numbers from 0 to 15 auto b = BitVector(10); // 10 bits of data auto c = BitVector(79); // etc. Say I now want to add b and c, and store the result in a, while preserving the property that "a" has to have a capacity of 4 bits only. Adding and truncating the upper bits to fit in "a" is the easy part. The problematic part is stating that the result of the addition goes to variable "a". I was first tempted to simply say "a=b+c", but b+c will return a new instance of BitVector, scrap the original "a", and loose the holding capacity of "a". I thought of overloading opAssign(), but I can't do "BitVector opAssign(BitVector other) { ... }". So I came to the conclusion the least non-intuitive thing to do would be "a.assign(b+c)", which preserves the storage capacity declared for variable "a", and does not create a new instance for "a". Question: Is there a better way to do this? Question: Is there a chance opAssign could be used for this kind of thing at all in the future? Thanks, MartinIf the size of your BitVector is fixed at compile-time, then you can integrate this into the type (by making the size a template parameter). In that case, BitVectors of different size are different types, making opAssign work (since you're assigning something of a different type). The following is working code: --- import std.stdio; int max(int a, int b) { return (a > b) ? a : b; } struct BitVector(int len) { alias typeof(*this) BV; static BV opCall() { BV result; return result; } BV opAssign(int len2)(BitVector!(len2) other) { writefln("Assigning vector of size %s to one of size %s", len2, len); return *this; } BitVector!(max(len, len2)) opAdd(int len2)(BitVector!(len2) other) { writefln("Adding a vector of size %s to one of size %s", len2, len); return BitVector!(max(len, len2))(); } } void main() { auto a = BitVector!(4)(); auto b = BitVector!(10)(); auto c = BitVector!(79)(); writefln("A"); a = b + c; writefln("B"); c = b + c; } --- The output is A Adding a vector of size 79 to one of size 10 Assigning vector of size 79 to one of size 4 B Adding a vector of size 79 to one of size 10 --- If, on the other hand, the size can't be made part of the type, then this doesn't work. Bill's expression templates are a solution that also currently work. However, I think a cleaner solution is the future "opCopy" mentioned in WalterAndrei.pdf -- you'll have to wait for it, though. -- Reiner
Sep 04 2007
void main() { auto a = BitVector!(4)(); auto b = BitVector!(10)(); auto c = BitVector!(79)(); writefln("A"); a = b + c; writefln("B"); c = b + c; }Nice. I got it to work too. Can the declaration look more like a built-in (more user friendly), e.g. "BitVector!(4) bv;"? Thanks, Martin
Sep 08 2007
Martin d Anjou wrote:Isn't that exactly what it is? BitVector!(4) bv = BitVector!(4)(); or did you mean alias BitVector!(4) BV4; BV4 a = BV4(); -- Reinervoid main() { auto a = BitVector!(4)(); auto b = BitVector!(10)(); auto c = BitVector!(79)(); writefln("A"); a = b + c; writefln("B"); c = b + c; }Nice. I got it to work too. Can the declaration look more like a built-in (more user friendly), e.g. "BitVector!(4) bv;"? Thanks, Martin
Sep 08 2007
Martin d Anjou wrote:Since BitVector's opCall does practically nothing, you might as well remove it. You could write just "BitVector!(4) bv;" even now, with no difference to "auto bv = BitVector!(4)();" -- E-mail address: matti.niemenmaa+news, domain is iki (DOT) fivoid main() { auto a = BitVector!(4)(); auto b = BitVector!(10)(); auto c = BitVector!(79)(); writefln("A"); a = b + c; writefln("B"); c = b + c; }Nice. I got it to work too. Can the declaration look more like a built-in (more user friendly), e.g. "BitVector!(4) bv;"?
Sep 08 2007