reassign single elements of a string (using foreach)

Derek Parnell derek at nomail.afraid.org
Thu Jul 5 18:57:47 PDT 2007


On Fri, 06 Jul 2007 03:41:11 +0200, Daniel919 wrote:

> Hi, please look at this code (I discuss the problem below):
> --------------------------------------------------------------------
> import std.stdio;
> 
> void main() {
>      string s = "test";
> 
>      writefln("s: ", typeid(typeof(s)));
> 
>      writefln("s[0]: ", typeid(typeof(s[0])));
>      //(s[0]) = 'a'; //Error: s[0] is not mutable
> 
>      foreach (ref char c; s) {
>          writefln("c: ", typeid(typeof(s[0])));
>          c = 'a';
>          break;
>      }
> 
>      writefln(s);
> 
>      writefln(typeid(invariant(char[]))); //invariant invariant char[]
>      writefln(typeid(invariant(char)[])); //invariant char[]
>      invariant(char)[] f;
>      writefln(typeid(typeof(f[0]))); //char ????
>      //what I expected: invariant(char) like:
>      writefln(typeid(invariant(char))); //invariant char
> }
> --------------------------------------------------------------------
> s: const char[]
> s[0]: char
> c: char
> aest
> invariant invariant char[]
> invariant char[]
> char
> invariant char
> 
> I can use c (within the foreach) to change elements of the string s.
> Exactly: I can assign single values of type char to each element of the 
> array of invariant chars. (Or I can break after doing so for the n-th 
> element)
> 
> But I can't use s[0] to do the same for exactly one element.
> 
> The funny thing is: c and s[0] are of the same type ! Both are char.
> 
> Lets take a look at string, which is an alias for invariant(char)[].
> What this means: A mutable array of chars (each char can be reassigned, 
> because of the brackets).
> "invariant(char) c;" means: You can assign an other value of type char to c.
> (PS: On the docs it should be mentioned, that invariant(builtintype) / 
> const(builtintype) is useless and behaves like: builtintype)
> not to be confused with "invariant char c;" which means: the value of c 
> will never change and you can't assign an other value to c.
> 
> s[0] is of type char, but to make clear, that you can't assign an other 
> char to it, it should be: "invariant char". Then it would be clear, that 
> you can't do: s[0] = 'a';
> 
> invariant(char[]) -> invariant invariant char[]
>                          |        \--------/  |
>                           \------------------/
> 
> invariant(char)[] -> invariant char[]
>                           \------/
> 
> So one element would mean the [] to be removed, and the rest is:
> invariant(char) -> invariant char
> 
> But as I pointed out above, this is not the same.
> And that's the reason invariant(char) is always translated to: char
> Because it behaves as if it was a normal char (also said above), if you do:
> invariant(char) c;
> You can assign to it, as if it was just: char c:
> c = 'a';
> 
> Now take exactly one element of a string,
> you get an invariant(char) -> char
> But you can't assign an other value of type char to it.
> 
> So for arrays it's different, although the type is the same (char).
> 
> Daniel

Seems like a bug to me. Here is a simpler example:

import std.stdio;
void main() {
     invariant(char)[] s = "test";
     foreach (ref char c; s) {
         c = 'a';
         break;
     }
     writefln(s);
}

The output is 'aest' which means that the invariant char array got changed.

-- 
Derek
(skype: derek.j.parnell)
Melbourne, Australia
6/07/2007 11:56:27 AM



More information about the Digitalmars-d mailing list