Thread overview
Doubt about Struct and members
Jan 08
matheus
Jan 08
matheus
January 08
Hi,

I was doing some tests and this code:

import std;

struct S{
    string[] s = ["ABC"];
    int i = 123;
}

void foo(bool b, string str){
    S t1;

    writeln("t1.s: ", t1.s, ", t1.s.ptr: ", t1.s.ptr, " t1.i: ", t1.i);

    if(b){
    	t1.s[0] = str;
    }else{
    	t1.s = [str];
    }

    t1.i = 456;
    S t2;

    writeln("t1.s: ", t1.s, ", t1.s.ptr: ", t1.s.ptr, " t1.i: ", t1.i);

    writeln("t2.s: ", t2.s, ", t2.s.ptr: ", t2.s.ptr, " t2.i: ", t2.i);
    writeln("");
}

void main(){
    foo(false, "DEF");
    foo(true, "DEF");
    foo(false, "XYZ");
}

Outputs:

t1.s: ["ABC"], t1.s.ptr: 56421C6D7010 t1.i: 123
t1.s: ["DEF"], t1.s.ptr: 7EFC725E6000 t1.i: 456
t2.s: ["ABC"], t2.s.ptr: 56421C6D7010 t2.i: 123

t1.s: ["ABC"], t1.s.ptr: 56421C6D7010 t1.i: 123
t1.s: ["DEF"], t1.s.ptr: 56421C6D7010 t1.i: 456
t2.s: ["DEF"], t2.s.ptr: 56421C6D7010 t2.i: 123

t1.s: ["DEF"], t1.s.ptr: 56421C6D7010 t1.i: 123
t1.s: ["XYZ"], t1.s.ptr: 7EFC725E6020 t1.i: 456
t2.s: ["DEF"], t2.s.ptr: 56421C6D7010 t2.i: 123

As you can see:

    t1.s = [str];

Just changed generated a new address only for t1.s, on the other hand:

    t1.s[0] = str;

Changed the value pointed by S.s entirely (The other "instance" t2 since points to the same address now has the new value too).

Is this intended?

Matheus.
January 08
On Mon, Jan 08, 2024 at 05:28:50PM +0000, matheus via Digitalmars-d-learn wrote:
> Hi,
> 
> I was doing some tests and this code:
> 
> import std;
> 
> struct S{
>     string[] s = ["ABC"];
>     int i = 123;
> }
[...]

It's not recommended to use initializers to initialize mutable array-valued members, because it probably does not do what you think it does.  What the above code does is to store the array ["ABC"] somewhere in the program's pre-initialized data segment and set s to point to that by default. It does NOT allocated a new array literal every time you create a new instance of S; every instance of S will *share* the same array value unless you reassign it.  As such, altering the contents array may cause the new contents to show up in other instances of S.

This behaviour is generally harmless if your array is immutable. In fact, it saves space in your executable by reusing the same data for multiple instances of s. It also avoids repeated GC allocations at runtime.

However, if you're banking on each instance of S getting its own copy of the array, you're in for a surprise. In this case, what you want is to use a ctor to initialize it rather than the above initializer.


T

-- 
Right now I'm having amnesia and deja vu at the same time. I think I've forgotten this before.
January 08
On Monday, 8 January 2024 at 17:56:19 UTC, H. S. Teoh wrote:
> ...
> It's not recommended to use initializers to initialize mutable array-valued members, because it probably does not do what you think it does.  What the above code does is to store the array ["ABC"] somewhere in the program's pre-initialized data segment and set s to point to that by default. It does NOT allocated a new array literal every time you create a new instance of S; every instance of S will *share* the same array value unless you reassign it.  As such, altering the contents array may cause the new contents to show up in other instances of S.
>
> This behaviour is generally harmless if your array is immutable. In fact, it saves space in your executable by reusing the same data for multiple instances of s. It also avoids repeated GC allocations at runtime.
> ...

First of all thanks for replying,

Yes I understood the behavior since I even looked the code generated: https://godbolt.org/z/xnsbern9f

=]

Maybe my question was poorly written (I'm ESL), but you answered in the other Topic:

> "(Whether the current behaviour should be changed is up for debate, though. This definitely isn't the first time users have run into this. In fact just today somebody else asked the same question on D.learn. So it's definitely in the territory of "does not do the expected thing", which is an indication that the default behaviour was poorly chosen.)"

I was in doubt about if this was intended or not... and it seems the case.

I'm not saying it is wrong by any means, I just found a bit tricky based on the two ways it could change the value pointed by the address and/or the address of the member itself.

Again thanks,

Matheus.