Cpp8 运算符重载和深浅拷贝

Cpp8 运算符重载和深浅拷贝

深浅拷贝

相同类型间可以直接拷贝

// _20180212.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"

#include <stdio.h>
#include <windows.h>

class A
{
private:
    int* a;
public:
    A()
    {
        a = new int[10];
    }
    virtual ~A()
    {
        delete a;
        printf("析构 A \n");
    }
};

int main(int argc,char* argv[])
{
    A a1;
    A a2;
    a1 = a2;
    return 0;
}

//反汇编


27:       A a1;
0040108D   lea         ecx,[ebp-14h]
00401090   call        @ILT+15(B::B) (00401014)
00401095   mov         dword ptr [ebp-4],0
28:       A a2;
0040109C   lea         ecx,[ebp-1Ch]
0040109F   call        @ILT+15(B::B) (00401014)
004010A4   mov         byte ptr [ebp-4],1
29:       a1 = a2;
004010A8   lea         eax,[ebp-1Ch]
004010AB   push        eax              //a2 作为参数传递
004010AC   lea         ecx,[ebp-14h]    //a1 作为this指针传递
004010AF   call        @ILT+45(A::operator=) (00401032)
30:       return 0;


A::operator=:
004012E0   push        ebp
004012E1   mov         ebp,esp
004012E3   sub         esp,44h
004012E6   push        ebx
004012E7   push        esi
004012E8   push        edi
004012E9   push        ecx
004012EA   lea         edi,[ebp-44h]
004012ED   mov         ecx,11h
004012F2   mov         eax,0CCCCCCCCh
004012F7   rep stos    dword ptr [edi]
004012F9   pop         ecx 

//eax = a1首地址
004012FA   mov         dword ptr [ebp-4],ecx 
004012FD   mov         eax,dword ptr [ebp-4]   
//ecx = a2首地址
00401300   mov         ecx,dword ptr [ebp+8]
//ecx + 4 这里是 class A中变量a的地址(ecx 对应虚表地址) 
//因为ecx是变量a2 所以这里的edx = a2.a   
00401303   mov         edx,dword ptr [ecx+4]   
//a2.a 赋值给a1.a
00401306   mov         dword ptr [eax+4],edx
//返回a1首地址 
00401309   mov         eax,dword ptr [ebp-4]

0040130C   pop         edi
0040130D   pop         esi
0040130E   pop         ebx
0040130F   mov         esp,ebp
00401311   pop         ebp
00401312   ret         4

我们发现使用=直接赋值对象,编译器自动生成了operator= 函数用于处理类型赋值
我们观察编译器自动生成的函数operator=,这里赋值直接略过了虚表(ecx+4)

// _20180212.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"

#include <stdio.h>
#include <windows.h>

class A
{
private:
    int* a;
public:
    A()
    {
        a = new int[10];
    }
    virtual ~A()
    {
        delete a;
        printf("析构 A \n");
    }

    int* get_a()
    {
        return a;
    }
};

int main(int argc,char* argv[])
{
    A a1;
    A* a2 = new A;
    a1 = *a2;
    delete a2;
    int* i = a1.get_a();

    for (int j =0;j<10;j++)
    {
        i[j] = j;
    }
    return 0;
}

上述代码看似正常,运行也不报错,原因是虽然delete a2;释放掉了a2的内存,内存管理器虽然认为此内存无用,但是却仍然可以访问,所以这里是在非法使用内存

步骤分析

A a1;
A* A2 = new A;

此时a1和a2对应的内存占用如图

image
a1 = *a2;

这一步是无脑拷贝 把a2.a 赋值给 a1.a

此时的内存结构

image

这里 a指向的 int* a 就造成了内存泄漏

delete a2;

这里释放掉a2
内存结构如图

image

这里a2中的int* a成员指向的内存已经释放,
但是此时a1的int* a成员仍然指向a2的成员指向的内存

int* i = a1.get_a();

    for (int j =0;j<10;j++)
    {
        i[j] = j;
    }

这里调用get_a 方法 获取 a1.a的地址,把指向的内存赋值。
这里获取到的 int* i指向的已经是个不存在的 非法的内存块
所以这里就造成了内存的非法访问,如果操作系统吧这里的位置分配给其他位置使用
那么就造成了数据混乱,造成比内存泄漏更危险的错误

以上的拷贝方式,就被称为浅拷贝
浅拷贝:只拷贝成员的值
深拷贝:拷贝的不只是成员。还有成员指向的内存的数据

所以在对象拥有指针成员的时候,就需要慎重考虑使用深拷贝还是浅拷贝

运算符重载

一个类,需要比较类的大小

class Number
{
public:
    Number(int x,int y):x(x),y(y)
    {
    }
    bool Max(Number& n)
    {
        return  this->x>n.x && this->y>n.y;
    }
private:
    int x;
    int y;
};

int main(int argc, char* argv[])
{
    Number n1(1,1),n2(2,2);
    
    bool r = n1.Max(n2);
    return 0;
}

bool类型,其实就是一个char true是1 false是0

为了比较 n1和n2的大小。我们定义了一个max函数。此时我们写的时候有没有感觉很繁琐,那么我们能不能像基础类型那样使用 >这样的符号来做运算呢 比如

boolr = n1 > n2;

答案是可以的!

// _20180223.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"

class Number
{
public:
    Number(int x,int y):x(x),y(y)
    {
    }
    bool operator>(Number& n)
    {
        return  this->x>n.x && this->y>n.y;
    }
private:
    int x;
    int y;
};

int main(int argc, char* argv[])
{
    Number n1(1,1),n2(2,2);
    
    bool r = n1 > (n2);
    return 0;
}

重载 ++ -- + - * / > <等等运算符

class Number
{
public:
    Number(int x,int y):x(x),y(y)
    {
    }
    Number operator++();
    Number operator--();
    Number operator+(const Number& p);
    Number operator-(const Number& p);
    Number operator*(const Number& p);
    Number operator/(const Number& p);

    bool operator<(const Number& p);
    bool operator==(const Number& p);
    bool operator>(const Number& n)
    {
        return  this->x>n.x && this->y>n.y;
    }
private:
    int x;
    int y;
};

int main(int argc, char* argv[])
{
    Number n1(1,1),n2(2,2);
    
    bool r = n1 > (n2);
    return 0;
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,122评论 6 505
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,070评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,491评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,636评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,676评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,541评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,292评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,211评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,655评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,846评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,965评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,684评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,295评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,894评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,012评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,126评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,914评论 2 355

推荐阅读更多精彩内容

  • @synthesize和@dynamic分别有什么作用?@property有两个对应的词,一个是 @synthes...
    笔笔请求阅读 515评论 0 1
  • C++语言的一个很有意思的特性就是除了支持函数重载外还支持运算符重载,原因就是在C++看来运算符也算是一种函数。比...
    欧阳大哥2013阅读 2,708评论 0 8
  • 一、Java 简介 Java是由Sun Microsystems公司于1995年5月推出的Java面向对象程序设计...
    子非鱼_t_阅读 4,186评论 1 44
  • Python爬虫——Beautiful Soup的用法 学习自崔庆才的个人博客静觅文章地址:http://cuiq...
    KevinCool阅读 1,509评论 0 3
  • 最近看文章说:谁都希望能如自己所愿,拥有善解人意的丈夫,懂得陪伴,愿意迁就,两个人一起生一个孩子,养一只狗,从此三...
    鹿妞阅读 253评论 0 0