除法
约定
- 两个无符号相除,结果仍然是无符号.
- 两个有符号相除,结果是有符号.
- 有符号数和无符号数混除,结果是无符号的.
基本概念
在C语言和其他多数高级语言中,对整数除法规定为向零取整.
- 向下取整
取得往负无穷方向接近x的整数值.
向下取整的除法,当除数为2的幂时,可以直接用带符号右移指令(sar)来完成.
但是,向下取整存在一个问题:
$$
⌊\frac{-a}{b}⌋ \not= -⌊\frac{a}{b}⌋ (假设\frac{a}{b}结果不为整数)
$$
- 向上取整
取得往正无穷方向接近x的整数值.
向上取整也存在一个问题:
$$
⌈\frac{-a}{b}⌉ \not= -⌈\frac{a}{b}⌉ (假设\frac{a}{b}结果不为整数)
$$
- 向零取整
取得往0方向最接近x的整数值.
向零取整的除法满足:
$$
[\frac{-a}{b}] = [\frac{a}{-b}] =-[\frac{a}{b}]
$$
无符号除法
除数为变量
除数为变量的时候是没有优化,只能使用除法指令,这里的变量是指类似argc这类在编译期间不能计算的变量.
1
2
3
4
5
|
int main(unsigned int argc, char* argv[])
{
printf("%d \r\n", 10 / argc);
return 0;
}
|
data:image/s3,"s3://crabby-images/f5a90/f5a90bbae69637a4a86273c8aae81bce9728551d" alt="无符号除数为变量"
除数为2的幂
1
2
3
4
5
|
int main(unsigned int argc, char* argv[])
{
printf("%d", argc / 4);
return 0;
}
|
data:image/s3,"s3://crabby-images/81cd5/81cd5f07eeb19417ab97575f070a86494eb5b567" alt="无符号除数为2的幂"
除数为非2的幂
MagicNumber无进位
1
2
3
4
5
|
int main(unsigned int argc, char* argv[])
{
printf("%d \r\n", argc / 3);
return 0;
}
|
data:image/s3,"s3://crabby-images/f7176/f7176322266191b30d7cf2aceb9273e1c221c175" alt="无符号除数为2的幂Magic无进位1"
推导
data:image/s3,"s3://crabby-images/69612/696126d1c70193889b6acf60a5266ef490702cc9" alt="无符号除数为2的幂Magic无进位2"
特征
1
2
3
|
mov eax, MagicNumber
mul 被除数
shr edx, n ;这条指令可无
|
还原
1
2
|
除数 = 2^(32 + n) / MagicNumber
结果向上取整
|
MagicNumber有进位
1
2
3
4
5
|
int main(unsigned int argc, char* argv[])
{
printf("%d", argc / 7);
return 0;
}
|
data:image/s3,"s3://crabby-images/24714/24714b3183576a42aabec157ca18014595ecf3c1" alt="无符号除数为2的幂Magic有进位1"
推导
data:image/s3,"s3://crabby-images/681fe/681fe59471bf0eac54da87004c308340df2385cc" alt="无符号除数为2的幂Magic有进位2"
data:image/s3,"s3://crabby-images/e9855/e985552adffdf99a2c4c57e61bb98305715635bc" alt="无符号除数为2的幂Magic有进位3"
特征
1
2
3
4
5
6
7
8
9
|
#乘减移加移
mov reg, 被除数
mov eax, MagicNumber
mul reg
sub reg, edx
shr reg, 1
add reg, edx
shr reg, A; 这句可能没有
;此后直接使用reg的值, eax弃而不用
|
还原
1
2
3
|
统计右移次数,n = 32 + 1 + A
除数 = 2^n / (2^32 + MagicNumber)
结果向上取整
|
有符号除法
除数为正2的幂
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d \r\n", argc / 8);
return 0;
}
|
data:image/s3,"s3://crabby-images/f0ed0/f0ed084bc3e77cc32268f3f3c8ed3878f62f6d76" alt="有符号除数为正2的幂1"
推导
data:image/s3,"s3://crabby-images/b7e26/b7e269c7e6ce1fa353b3c6b0719efd6e6afaf3d7" alt="有符号除数为正2的幂2"
特征
1
2
3
4
5
|
mov eax, 被除数
cdq
and edx, 2 ^ n - 1
add eax, edx
sar eax, n
|
还原
除数为正非2的幂
MagicNumber为正
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d \r\n", argc / 5);
return 0;
}
|
data:image/s3,"s3://crabby-images/7f9e0/7f9e090201e25906cec392c8a801b94652a7e88f" alt="除数为正非2的幂Magic为正1"
推导
data:image/s3,"s3://crabby-images/47d06/47d060cc8aa18737cff92fe3929ca717b177e1a7" alt="除数为正非2的幂Magic为正2"
特征
1
2
3
4
5
6
7
8
9
10
11
|
mov reg, 被除数
mov eax, MagicNumber ;(MagicNumber <= 0x7FFFFFFF)
imul reg
sar edx, n ;没有这条指令则指数为32
mov reg, edx
shr reg, 1Fh
add edx, reg
;此后直接使用edx的值
MagicNumber为正
乘法和移位之间无调整
|
还原
1
2
|
除数 = 2^(32 + n) / MagicNumber
结果向上取整
|
MagicNumber为负
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d", argc / 73);
return 0;
}
|
data:image/s3,"s3://crabby-images/346ba/346ba1521ce56135b6fbff18cb632c10ebd47eca" alt="除数为正非2的幂Magic为负1"
推导
1
2
3
4
5
6
7
8
|
在16位机器上计算无符号数0x8086与有符号数A,保证计算结果正确.
0x8086用imul指令会被当做负数,用补码进行计算
计算机计算= - ~0x8086 * A
= - (0x10000 - 0x8086) * A
=(0x8086 - 0x10000) * A
=0x8086 * A - 0x10000 * A
而我们想要得到0x8086 * A,故计算出来的结果应加上0x10000 * A
对应上图中的add edx, ecx
|
特征
1
2
3
4
5
6
7
8
9
10
11
12
|
mov reg, 被除数
mov eax, MagicNumber ;(MagicNumber > 7FFFFFFFh)
imul reg
add edx, reg
sar edx, n ;没有这条指令则指数为32
mov reg, edx
shr reg, 1Fh
add edx, reg
;此后直接使用edx的值
MagicNumber为负
乘法和移位之间有加调整
|
还原
1
2
|
除数 = 2^(32 + n) / MagicNumber
结果向上取整
|
除数为负2的幂
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d", argc / -8);
return 0;
}
|
data:image/s3,"s3://crabby-images/aba76/aba76c1dd0759f2af57253e0b8492486b9af2cef" alt="除数为负2的幂1"
推导
data:image/s3,"s3://crabby-images/b7e26/b7e269c7e6ce1fa353b3c6b0719efd6e6afaf3d7" alt="有符号除数为正2的幂2"
特征
1
2
3
4
5
6
|
mov eax, 被除数
cdq
and edx, 2^n - 1
add eax, edx
sar eax, n
neg eax
|
还原
除数为负非2的幂
MagicNumber为正
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d", argc / -73);
return 0;
}
|
data:image/s3,"s3://crabby-images/ac872/ac87258bbcb5f9696d4dbac8bea19b85dce89b6d" alt="除数为负非2的幂Magic为正1"
推导
1
2
3
4
5
6
7
8
9
10
11
12
13
|
在16位机器上计算无符号数0x8086与有符号数A,保证计算结果正确.
0x8086用imul指令会被当做负数,用补码进行计算
计算机计算= - ~0x8086 * A
= - (0x10000 - 0x8086) * A
=(0x8086 - 0x10000) * A
=0x8086 * A - 0x10000 * A
假设上面0x8086为MagicNumber
我们的MagicNumber为正,故-0x8086假定是MagicNumber
所以上面的计算机计算结果应加上符号,变为
-(0x8086 * A - 0x10000 * A)
=-0x8086 * A + 0x10000 * A
但我们想得到-0x8086 * A,故应再加上0x10000 * A
对应上图中的sub edx, ecx
|
特征
1
2
3
4
5
6
7
8
9
10
11
12
|
mov reg, 被除数
mov eax, MagicNumber
imul reg
sub edx, reg
sar edx, n ;没有这句, 指数为32
mov reg, edx
shr reg, 1Fh
add edx, eax
;此后使用edx的值
MagicNumber为正
乘法和移位之间有减调整
|
还原
1
2
3
4
5
6
7
8
9
10
11
12
|
先将MagicNumber求补(取反+1)拿到原来的MagicNumbeSrc
除数的绝对值 = 2^(32 + n) /MagicNumberSrc
结果向上取整
除数为负
由于MagicNumber为正,故求补得到的MagicNumberSrc为负
注意参与计算的MagicNumSrc应当做无符号数进行计算
本例中MagicNumber = 0x1F8FC7E3
MagicNumberSrc = 0xE070381D, 对应的无符号十进制数为3765450781
故除数的绝对值 = 2^38 / 3765450781
结果向上取整为73
再添加负号就还原了除数为-73
|
MagicNumber为负
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d", argc / -5);
return 0;
}
|
data:image/s3,"s3://crabby-images/5310c/5310c82dc361d1e241d71f2fd0d9a87b6da7cd5c" alt="除数为负非2的幂Magic为负1"
推导
1
2
3
4
|
A/(-C),C>0
-> -A * M >> n
-> A * -M >> n
故汇编显示的MagicNumber为原来MagicNumber求补后的结果.
|
特征
1
2
3
4
5
6
7
8
9
10
11
|
mov reg, 被除数
mov eax, MagicNumber
imul reg
sar edx, n ;没有这句, 指数为32
mov reg, edx
shr reg, 1Fh
add edx, reg
;此后使用edx的值
MagicNumber为负
乘法和移位之间没有调整
|
还原
1
2
3
4
|
先将MagicNumber求补(取反+1)拿到原来的MagicNumbeSrc
除数的绝对值 = 2^(32 + n) /MagicNumberSrc
结果向上取整
除数为负
|
取模
基础
模的符号与被除数相同.
除数为变量,无优化,只有当除数为常量时,才有优化空间.
无符号取模
除数为2的幂
1
2
3
4
5
|
int main(unsigned int argc, char* argv[])
{
printf("%d\r\n", argc % 8);
return 0;
}
|
data:image/s3,"s3://crabby-images/f1aa0/f1aa05c82d56cf4dcd24fe2fc8bf86081778ebb2" alt="无符号除数为2的幂"
推导
1
2
|
对2的n次方取模,实际上是取该数值的二进制低n位
如10进制中,321%100,我们取321的低2位,就可以知道余数为21
|
特征
1
2
|
mov reg, 被除数
and reg, 2 ^ n - 1
|
还原
除数为非2的幂
1
2
3
4
5
|
int main(unsigned int argc, char* argv[])
{
printf("%d\r\n", argc % 7);
return 0;
}
|
data:image/s3,"s3://crabby-images/2d289/2d289e87f2d50d9fa741f969026c5c853724c001" alt="无符号除数为非2的幂"
推导
还原
1
2
|
可以通过上面的除法部分得到除数
也可以通过中间商乘以除数的部分得到除数
|
有符号取模
除数为正2的幂
带分支
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc % 8);
return 0;
}
|
data:image/s3,"s3://crabby-images/e7981/e798138d20e9792ff04392efe1b264f22baf54d3" alt="除数为正2的幂带分支1"
data:image/s3,"s3://crabby-images/f0a1a/f0a1ac467ea0975be0c53850133f43b366da32e6" alt="除数为正2的幂带分支2"
推导
1
2
3
4
5
6
7
8
|
mov eax, [esp+argc] ;将被除数给eax
and eax, 80000007h ;取被除数的符号位和低n位
jns short loc_401010 ;如果被除数为正,跳走,直接使用eax
dec eax ;处理负数取模为0的情况 减1后为全f,如果取模不为0的情况是不需要dec和inc指令
or eax, 0FFFFFFF8h ;被除数为负,需要将模的结果高位填补为1
inc eax ;处理负数取模为0的情况 上面减后全为f,此处加1,调整为0
loc_401010:
;使用eax
|
特征
1
2
3
4
5
6
7
8
|
mov reg, 被除数
and reg, 0x8000000(2^n - 1)
jns LABLE
dec reg
or reg, 0xFFFFFFF(2^n)
inc reg
LABLE:
;使用reg
|
还原
不带分支
见除数为负2的幂.
除数为正非2的幂
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc % 7);
return 0;
}
|
data:image/s3,"s3://crabby-images/b0bba/b0bba7a945c172dcee21d779be263dbae03a68e1" alt="除数为正非2的幂"
推导
还原
1
2
|
可以通过上面的除法部分得到除数
也可以通过中间商乘以除数的部分得到除数
|
除数为负2的幂
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc % -8);
return 0;
}
|
data:image/s3,"s3://crabby-images/75e79/75e796096813024193d235590044733e4cafc2d9" alt="除数为负2的幂"
推导
1
2
|
对2的n次方取模,实际上是取该数值的二进制低n位
模的符号与被除数相同
|
还原
1
2
|
上图中的7为2^n - 1
除数 = 2^n
|
除数为负非2的幂
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc % -7);
return 0;
}
|
data:image/s3,"s3://crabby-images/e688b/e688ba45504fe9289b473740f41dd279cf471c57" alt="除数为负非2的幂"
推导
还原
1
2
|
可以通过上面的除法部分得到除数
也可以通过中间商乘以除数的部分得到除数
|
三目运算
基础
表达式1 ? 表达式2 : 表达式3,只有当表达式2和表达式3为常量时,才可以优化,否则和if分支语句产生的代码一样.
低版本优化
原型:
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc == 0 ? 0 : -1);
return 0;
}
|
data:image/s3,"s3://crabby-images/4cb1b/4cb1b035167ea9c4705e1bc796323695a01324c4" alt="低版本原型"
特征
1
2
3
|
mov reg, 表达式1中的第1项值
neg reg
sbb reg, reg
|
演变1:
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc == 100 ? 6 : 2);
return 0;
}
|
data:image/s3,"s3://crabby-images/a237f/a237fe672c89820638d935cb4490def95b5b7a65" alt="低版本演变一"
特征
1
2
3
4
5
6
|
mov reg,表达式1第一项值
sub reg, 表达式1第二项值
neg reg
sbb reg, reg
and rl, 表达式3-表达式2
add reg, 表达式2
|
演变2:
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc > 100 ? 2 : 6);
return 0;
}
|
data:image/s3,"s3://crabby-images/8938a/8938a39ef0766b916564550e1c9ccfe93bfdef1a" alt="低版本演变二"
高版本优化
1
2
3
4
5
|
int main(int argc, char* argv[])
{
printf("%d\r\n", argc >= 100 ? 5 : 2);
return 0;
}
|
data:image/s3,"s3://crabby-images/71a04/71a046120fc1e3b35037fd43426dd7ca3714f3c8" alt="高版本优化"
流程控制语句
IF
单分支结构
1
2
3
4
5
6
7
8
|
int main(int argc, char* argv[])
{
if (argc > 10)
{
printf("argc > 10\r\n");
}
return 0;
}
|
data:image/s3,"s3://crabby-images/744b3/744b37c31a31ab3c593e5a176b48b6e599538fcc" alt="IF单分支结构"
特征
1
2
3
4
5
6
|
IF_BEGIN
jxx IF_END
.
. ;中间为条件体
. ;注意语句上面无跳转
IF_END
|
双分支结构
代码外提:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
int main(int argc, char* argv[])
{
if (argc > 10)
{
printf("argc > 10\r\n");
}
else
{
printf("argc <= 10\r\n");
}
system("pause");
return 0;
}
|
data:image/s3,"s3://crabby-images/06e6e/06e6e4ab82adda4ab3b4273b2626d0e33299c5f6" alt="IF双分支结构代码外提"
特征
1
2
3
4
5
6
7
8
|
IF_BEGIN
jxx ELSE_BEGIN
...
IF_END
jmp ELSE_END
ELSE_BEGIN
...
ELSE_END
|
代码内提:
1
2
3
4
5
6
7
8
9
10
11
12
|
int main(int argc, char* argv[])
{
if (argc > 10)
{
printf("argc > 10\r\n");
}
else
{
printf("argc <= 10\r\n");
}
return 0;
}
|
data:image/s3,"s3://crabby-images/5c66d/5c66d5935c57011b1b0065550ca96b85ebf2d2d7" alt="IF双分支结构代码内提"
多分支结构
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
int main(int argc, char* argv[])
{
if (argc > 10)
{
printf("argc > 10\r\n");
}
else if (argc < 10)
{
printf("argc < 10\r\n");
}
else
{
printf("argc == 10\r\n");
}
system("pause");
return 0;
}
|
data:image/s3,"s3://crabby-images/5da25/5da2563d5b8578e28d2fa05a2e18e4f03b3ade6d" alt="IF多分支结构"
特征
1
2
3
4
5
6
7
8
9
10
11
12
13
|
IF_BEGIN
jxx ELSE_IF_BEGIN
...
IF_END
jmp ELSE_END
ELSE_IF_BEGIN
jxx ELSE_BEGIN
...
ELSE_IF_END
jmp ELSE_END
ELSE_BEGIN
...
ELSE_END
|
SwitchCase
Case语句<=3
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
int main(int argc, char* argv[])
{
switch (argc)
{
case 0:
printf("argc == 0\r\n");
case 1:
printf("argc == 1\r\n");
break;
case 2:
printf("argc == 2\r\n");
break;
default:
printf("default\r\n");
}
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/8cfcc/8cfcc7f3a9dc58e9f80286c6aced7dd0610dfa0e" alt="Case1Debug"
Release
data:image/s3,"s3://crabby-images/29d2b/29d2b7739692b79e40a3ffde2144dd1a173f14bb" alt="Case1Release"
特征
1
2
|
一个个比较,中间无代码块
这样的话,匹配跳转成功时,没有break语句会继续向下执行
|
case语句>3,且每两个Case值之间的差值<=6
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
int main(int argc, char* argv[])
{
switch (argc)
{
case 0:
printf("argc == 0\r\n");
case 1:
printf("argc == 1\r\n");
break;
case 2:
printf("argc == 2\r\n");
break;
case 8:
printf("argc == 8\r\n");
break;
default:
printf("default\r\n");
}
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/6236b/6236b9d2cd545114f60cda65d37247440b82a3b3" alt="Case2Debug"
Release
data:image/s3,"s3://crabby-images/40f8c/40f8c30e7a9fe35971e95c6192e5f914897e928c" alt="Case2Release"
特征
1
2
|
编译器会给所有case情况做一个表
当提供的序号大于最大索引时,直接跳转到DEFAULT或SWITCH_END
|
case语句>3,最大case值-最小case值<=255
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
int main(int argc, char* argv[])
{
switch (argc)
{
case 0:
printf("argc == 0\r\n");
case 46:
printf("argc == 46\r\n");
break;
case 55:
printf("argc == 55\r\n");
break;
case 77:
printf("argc == 77\r\n");
break;
default:
printf("default\r\n");
}
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/41ee0/41ee088f147a2852f9b4120fde91ec3b527fb00f" alt="Case3Debug"
Release
data:image/s3,"s3://crabby-images/d1bc2/d1bc21f526435a49b9742c3049493660b99cb01e" alt="Case3Release"
特征
1
2
3
|
编译器会生成两个表
下标表:记录对应case地址表的下标
case地址表:记录case语句块的地址
|
还原
data:image/s3,"s3://crabby-images/60517/6051777df10f895fa2815fdbfd218f4eda7e0b0c" alt="Case3还原"
case语句>3,最大case值-最小case值>255
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
int main(int argc, char* argv[])
{
switch (argc)
{
case 0:
printf("argc == 0\r\n");
case 46:
printf("argc == 46\r\n");
break;
case 555:
printf("argc == 555\r\n");
break;
case 777:
printf("argc == 777\r\n");
break;
case 888:
printf("argc == 888\r\n");
break;
default:
printf("default\r\n");
}
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/e49ba/e49ba611812c498e9be058b7d9b1576f6fa065a5" alt="Case4Debug"
Release
data:image/s3,"s3://crabby-images/069ac/069ac2a18d3adbe4593f80bda0546e70be372406" alt="Case4Release"
特征
1
2
|
二分优化
注意:会进行二分优化和前面三种混合优化
|
注意
break语句,对应的跳转一定是跳向SWITCH_END.
循环语句
DoWhile
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
int main(int argc, char* argv[])
{
int nSum = 0;
int n = 1;
do
{
nSum += n;
n++;
} while (n <= 100);
printf("%d\r\n", nSum);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/c6c90/c6c905888ce7448b29a9ff68cff788dd6d742581" alt="DoWhileDebug"
Release
data:image/s3,"s3://crabby-images/bb7f7/bb7f769c3c673229e5852119532ee3c95831b7a3" alt="DoWhileRelease"
特征
1
2
3
4
5
6
|
Do_Begin
...
...
...
jxx Do_Begin
Do_End
|
While
1
2
3
4
5
6
7
8
9
10
11
12
13
|
int main(int argc, char* argv[])
{
int nSum = 0;
while (argc <= 100)
{
nSum += argc;
argc++;
}
printf("%d\r\n", nSum);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/530eb/530eb2b08bf19536326b3a0d8036bad8b455007c" alt="WhileDebug"
特征
1
2
3
4
5
6
7
|
While_Begin
jxx While_End
...
...
...
jmp While_Begin
While_End
|
Release
data:image/s3,"s3://crabby-images/64efa/64efac75b31418e5946ff0e52c29371cfd0cab96" alt="WhileRelease"
特征
1
2
3
|
这个地方While循环会被优化成DoWhile循环
多了一个跳转提前进行判断
注意:跳转的判定条件和循环的判定条件具有相关性时,才可还原为While循环
|
For
1
2
3
4
5
6
7
8
9
10
11
12
|
int main(int argc, char* argv[])
{
int nSum = 0;
for (int i = 0; i <= 100; i++)
{
nSum += i;
}
printf("%d\r\n", nSum);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/071e4/071e41f7f4bd3e1e6b8d6a5ae65ec86babcde16b" alt="ForDebug"
特征
1
2
3
4
5
6
7
8
9
10
11
12
|
For_Init
...
jmp For_Cmp
For_Step
...
For_Cmp
jxx For_End
For_Body
...
...
jmp For_Step
For_End
|
Release
data:image/s3,"s3://crabby-images/1e994/1e9941800a4db0ac91c26986ac5e895fe40198dc" alt="ForRelease"
特征
注意
//Break语句
循环语句中,跳转循环体,可识别为break语句
//Continue语句
Release版,经常会将Continue语句优化为If_Else双分支结构
函数调用方式
_cdecl
1
2
3
|
C\C++默认的调用方式
调用方平栈
参数:从右到左以此入栈
|
_stdcall
_fastcall
1
2
|
被调方平栈
参数:前两个参数由寄存器传递(VS编译器用edx, ecx),其余参数通过堆栈传递
|
变量相关
全局变量
特征:
- 所在地址为数据区,生命周期与所在模块一致.
- 使用立即数访问.
全局变量初始化:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
int foo()
{
return 1234;
}
int g_nTest = foo();
int main(int argc, char* argv[])
{
printf("g_nTest = %d\r\n", g_nTest);
system("pause");
return 0;
}
|
定位初始化位置
data:image/s3,"s3://crabby-images/2cb4c/2cb4c7104beca20fa6fb7c9b1c78a15dd079a5f0" alt="定位初始化位置1"
data:image/s3,"s3://crabby-images/f81da/f81da3d7071ea698ed881a9beef10eed4a75511b" alt="定位初始化位置2"
注意:高版本_cinit函数被内联了.
data:image/s3,"s3://crabby-images/031f2/031f20f1a06bbf51fb315d1f464d1340fac88b04" alt="定位初始化位置3"
data:image/s3,"s3://crabby-images/dc525/dc525e8d597c0debc3e43ff3dbed1b53bbcc5fc6" alt="定位初始化位置4"
data:image/s3,"s3://crabby-images/f642f/f642fe1984ebe67a6f04c2ca0f06c3ca26ccd843" alt="定位初始化位置5"
data:image/s3,"s3://crabby-images/01d98/01d98d5ea21bb420c9641f8a72e7f3e080bc5797" alt="定位初始化位置6"
data:image/s3,"s3://crabby-images/59e7b/59e7b91afdea6acc79104feb0fb5188d58aab3f6" alt="定位初始化位置7"
使用交叉引用
data:image/s3,"s3://crabby-images/d0211/d02110b3cb58e54c0373da3868c44bf537911bd6" alt="使用交叉引用1"
data:image/s3,"s3://crabby-images/5a25b/5a25b7bd908b420ab7b4672ff28be875b5901d10" alt="使用交叉引用2"
data:image/s3,"s3://crabby-images/9e87a/9e87a9004c5bd6efeb72e21f788a3dc643983472" alt="使用交叉引用3"
data:image/s3,"s3://crabby-images/cacde/cacde00680beaf84102f70e12356d76c37710e37" alt="使用交叉引用4"
局部变量
特征:
- 所在地址为栈区,生命周期与所在的函数作用域一致.
- 使用ebp或esp间接访问.
局部静态变量
C++语法规定局部静态变量只被初始化一次.
1
2
3
4
5
6
7
8
|
int main(int argc, char* argv[])
{
static float f = 3.14f / argc;
printf("f = %f\r\n", f);
system("pause");
return 0;
}
|
VC6.0Release版
data:image/s3,"s3://crabby-images/276dc/276dc7973db4155008a04aa73351ce7a2b7bf0eb" alt="局部静态变量1"
高版本(如VS2019)将这个标志位放到了Tls中.
堆变量
在C\C++中,使用malloc与new实现堆空间的申请.
X87浮点指令集
VC6.0 ~ VS2013.
特征
- 使用80位浮点协处理器处理浮点运算.
- 浮点协处理器内部为栈结构.
data:image/s3,"s3://crabby-images/f5380/f5380f9fe90c16a3f52e5a9ecfe422ed166967f8" alt="X87浮点指令集特征"
运算过程
data:image/s3,"s3://crabby-images/1c9b9/1c9b9b0ee37f9406ce08dc9ec12a49a4236723b9" alt="X87浮点指令集运算过程"
指令
data:image/s3,"s3://crabby-images/898d3/898d3982b1cc71add1234d256a266afe2fc8c61c" alt="X87浮点指令集指令"
探测浮点标志位
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
int main(int argc, char* argv[])
{
float f = 3.14f / argc;
if (f == (float)argc) //ZF = 1
{
printf("f == (float)argc\r\n");
}
else //ZF = 0
{
printf("f != (float)argc\r\n");
}
if (f > 1.23f) //SF = 0 and ZF = 0
{
printf("f > 1.23f\r\n");
}
else //SF = 1 or ZF = 1
{
printf("f <= 1.23f\r\n");
}
if (f < 1.23f) //SF = 1
{
printf("f < 1.23f\r\n");
}
else
{
printf("f >= 1.23f\r\n");
}
system("pause");
return 0;
}
|
参考资料
Intel奔腾指令速查手册.doc
多媒体指令集
MMX
VS中使用MMX指令,包含头文件mmintrin.h
特征如下:
- 有8个64位寄存器(MM0~MM7),借用的是FPU原来的8个80位寄存器(st(0)~st(7)),故使用MMX指令后,需要加上一条EMMS指令,用以复位.
- 不再使用栈结构.
- 寄存器在拆分做独立运算时有两种模式.
- 支持并行计算.
指令
详细查看AMD开发手册(卷5).
data:image/s3,"s3://crabby-images/1bfa9/1bfa93abe7b164dcece90ed4096c16537fa4d908" alt="MMX"
SSE
VS中使用SSE指令
1
2
3
4
|
xmmintrin.h //SSE
emmintrin.h //SSE2
pmmintrin.h //SSE3
smmintrin.h //SSE4
|
特征如下:
- 有8个128位独立寄存器(XMM0~XMM7).
- 支持并行计算.
指令
详情查看AMD开发手册(卷4).
data:image/s3,"s3://crabby-images/f73c2/f73c2584cfec7bb4d10feec3e063c5e66f75a1bd" alt="SSE1"
data:image/s3,"s3://crabby-images/9cb08/9cb083243208cc105356852f5ab6385e69ae4614" alt="SSE2"
AVX
VS中使用AVX指令,包头文件immintrin.h
特征如下:
- 扩展之前的8个128位寄存器为8个256位寄存器(YMM0~YMM7).
- 支持并行计算.
指令
相对于SSE指令语法,前缀加v即可.
VS中设置多媒体指令集
data:image/s3,"s3://crabby-images/17653/176530f2a6f5056e99b0cbb53cc344de5af2cc33" alt="VS中设置多媒体指令集"
数组
特性:
连续性:数组内元素连续排列,且互相不会覆盖.
一致性:具备同类业务功能,即类型和业务功能一样.
注意:身高,体重,年龄都是int类型,但数据功能代表的不一样,这种不属于数组,属于结构体.
体现数组特性:
一维数组
1
2
3
4
5
6
7
8
9
10
11
|
int main(int argc, char* argv[])
{
int ary[5] = { 0, 1, 2, 3, 4 };
for (int i = 0; i < 5; i++)
{
printf("%d\r\n", ary[i]);
}
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/fe78a/fe78a5490300921f3afc7ab035be27f455dfc910" alt="一维数组1"
Release
data:image/s3,"s3://crabby-images/58e71/58e715a6bd767c039189c1fb5db1d6b41fc3300e" alt="一维数组2"
高版本(VS2015)Release
data:image/s3,"s3://crabby-images/4375c/4375c836be7097d1b9d22e8ef0d0831ef5ad3431" alt="一维数组3"
数组寻址公式
1
2
3
|
假设数组为ary[n]
ary[x]寻址:
数组首地址 + x * sizeof(type)
|
二维数组
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
int main(int argc, char* argv[])
{
int ary[3][5] = {
{ 1, 2, 3, 4, 5 },
{ 10, 20, 30, 40, 50 },
{ 100, 200, 300, 400, 500 }
};
int x = 0;
int y = 0;
scanf("%d%d", &x, &y);
printf("%d\r\n", ary[x][y]);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/720a3/720a3e52a8d9760a3a40e7a048eef2575319b4fb" alt="二维数组1"
Release
data:image/s3,"s3://crabby-images/cd1f3/cd1f3adb5fe0808b7c4982a5c7796a25abaaf7e8" alt="二维数组2"
数组寻址公式
1
2
3
4
|
假设数组为ary[M][N]
ary[x][y]寻址
Debug版:ary + x * sizeof(type[N]) + y * sizeof(type)
Release版:ary + (x * N + y) * sizeof(type)
|
三维数组
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
int main(int argc, char* argv[])
{
int ary[2][3][5] = {
{
{ 1, 2, 3, 4, 5 },
{ 10, 20, 30, 40, 50 },
{ 100, 200, 300, 400, 500 }
},
{
{ 1, 2, 3, 4, 5 },
{ 10, 20, 30, 40, 50 },
{ 100, 200, 300, 400, 500 }
}
};
int x = 0;
int y = 0;
int z = 0;
scanf("%d%d%d", &x, &y, &z);
printf("%d\r\n", ary[x][y][z]);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/4ea38/4ea3819011ced37e623514f72ab3f5dbbe0f6555" alt="三维数组1"
Release
data:image/s3,"s3://crabby-images/28f3c/28f3c923a8365970c84c58cae13fa2b56f21fa3f" alt="三维数组2"
数组寻址公式
1
2
3
4
|
假设数组为ary[M][N][K]
ar[x][y][z]寻址
Debug版:ary + x * sizeof(type[N][K]) + y * sizeof(type[K]) + z * sizeof(type)
Release版:ary + ((x * N + y) * K + z) * sizeof(type)
|
结构体
1
2
3
4
5
6
7
8
|
struct tagTest
{
char szName[5]; // + 0
int n; // + 8
double dbl; // + 16
short int w; // + 24
float f; // + 28
}; // sizeof 32
|
指针访问结构体成员
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
int main(int argc, char* argv[])
{
tagTest t = {
"Luo",
1234,
3.14,
12,
0.618f
};
tagTest* p = &t;
printf("%d\r\n", p->n);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/26df1/26df12fc211c04795a3c32a49e97e2e8484a5a36" alt="指针访问结构体成员"
特征
1
2
3
|
会出现寄存器相对间接寻址
出现这种情形,只能是数组常量下标访问或结构成员访问
识别的关键在于论证其中的元素是否具备存储连续性,以及作用是否一致
|
参数为结构体变量
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
void foo(struct tagTest t)
{
printf("%d\r\n", t.n);
}
int main(int argc, char* argv[])
{
tagTest t = {
"Luo",
1234,
3.14,
12,
0.618f
};
foo(t);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/6e841/6e841909ee9f88d7e73d601bec9947f927e1947d" alt="参数为结构体变量"
高版本(VS2019)
会使用多媒体指令集完成,如果结构体足够大,就会使用内联memcpy分方式拷贝到栈上.
特征
1
2
3
|
会将结构体内容拷贝到栈上
关键语句:mov edi, esp
其实相当于一个内联的memcpy(esp, addr, size)
|
参数为结构体指针
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
void foo(struct tagTest* pt)
{
printf("%d\r\n", pt->n);
}
int main(int argc, char* argv[])
{
tagTest t = {
"Luo",
1234,
3.14,
12,
0.618f
};
foo(&t);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/79e3f/79e3fd11df97b5e928160c1a9f6627a20350496d" alt="参数为结构体指针1"
data:image/s3,"s3://crabby-images/306e5/306e5ae5445bdc41ee1ff75eb10cc317cb6abe93" alt="参数为结构体指针2"
特征
访问的时候,会出现寄存器相对间接寻址.
函数返回值为结构体
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
tagTest foo()
{
tagTest t = {
"Luo",
1234,
3.14,
12,
0.618f
};
return t;
}
int main(int argc, char* argv[])
{
tagTest t2 = foo();
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/45928/459283aeedf530585156776f2f1fa5bb0daec6ba" alt="函数返回值为结构体1"
data:image/s3,"s3://crabby-images/88541/885415683f781ecad4e7336456c783ba31da7819" alt="函数返回值为结构体2"
data:image/s3,"s3://crabby-images/6c366/6c366b37256ef84731ed04cf6c58709dfe40a791" alt="函数返回值为结构体3"
返回值给临时变量,因为我们有可能会访问返回值,如foo().n
特征
1
2
3
|
多一个参数返回结构体地址
函数退出时,以参数为目标做memcpy复制结构体内容
函数的返回值为参数1
|
还原
1
2
3
4
5
6
7
8
9
10
11
|
我们按照汇编代码还原,可还原成以下结构
tagTest* GetObj(tagTest* p)
{
...
...
...
memcpy(p, xxx, xxx);
return p;
}
此时等价于
tagTest GetObj();
|
C++之变量相关
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
class CLuoTest
{
public:
CLuoTest();
~CLuoTest();
int IAdd();
void SetVal(int nVal1, int nVal2);
private:
int m_nVal1;
int m_nVal2;
};
CLuoTest::CLuoTest()
{
m_nVal1 = 10;
m_nVal2 = 20;
};
CLuoTest::~CLuoTest()
{
printf("CLuoTest::~CLuoTest\r\n");
};
int CLuoTest::IAdd()
{
return m_nVal1 + m_nVal2;
};
void CLuoTest::SetVal(int nVal1, int nVal2)
{
m_nVal1 = nVal1;
m_nVal2 = nVal2;
}
|
类对象为局部变量
1
2
3
4
5
6
7
8
9
10
|
int main(int argc, char* argv[])
{
CLuoTest LuoTest;
LuoTest.SetVal(1, 2);
int nSum = LuoTest.IAdd();
printf("%d\r\n", nSum);
system("pause");
return 0;
}
|
识别成员函数
Release
data:image/s3,"s3://crabby-images/37948/37948173150acdd93695b4a335b0a6c7a648134a" alt="类对象为局部变量1"
data:image/s3,"s3://crabby-images/6c67b/6c67b27af81af8c4aa43705930cf213921c89b24" alt="类对象为局部变量2"
特征
1
2
3
|
①第一参数为this指针,默认使用ecx传参.
②函数内对第一参数this指针做间接访问.
注意:如果成员函数为_cdecl或_stdcall调用约定,this指针通过栈传送且为第一参数.
|
识别构造函数
Release
data:image/s3,"s3://crabby-images/efb01/efb0170806565f1bfd1062067de848b13f22855d" alt="识别构造函数1"
data:image/s3,"s3://crabby-images/e0ec3/e0ec3b86215e3381bd3385e2321a2373e9dac5e8" alt="识别构造函数2"
特征
1
2
3
4
|
必要条件:
①必为thiscall调用约定.
②构造函数是该对象进入作用域后的第一次成员函数调用.
③返回this指针.
|
识别析构函数
Release
data:image/s3,"s3://crabby-images/f421c/f421c299332984024cefe653684a7747635d6055" alt="识别析构函数1"
data:image/s3,"s3://crabby-images/37e86/37e86bddfdbd191901dc1934def1894bb3b0c9b7" alt="识别析构函数2"
特征
1
2
3
4
|
必要条件:
①必为thiscall调用约定.
②析构函数是该对象进入作用域后的最后一次成员函数调用.
③无返回值定义.
|
类对象为全局变量
1
2
3
4
5
6
7
8
9
10
|
CLuoTest g_LuoTest;
int main(int argc, char* argv[])
{
g_LuoTest.SetVal(1, 2);
int nSum = g_LuoTest.IAdd();
printf("%d\r\n", nSum);
system("pause");
return 0;
}
|
构造函数和析构函数:
在__cinit函数的第二个__initterm函数中调用.
Release
data:image/s3,"s3://crabby-images/2591b/2591b8f45c70a04d8ddbacc7f08d67840dd5731d" alt="类对象为全局变量Release1"
data:image/s3,"s3://crabby-images/e929c/e929c2884226e4ae57585bf997f764ec6bc264f8" alt="类对象为全局变量Release2"
data:image/s3,"s3://crabby-images/30fd1/30fd103c855732b482edf57bbe11714054d36344" alt="类对象为全局变量Release3"
Main函数执行完,会执行atexit注册的函数.
构造函数
data:image/s3,"s3://crabby-images/3595a/3595a11368ea9389ad9ab0d355f1b0a176967b60" alt="类对象为全局变量Release4"
析构函数
data:image/s3,"s3://crabby-images/a1ddc/a1ddc8919aebc177dddad569c5012975026b200b" alt="类对象为全局变量Release5"
data:image/s3,"s3://crabby-images/c9db8/c9db82ba0093f69a6a682a4cca8b218a955493e9" alt="类对象为全局变量Release6"
Debug
在__initterm的函数指针调用.
data:image/s3,"s3://crabby-images/fa6e9/fa6e9de45ea09bb63947349615d03507fe26bf0f" alt="类对象为全局变量Debug1"
构造函数
data:image/s3,"s3://crabby-images/82d2e/82d2eaf5fadb5526311073816670c1d3229ac144" alt="类对象为全局变量Debug2"
析构函数
data:image/s3,"s3://crabby-images/aaba9/aaba9e0e3999fc9ffe3e5677e9853a484d598ef1" alt="类对象为全局变量Debug3"
data:image/s3,"s3://crabby-images/e8fdb/e8fdb788b6b57ad27f1f6d05d6d6907827c0d608" alt="类对象为全局变量Debug4"
总结
data:image/s3,"s3://crabby-images/7ff65/7ff6500eac6bced11ed9f512a159807e60d7681d" alt="类对象为全局变量Debug5"
类对象为堆变量
New单个对象
1
2
3
4
5
6
7
8
9
10
11
12
13
|
int main(int argc, char* argv[])
{
CLuoTest* pLuoTest = new CLuoTest;
pLuoTest->SetVal(1, 2);
int nSum = pLuoTest->IAdd();
printf("%d\r\n", nSum);
delete pLuoTest;
system("pause");
return 0;
}
|
Release
new
data:image/s3,"s3://crabby-images/08fc9/08fc9b53e15a3810463b988ad7fc169c1c3de3c5" alt="New单个对象1"
delete
data:image/s3,"s3://crabby-images/3a86b/3a86b53e760d99839006a429d310e295ae408e44" alt="New单个对象2"
注意
1
2
3
4
|
new和delete函数会自动检查指针是否为空.
但我们仍然需要自己检查指针是否为空.
因为编译器做的检查,是为了是否执行构造函数或析构函数.
而我们做的检查,是为了让我们在别的地方使用这个指针的时候,指针不为空.
|
Debug
new
data:image/s3,"s3://crabby-images/17131/17131cdb629d7e27e22a76aaf6ba8e47866d0dee" alt="New单个对象Debug1"
delete
data:image/s3,"s3://crabby-images/ddf88/ddf88e0f92856df799eb0486e54d011e79f38614" alt="New单个对象Debug2"
data:image/s3,"s3://crabby-images/61de6/61de6ada2289c9027c8a2aeb57b889e8e7e6a626" alt="New单个对象Debug3"
之所以会出现上述代码,是因为我们有可能写出如下代码.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
int main(int argc, char* argv[])
{
CLuoTest* pLuoTest = new CLuoTest;
pLuoTest->SetVal(1, 2);
int nSum = pLuoTest->IAdd();
printf("%d\r\n", nSum);
pLuoTest->~CLuoTest(); //人工调析构,反复利用同一个空间
pLuoTest->CLuoTest::CLuoTest();
pLuoTest->SetVal(10, 20);
nSum = pLuoTest->IAdd();
printf("%d\r\n", nSum);
delete pLuoTest;
system("pause");
return 0;
}
|
data:image/s3,"s3://crabby-images/1f810/1f810f79e332fd7e58b9b5ad7ab227ba708cba2b" alt="New单个对象Debug4"
此时push的参数就会是0,只执行析构函数,不释放空间.
New对象数组
1
2
3
4
5
6
7
8
9
10
11
12
13
|
int main(int argc, char* argv[])
{
CLuoTest* pLuoTest = new CLuoTest[5];
pLuoTest->SetVal(1, 2);
int nSum = pLuoTest->IAdd();
printf("%d\r\n", nSum);
delete[] pLuoTest;
system("pause");
return 0;
}
|
Debug
new
data:image/s3,"s3://crabby-images/3650f/3650fa27db6dcf7b301a9f9612a30f98cea71ac3" alt="New对象数组1"
这个地方需要析构函数是因为,遇到new失败时,虽然前面的new成功,,但也算整个new对象数组的失败,需要把前面成功new的对象析构掉.
detete[]
data:image/s3,"s3://crabby-images/8cc63/8cc631f5da159506c7ffbb3e7e1cc33a80777021" alt="New对象数组2"
data:image/s3,"s3://crabby-images/4d191/4d1917f853f22afe0fb260e3d837ae8f4a1058e7" alt="New对象数组3"
C++之对象相关
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
class CLuoTest
{
public:
CLuoTest();
~CLuoTest();
void Show();
void SetVal(int nVal1, int nVal2);
private:
int m_nVal1;
int m_nVal2;
char m_szName[20];
};
CLuoTest::CLuoTest()
{
m_nVal1 = 10;
m_nVal2 = 20;
strcpy(m_szName, "Luo");
printf("CLuoTest::CLuoTest\r\n");
};
CLuoTest::~CLuoTest()
{
printf("CLuoTest::~CLuoTest\r\n");
};
void CLuoTest::Show()
{
printf("%d, %d, %s\r\n", m_nVal1, m_nVal2, m_szName);
};
void CLuoTest::SetVal(int nVal1, int nVal2)
{
m_nVal1 = nVal1;
m_nVal2 = nVal2;
strcpy(m_szName, "Hun");
}
|
对象传参
1
2
3
4
|
void foo(CLuoTest LuoTest)
{
LuoTest.Show();
}
|
无拷贝构造
1
2
3
4
5
6
7
8
9
|
int main(int argc, char* argv[])
{
CLuoTest LuoTest;
LuoTest.SetVal(10, 20);
foo(LuoTest);
system("pause");
return 0;
}
|
Release
data:image/s3,"s3://crabby-images/4106c/4106cfb2449a1e4a62c3d8d8684c0eb226532c1b" alt="对象传参1"
特征
1
|
以栈顶为Dst,对象地址为Src,执行memcpy,浅拷贝.
|
有拷贝构造
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
/*
//拷贝构造
CLuoTest::CLuoTest(CLuoTest& Obj)
{
m_nVal1 = Obj.m_nVal1;
m_nVal2 = Obj. m_nVal2;
strncpy(m_szName, Obj. m_szName, 20);
}
*/
int main(int argc, char* argv[])
{
CLuoTest LuoTest;
LuoTest.SetVal(10, 20);
foo(LuoTest);
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/3f802/3f802c04c36aa9eac8ea369bad88f95825ab4ec3" alt="对象传参2"
data:image/s3,"s3://crabby-images/77996/779967fd836463ee9b400b623fee96c2b4ef69a7" alt="对象传参3"
Release
data:image/s3,"s3://crabby-images/c53af/c53afc8c719d14c62362f4a9764ee26d656e6940" alt="对象传参4"
data:image/s3,"s3://crabby-images/a7cec/a7cecf7cbb631353d04530d86deef5bc9e7f7059" alt="对象传参5"
特征
1
2
3
|
①将栈顶作为this指针给ecx,调用拷贝构造
关键语句:mov ecx, esp
②函数外调用构造函数,函数内调用析构函数
|
对象返回
1
2
3
4
5
6
|
CLuoTest GetObj()
{
CLuoTest LuoTest;
LuoTest.SetVal(10, 20);
return LuoTest;
}
|
拷贝构造
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
/*
//拷贝构造
CLuoTest::CLuoTest(CLuoTest& obj)
{
m_nVal1 = obj.m_nVal1;
m_nVal2 = obj. m_nVal2;
strncpy(m_szName, obj. m_szName, 20);
}
*/
int main(int argc, char* argv[])
{
CLuoTest HunTest = GetObj(); //拷贝构造
HunTest.Show();
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/36bdf/36bdf85259dbcf8a2c01c41a9df54e0174679897" alt="对象返回拷贝构造1"
data:image/s3,"s3://crabby-images/3cf8e/3cf8ebb11f48724a3359d9247421c2a7624ec6b6" alt="对象返回拷贝构造2"
浅拷贝
1
2
3
4
5
6
7
8
9
|
int main(int argc, char* argv[])
{
CLuoTest HunTest;
HunTest = GetObj(); //浅拷贝,产生临时对象
HunTest.Show();
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/62b64/62b64c2e7d500b4548ed4729e243335e616a8b8c" alt="对象返回浅拷贝1"
data:image/s3,"s3://crabby-images/b3ad2/b3ad2ab1cecac4decebe043a60850b2130caea9d" alt="对象返回浅拷贝2"
特征
1
2
|
①调用函数结束后,会执行以返回值为Src,目标对象为Dst的memcpy.
②临时对象,即多传进去的那个参数会执行析构.
|
无名对象
1
2
3
4
5
6
7
8
|
int main(int argc, char* argv[])
{
CLuoTest& Ref = GetObj(); //产生无名对象,遇到分号不析构,生命周期随Ref
Ref.Show();
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/643ea/643eaf505a9f6da89792b2de3598e56a8fdb345f" alt="对象返回无名对象1"
特征
1
2
3
|
①函数多了一个参数.
②函数内构造,函数外析构.
③返回参数1.
|
运算符重载
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
/*
//运算符重载
//类内声明为友元函数
friend CLuoTest& operator+(CLuoTest& ObjDst, CLuoTest& ObjSrc);
CLuoTest& operator+(CLuoTest& ObjDst, CLuoTest& ObjSrc)
{
ObjDst.m_nVal1 += ObjSrc.m_nVal1;
ObjDst.m_nVal2 += ObjSrc.m_nVal2;
return ObjDst;
}
*/
int main(int argc, char* argv[])
{
CLuoTest LuoTest;
LuoTest.SetVal(10, 20);
CLuoTest HunTest;
LuoTest = LuoTest + HunTest; //中缀式,实际调用方式为波兰式 LuoTest = operator+(LuoTest, HunTest);
LuoTest.Show();
system("pause");
return 0;
}
|
Release
data:image/s3,"s3://crabby-images/456a1/456a169bb2ddff1c86924c2d2b4426a83a6f76d7" alt="对象返回运算符重载1"
静态局部
1
2
3
4
5
6
7
8
|
int main(int argc, char* argv[])
{
static CLuoTest LuoTest;
LuoTest.Show();
system("pause");
return 0;
}
|
Release
data:image/s3,"s3://crabby-images/35552/35552e054a8dd865c1146d3a1b2fc6b6a3901932" alt="静态局部1"
data:image/s3,"s3://crabby-images/139f5/139f559ae7280cd39d6cfb677612bfd4e3eb0512" alt="静态局部2"
特征
1
2
|
①用一个标记来记录静态局部对象是否已初始化.
②若未初始化,则执行构造函数,并注册析构代理.
|
C++之单重继承
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
class CLuoTest
{
public:
CLuoTest();
CLuoTest::CLuoTest(CLuoTest& Obj);
virtual ~CLuoTest();
virtual void Show();
void SetVal(int nVal1, int nVal2);
private:
int m_nVal1;
int m_nVal2;
char m_szName[20];
};
CLuoTest::CLuoTest()
{
m_nVal1 = 10;
m_nVal2 = 20;
strcpy(m_szName, "Luo");
printf("CLuoTest::CLuoTest\r\n");
};
CLuoTest::CLuoTest(CLuoTest& Obj)
{
m_nVal1 = Obj.m_nVal1;
m_nVal2 = Obj.m_nVal2;
strncpy(m_szName, Obj.m_szName, 20);
}
CLuoTest::~CLuoTest()
{
printf("CLuoTest::~CLuoTest\r\n");
};
void CLuoTest::Show()
{
printf("CLuoTest:%d, %d, %s\r\n", m_nVal1, m_nVal2, m_szName);
};
void CLuoTest::SetVal(int nVal1, int nVal2)
{
m_nVal1 = nVal1;
m_nVal2 = nVal2;
strcpy(m_szName, "Hun");
}
|
虚函数
只有使用对象的指针或引用调用虚函数时,才有访问虚表,间接调用的动作.
1
2
3
4
5
6
7
8
9
10
11
|
int main(int argc, char* argv[])
{
CLuoTest LuoObj;
LuoObj.Show();
CLuoTest& HunObj = LuoObj;
HunObj.Show();
system("pause");
return 0;
}
|
调用方式
Debug
data:image/s3,"s3://crabby-images/400dd/400ddb4d1239bbf6d70899cac8948876756d1df5" alt="虚函数调用方式1"
特征
构造函数
Debug
data:image/s3,"s3://crabby-images/6cae3/6cae323e0c92b635d7f8b1dd77c2f04d48482c2a" alt="虚函数构造函数1"
特征
1
2
|
构造函数会填写虚表,这个地址在只读数据区.
注意:虚表中的函数指针不是0结尾,需通过实际调用,来判断虚函数.
|
初始化列表
1
2
3
4
5
6
|
CLuoTest::CLuoTest()
:m_nVal1(10), m_nVal2(20)
{
strcpy(m_szName, "Luo");
printf("CLuoTest::CLuoTest\r\n");
}
|
Debug
data:image/s3,"s3://crabby-images/60723/60723b8485729266488c652a271baf9a8c3f3578" alt="虚函数初始化列表1"
析构函数
Debug
data:image/s3,"s3://crabby-images/c0e0b/c0e0b8bbf016cbffdef96b8f7e42d6fb35f701e9" alt="虚函数析构函数"
特征
注意
为什么析构函数会填写虚表?
假设有类A和类B.
B继承A,构造时填写的是B的虚表.
而当B析构完,还需要析构A,但此时this指针中的虚表填写的是B的虚表.
如果此时我们通过虚表间接访问,就会访问到B的函数,但是B已经析构完了.
故需要在析构的时候回填虚表.
单重继承有虚函数
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
class CHunTest : public CLuoTest
{
public:
CHunTest();
virtual ~CHunTest();
virtual void Show();
private:
int m_nVal1;
int m_nVal2;
char m_szName[20];
};
CHunTest::CHunTest()
{
m_nVal1 = 100;
m_nVal2 = 200;
strcpy(m_szName, "Hun");
printf("CHunTest::CHunTest\r\n");
}
CHunTest::~CHunTest()
{
printf("CHunTest::~CHunTest\r\n");
}
void CHunTest::Show()
{
printf("CHunTest:%d, %d, %s\r\n", m_nVal1, m_nVal2, m_szName);
}
|
构造函数
1
2
3
4
5
6
7
8
9
|
int main(int argc, char* argv[])
{
CLuoTest* pLuoObj = new CHunTest;
pLuoObj->Show();
delete pLuoObj;
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/1c532/1c5325bc929981219689a8a5d46dd5594d8b5aa2" alt="单重继承有虚函数构造1"
虚表内容
data:image/s3,"s3://crabby-images/8fa8e/8fa8ecf6150aebad82e105637aabebb153b3c4a8" alt="单重继承有虚函数构造2"
data:image/s3,"s3://crabby-images/bee5b/bee5b0df098fa1b982fb6b3fcc48b6f208f6e0e2" alt="单重继承有虚函数构造3"
技巧之交叉引用
- 查看虚表引用
1
2
3
4
5
6
7
8
9
|
int main(int argc, char* argv[])
{
CLuoTest* pLuoObj = new CHunTest;
pLuoObj->Show();
delete pLuoObj;
system("pause");
return 0;
}
|
- 找到构造函数,识别出虚表地址
data:image/s3,"s3://crabby-images/fbc14/fbc145747505b0e7544998356af1e7d0d30c94e5" alt="查看虚表引用1"
- 右键Xrefs graph to
data:image/s3,"s3://crabby-images/8a735/8a735aa413e799ed2525156607d8bbc87b957409" alt="查看虚表引用2"
data:image/s3,"s3://crabby-images/a1d56/a1d569db4570be8db565a496e82f7b688bb79ed8" alt="查看虚表引用3"
从这张图中,我们就可以知道构造函数和析构函数.
- 查看构造函数引用
1
2
3
4
5
6
7
8
9
10
|
CHunTest g_HunObj;
int main(int argc, char* argv[])
{
CHunTest HunObj1;
CHunTest HunObj2;
CHunTest HunObj3;
system("pause");
return 0;
}
|
data:image/s3,"s3://crabby-images/ed2b9/ed2b91d0772ce55f0cc1802e4153ac088cae36d6" alt="查看构造函数引用"
- 查看父类构造函数引用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
class CHunTest1 : public CLuoTest
{
};
class CHunTest2 : public CLuoTest
{
};
class CHunTest3 : public CLuoTest
{
};
int main(int argc, char* argv[])
{
CHunTest HunObj;
CHunTest1 HunObj1;
CHunTest2 HunObj2;
CHunTest3 HunObj3;
system("pause");
return 0;
}
|
data:image/s3,"s3://crabby-images/8c625/8c62558053606f507caa341ece4dc705043e25e2" alt="查看父类构造函数引用1"
data:image/s3,"s3://crabby-images/49cd5/49cd5f2ba35716e61c5c4b1b6a7ec802461dfad4" alt="查看父类构造函数引用2"
data:image/s3,"s3://crabby-images/558f4/558f422e5845cb3e4c90aa0d65ef61fb8484f42a" alt="查看父类构造函数引用3"
C++之对象关系
继承:
data:image/s3,"s3://crabby-images/7a68f/7a68fd370196c356e5a9d0821e196eb59505b679" alt="继承"
特征
包含:
data:image/s3,"s3://crabby-images/8b96c/8b96c9c77dc9560755b4f023fe6f767c49a624c5" alt="包含"
特征
1
2
|
不共用虚表,同一虚表位置写一次.
当红色对象析构时,紫色对象也不在了.
|
聚合:
data:image/s3,"s3://crabby-images/958a3/958a357326f2ae8d4b45bae47c96f3637e980f18" alt="聚合"
特征
注意
当无虚表时,继承和包含无法区分.
因为面向对象,是通过行为区分的,不是通过数据区分的.
虚表用来区分每个对象是什么.
数据用来区分同类型实体的差异.
C++之多重继承
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
class CLuoTest
{
public:
CLuoTest();
virtual ~CLuoTest();
virtual void Show();
private:
int m_nVal1;
int m_nVal2;
char m_szName[20];
};
CLuoTest::CLuoTest()
{
m_nVal1 = 10;
m_nVal2 = 20;
strcpy(m_szName, "Luo");
printf("CLuoTest::CLuoTest\r\n");
};
CLuoTest::~CLuoTest()
{
printf("CLuoTest::~CLuoTest\r\n");
};
void CLuoTest::Show()
{
printf("CLuoTest:%d, %d, %s\r\n", m_nVal1, m_nVal2, m_szName);
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
class CHunTest
{
public:
CHunTest();
virtual ~CHunTest();
virtual void Show();
private:
int m_nVal1;
int m_nVal2;
char m_szName[20];
};
CHunTest::CHunTest()
{
m_nVal1 = 100;
m_nVal2 = 200;
strcpy(m_szName, "Hun");
printf("CHunTest::CHunTest\r\n");
}
CHunTest::~CHunTest()
{
printf("CHunTest::~CHunTest\r\n");
}
void CHunTest::Show()
{
printf("CHunTest:%d, %d, %s\r\n", m_nVal1, m_nVal2, m_szName);
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
class CLuoHunTest : public CLuoTest, public CHunTest
{
public:
CLuoHunTest();
virtual ~CLuoHunTest();
virtual void Show();
private:
int m_nVal1;
int m_nVal2;
char m_szName[20];
};
CLuoHunTest::CLuoHunTest()
{
m_nVal1 = 1000;
m_nVal2 = 2000;
strcpy(m_szName, "LuoHun");
printf("CLuoHunTest::CLuoHunTest\r\n");
}
CLuoHunTest::~CLuoHunTest()
{
printf("CLuoHunTest::~CLuoHunTest\r\n");
}
void CLuoHunTest::Show()
{
printf("CLuoHunTest:%d, %d, %s\r\n", m_nVal1, m_nVal2, m_szName);
}
|
内存模型:
data:image/s3,"s3://crabby-images/29e92/29e929f32e357fbbbf6e2f2e72a6433e51070cab" alt="多重继承内存模型"
CLuo和CHun的顺序,按继承时的书写顺序,从左到右.
构造函数:
1
2
3
4
5
6
7
8
9
|
int main(int argc, char* argv[])
{
CLuoHunTest* pObj = new CLuoHunTest;
pObj->Show();
delete pObj;
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/39b3f/39b3ff532d3b58a9bfd439c81a93b2d6df49868e" alt="多重继承构造函数"
特征
1
|
先调父类构造,然后有几个父类,就填写几次虚表.
|
同类型指针转换:
1
2
3
4
5
6
7
8
9
|
int main(int argc, char* argv[])
{
CLuoHunTest LuoHunObj;
CLuoTest* pLuoObj = &LuoHunObj;
CHunTest* pHunObj = &LuoHunObj; //这个地方会产生一个分支
system("pause");
return 0;
}
|
Debug
data:image/s3,"s3://crabby-images/e9d63/e9d636f3ce65e52d4ffa2d54e5289dd49df0ee45" alt="同类型指针转换"
注意
1
2
3
4
5
6
|
为什么会产生一个分支?
CLuoHunTest LuoHunObj;
CLuoTest* pLuoObj = &LuoHunObj;
CHunTest* pHunObj = &LuoHunObj;//这个地方会移动This指针,假如说&LuoHunObj +0x10位置是CHunTest对象的位置,当&LuoHunObj 不为NULL时,这样是正确的,但如果&LuoHunObj为NULL,即下列代码,若无脑+0x10,结果就是错误的了
CLuoTest* pLuoObj = NULL;
CHunTest* pHunObj = NULL;
|
纯虚函数识别:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
class CLuoTest
{
public:
CLuoTest();
virtual ~CLuoTest();
virtual void Show() = 0;
};
CLuoTest::CLuoTest()
{
};
CLuoTest::~CLuoTest()
{
printf("CLuoTest::~CLuoTest\r\n");
};
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
class CLuoHunTest : public CLuoTest
{
public:
CLuoHunTest();
virtual ~CLuoHunTest();
virtual void Show();
private:
int m_nVal1;
int m_nVal2;
char m_szName[20];
};
CLuoHunTest::CLuoHunTest()
{
m_nVal1 = 1000;
m_nVal2 = 2000;
strcpy(m_szName, "LuoHun");
printf("CLuoHunTest::CLuoHunTest\r\n");
}
CLuoHunTest::~CLuoHunTest()
{
printf("CLuoHunTest::~CLuoHunTest\r\n");
}
void CLuoHunTest::Show()
{
printf("CLuoHunTest:%d, %d, %s\r\n", m_nVal1, m_nVal2, m_szName);
}
|
1
2
3
4
5
6
7
8
|
int main(int argc, char* argv[])
{
CLuoHunTest LuoHunObj;
LuoHunObj.Show();
system("pause");
return 0;
}
|
定位到父类虚表的第二项,纯虚函数项
Debug
data:image/s3,"s3://crabby-images/a0a6b/a0a6baf0ef9b81c6ac41467eb9bdc39a91d86da0" alt="纯虚函数识别1"
data:image/s3,"s3://crabby-images/a22cf/a22cfd6471735852104bd174cd8b7549071977df" alt="纯虚函数识别2"
特征
1
2
|
若强制调用纯虚函数,会出现一个错误.
因为虚表中,纯虚函数项填了一个提示错误信息,并关闭的函数指针即_purecall的函数指针.
|
注意
在构造函数或析构函数中调虚函数,不会触发多态.
因为构造函数或析构函数中会回填虚表,填自身对象的虚表地址.
C++之菱形结构
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
class CLuoA
{
public:
virtual void TestA() {}
virtual void TestCoverA() {}
private:
int m_nValA = 0xAAAAAAAA;
};
class CLuoB : virtual public CLuoA
{
public:
virtual void TestCoverA() {}
virtual void TestB() {}
private:
int m_nValB = 0xBBBBBBBB;
};
class CLuoD : virtual public CLuoA
{
public:
virtual void TestCoverA() {}
virtual void TestD() {}
private:
int m_nValD = 0xDDDDDDDD;
};
class CLuoE :public CLuoB, public CLuoD
{
public:
virtual void TestCoverA() {}
virtual void TestCoverB() {}
virtual void TestCoverD() {}
virtual void TestE() {}
private:
int m_nValE = 0xEEEEEEEE;
};
|
内存模型:
data:image/s3,"s3://crabby-images/e687c/e687cbfa572f1d69d7cad10ca895c9bd89c5a80e" alt="菱形结构1"
data:image/s3,"s3://crabby-images/c369f/c369f4a0d99d9c18b941a3b0498afcc24e431e0a" alt="菱形结构2"
构造函数:
Debug
data:image/s3,"s3://crabby-images/b5240/b5240fba1109d2e4bb18eb83a42fe7fa078f6ee7" alt="菱形结构3"
这个地方传1,表明需要填写偏移以及调祖先类构造函数.
data:image/s3,"s3://crabby-images/9b86c/9b86c38c7db147943216cfe0b3ff58a1f8e6d395" alt="菱形结构4"
data:image/s3,"s3://crabby-images/92930/92930f53bcd5014c3730c9711a3eb055f673c325" alt="菱形结构5"
异常
基本数据类型异常
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
|
#include <stdio.h>
#include <stdlib.h>
void LuoTestException(int n)
{
switch (n)
{
case 0:
{
throw 1;
break;
}
case 1:
{
throw 3.14f;
break;
}
case 2:
{
throw 6.28;
break;
}
case 3:
{
throw 'L';
break;
}
case 4:
{
bool bl = false;
throw bl;
break;
}
default:
break;
}
}
int main(int argc, char* argv[])
{
int nVal = 0;
scanf("%d", &nVal);
try
{
LuoTestException(nVal);
}
catch (int n)
{
printf("catch int %d\r\n", n);
}
catch (float f)
{
printf("catch float %f\r\n", f);
}
catch (double dbl)
{
printf("catch double %lf\r\n", dbl);
}
catch (char ch)
{
printf("catch char %c\r\n", ch);
}
catch (bool b)
{
printf("catch bool %d\r\n", b);
}
catch(...)
{
printf("catch all\r\n");
}
system("pause");
return 0;
}
|
无脑定位异常处理函数
- 如果我们在调试的时候,看到抛异常的函数.
data:image/s3,"s3://crabby-images/414b8/414b80039a22e75b4408745604baeb7344705af8" alt="无脑定位异常处理函数1"
- 我们找到SEH链第一个异常处理的地方,下断,F9运行.
data:image/s3,"s3://crabby-images/21ae3/21ae37b23934b18ed74408d3ef676a4c3b165ff0" alt="无脑定位异常处理函数2"
- 按照以下定则进call.
- 找参数最多的call下断点.
- 参数一样,就同时下断,看哪个到达.
- 最后一层是call寄存器(eax).
data:image/s3,"s3://crabby-images/cc911/cc9111bc335dc6bc1c19447879ae0c128ca23e80" alt="无脑定位异常处理函数3"
异常信息关系表
data:image/s3,"s3://crabby-images/0ba3e/0ba3ea1aa7286b61bba366dac5b802af7f9eb0b3" alt="异常信息关系表"
还原异常处理结构(上半部分表)
- 函数开头进入SEH异常链.
data:image/s3,"s3://crabby-images/92c70/92c70df6806113478588dc8403fb177b95dea7f0" alt="还原异常处理结构1"
- 找到异常函数信息结构体.
data:image/s3,"s3://crabby-images/5a66a/5a66af6aa1718a79488416c6f049d98503a6e981" alt="还原异常处理结构2"
- 分析数据关系.
- 第一张表记录了该函数有几个try,以及对应的try块信息表指针数组.
data:image/s3,"s3://crabby-images/ef33c/ef33cdbb8589b507ae1a8baa4ed06d3b0be92e34" alt="还原异常处理结构3"
- 第二张表记录了try对应的catch个数,以及Rtti描述.
- 第三张表记录了catch的类型以及catch的处理函数.
data:image/s3,"s3://crabby-images/9fefa/9fefac9a800e2952d5a6a2bb6b3db6ae0e0c40c6" alt="还原异常处理结构4"
注意:catch的处理函数是有返回值的,返回到catch结束的位置.
data:image/s3,"s3://crabby-images/20ea8/20ea8d8040f6dcc9974c14276b7e95e96bd8cf94" alt="还原异常处理结构5"
还原具体的异常处理代码(下半部分表)
- 找到函数内抛异常的函数,参数中找抛出的信息表.
data:image/s3,"s3://crabby-images/bb008/bb00882c1d687532b84a3414a1cf1fba6b110c98" alt="还原具体的异常处理代码1"
- 找到匹配列表.
data:image/s3,"s3://crabby-images/5e238/5e2387e89530ece48d1ac7b51d4c4f9355e77143" alt="还原具体的异常处理代码2"
data:image/s3,"s3://crabby-images/c9c97/c9c97b6a3204ea8ffa6548a62ee5785a649a1375" alt="还原具体的异常处理代码3"
- 在catch信息表中找到类型信息表.
data:image/s3,"s3://crabby-images/731ea/731ea7943064c861ea95366d358fa00562bf1a01" alt="还原具体的异常处理代码4"
- 对类型信息表,做交叉引用,就可以找到对应的catch处理代码.
data:image/s3,"s3://crabby-images/b8955/b89557c956c7b688c9b63f4abb94d0ac13b8ae80" alt="还原具体的异常处理代码5"
data:image/s3,"s3://crabby-images/ad418/ad418f0840c376214e795a55e14afb1e82fddba2" alt="还原具体的异常处理代码6"