此段代码可以用于以下情形:
两个对象的类名不同,但是内部字段名称相同,可以用以下函数进行浅拷贝。
在类很多的时候,省事不少。
void copyObj(SRC,DEST)(ref SRC src,ref DEST dest)
{
foreach (i, type; typeof(SRC.tupleof)) {
__traits(getMember, dest, SRC.tupleof[i].stringof) = __traits(getMember, src, SRC.tupleof[i].stringof);
}
}
两个对象的类名不同,但是内部字段名称相同,可以用以下函数进行浅拷贝。
在类很多的时候,省事不少。
void copyObj(SRC,DEST)(ref SRC src,ref DEST dest)
{
foreach (i, type; typeof(SRC.tupleof)) {
__traits(getMember, dest, SRC.tupleof[i].stringof) = __traits(getMember, src, SRC.tupleof[i].stringof);
}
}
我在d的官方论坛发起了讨论,这里的主要注意点是,这里的类型识别是在编译时处理的,和java的反射完全不同,java反射是运行时的。所以这里不能使用运行时变量。
论坛里的网友指出了一个改进SRC.tupleof[i].stringof可以定义为一个变量,但是类型必须问enum:
void copyObj(SRC,DEST)(ref SRC src,ref DEST dest)
{
foreach (i, type; typeof(SRC.tupleof)) {
enum name = SRC.tupleof[i].stringof;
__traits(getMember, dest, name) = __traits(getMember, src, name);
}
}
{
foreach (i, type; typeof(SRC.tupleof)) {
enum name = SRC.tupleof[i].stringof;
__traits(getMember, dest, name) = __traits(getMember, src, name);
}
}
为什么必须为enum,我也不是很清楚,留待深入学习后再理解。
最后,国外网友提供了一种很特别的方法,这里是利用了union的特性,很聪明的做法:
struct Foo { public: string a; int b; } class FooClass { public: union { struct { string a; int b; }; Foo foo; } } void main() { Foo f = Foo("a", 10); FooClass c = new FooClass(); c.foo = f; writefln("%s %s", c.a, c.b); }
我现在发现学习一门新语言的好方法,也许只是对于我有效,不过有兴趣的也可以试试:首先通读语言参考,了解里面有哪些特性,知道有什么用法,没必要完全理解、记住。然后,用这门语言开发一个项目,识别哪个部分应该用那种特性,哪个组件应该选用哪个库,在实践中用到什么学习什么,这个项目开发完了,对这门语言的理解是看书做习题无法相比的。然后在选择自己喜欢的部分深入学习。