达夫设备(Duff's Device)的详细说明
前几天在网上看见了一段代码,叫做“Duff's Device”,后经验证它曾出现在Bjarne的TC++ PL里面:
代码的结构显得非常巧妙,把一个switch语句和一个do-
怎么办?先看看它的汇编代码吧,
gcc -S send.cpp
粗略地一看,汇编代码都已经上百行了,而且里面还有一个跳转表,
既然直接来太麻烦,那就用简便一点的方法吧:
n的值 | 程序输出 |
0 | 0 1 2 3 |
1 | 1 2 3 |
2 | 2 3 0 1 2 3 |
3 | 3 0 1 2 3 0 1 2 3 |
其他 | (无输出) |
这 下终于弄清楚了。原来,那段代码的主体还是do-
忙活了几个小时,终于明白这段代码是怎么回事了。回想一下,
至于它是怎么正确编译并运行的,这需要去研究一下C编译器,
这段代码的确很简洁,也是正确的,
它的思路是这样的:把原数组以8个int为单位分成若干个小组,
总结:像达夫设 备这样的代码,从语言的角度来看,我个人觉得不值得我们借鉴。
附
void duff_memcpy( char* to, char* from, size_t count )
{
size_t n = (count+7)/8;
switch( count%8 )
{
case 0: do{ *to++ = *from++;
case 7: *to++ = *from++;
case 6: *to++ = *from++;
case 5: *to++ = *from++;
case 4: *to++ = *from++;
case 3: *to++ = *from++;
case 2: *to++ = *from++;
case 1: *to++ = *from++;
}while(--n>0);
}
}
duff’s device,是用Tom Duff的名字来命名的。很有名的一个东西,用来优化拷贝的, 据说和Rob Pike此牛还有点儿关系~!不过注意,原始的duff’s device中的to可是不变的, 因为它指向一个映射到内存的寄存器。
这是个很棒的迂回循环展开法, 由 Tom Duff 在 Lucasfilm 时所设计。它的 ``传统" 形态, 是用来复制多个字节:
register n = (count + 7) / 8; /* count > 0 assumed */
switch (count % 8)
{
case 0: do { *to = *from++;
case 7: *to = *from++;
case 6: *to = *from++;
case 5: *to = *from++;
case 4: *to = *from++;
case 3: *to = *from++;
case 2: *to = *from++;
case 1: *to = *from++;
} while (--n > 0);
}
register n = (count + 7) / 8; /* count > 0 assumed */
switch (count % 8)
{
case 0: do { *to = *from++;
case 7: *to = *from++;
case 6: *to = *from++;
case 5: *to = *from++;
case 4: *to = *from++;
case 3: *to = *from++;
case 2: *to = *from++;
case 1: *to = *from++;
} while (--n > 0);
}
这里 count 个字节从 from 指向的数组复制到 to 指向的内存地址 (这是个内存映射的输出寄存器, 这也是为什么它没有被增加)。它把 swtich 语句和复制 8 个字节的循环交织在一起, 从而解决了剩余字节的处理问题 (当 count 不是 8 的倍数时)。相信不相信, 象这样的把 case 标志放在嵌套在 swtich 语句内的模块中是合法的。当他公布这个技巧给 C 的开发者和世界时, Duff 注意到 C 的 swtich 语法, 特别是 ``跌落" 行为, 一直是被争议的, 而 ``这段代码在争论中形成了某种论据, 但我不清楚是赞成还是反对"。