本技巧属于 编程中的 奇技淫巧。
我们封装接口时 经常 会涉及到 类似以下的代码,它的init 函数 对接口涉及到的结构体 进行初始化 ,通常是清0,
大多数情况下我们会用memset 操作结构体 ,如下。
typedef struct tag_interface
{
int a;
int b;
char sztest[10];
} interface_t, * interface_p;
void interface_init(interface_t * a)
{
if(!a)
return ;
memset(a,0,sizeof(interface_t ));
}
int _tmain(int argc, _TCHAR* argv[])
{
interface_t test;
interface_init(&test);
return 0;
}
0000000000401122 <interface_init>:
401122: 55 push %rbp
401123: 48 89 e5 mov %rsp,%rbp
401126: 48 83 ec 10 sub $0x10,%rsp
40112a: 48 89 7d f8 mov %rdi,-0x8(%rbp)
40112e: 48 83 7d f8 00 cmpq $0x0,-0x8(%rbp)
401133: 74 18 je 40114d <interface_init+0x2b>
401135: 48 8b 45 f8 mov -0x8(%rbp),%rax
401139: ba 08 00 00 00 mov $0x8,%edx
40113e: be 00 00 00 00 mov $0x0,%esi
401143: 48 89 c7 mov %rax,%rdi
401146: e8 e5 fe ff ff callq 401030 <memset@plt>
40114b: eb 01 jmp 40114e <interface_init+0x2c>
40114d: 90 nop
40114e: c9 leaveq
40114f: c3 retq
其实我们可以利用 程序bss段中的变量 编译器自动初始化为0 的特性。
对比一下代码
typedef struct tag_interface
{
int a;
int b;
char sztest[10];
} interface_t, * interface_p;
void interface_init(interface_t * a)
{
static interface_t s_t;
if(!a)
return ;
*a = s_t; // 没有使用memset
}
int _tmain(int argc, _TCHAR* argv[])
{
interface_t test;
interface_init(&test);
return 0;
}
401102: 55 push %rbp
401103: 48 89 e5 mov %rsp,%rbp
401106: 48 89 7d f8 mov %rdi,-0x8(%rbp)
40110a: 48 83 7d f8 00 cmpq $0x0,-0x8(%rbp)
40110f: 74 24 je 401135 <interface_init+0x33>
401111: 48 8b 4d f8 mov -0x8(%rbp),%rcx
401115: 48 8b 05 24 2f 00 00 mov 0x2f24(%rip),%rax # 404040 <s_t.2367>
40111c: 48 8b 15 25 2f 00 00 mov 0x2f25(%rip),%rdx # 404048 <s_t.2367+0x8>
401123: 48 89 01 mov %rax,(%rcx)
401126: 48 89 51 08 mov %rdx,0x8(%rcx)
40112a: 8b 05 20 2f 00 00 mov 0x2f20(%rip),%eax # 404050 <s_t.2367+0x10>
401130: 89 41 10 mov %eax,0x10(%rcx)
401133: eb 01 jmp 401136 <interface_init+0x34>
401135: 90 nop
401136: 5d pop %rbp
401137: c3 retq
达到同样的效果,节省了函数调用的开销。