作业分析 c++四大函数练习

题目:

为下面的Rectangle类实现构造函数,拷贝构造函数,赋值操作符,析构函数。

class Shape

{

int no;

};

class Point

{

int x;

int y;

};

class Rectangle: public Shape

{

int width;

int height;

Point * leftUp;

public:

Rectangle(int width, int height, int x, int y);

Rectangle(const Rectangle& other);

Rectangle& operator=(const Rectangle& other);

~Rectangle();

};

解析:

一 构造函数:

1.尽量使用初始化列表;

2.对leftUp指针的构造,leftUp指向一个Point对象,构造函数需要在堆内生成一个新的point对象,并用leftUp指向该对象

inline
Rectangle::Rectangle(int width,int heigt, int x,int y):width(width),height(height),leftUp(new Point(x,y)){}
//尽量使用初始化列表,包括对leftUp的初始化  

二 拷贝构造函数:

1.尽量使用初始化列表

2.注意对父类继承而来的no的拷贝构造,方法是调用父类shape的拷贝构造函数 Shape(other)

3.对this->leftUp的拷贝构造,调用Point的拷贝构造函数,这样在Point内部有变化时,避免大量修改。

 this->leftUp = new Point(*other.leftUp); 

4.针对other.leftUp是否为空的讨论,空指针情况没必要在堆内生成对象,同时当other.leftUp为空时,this->leftUp初始默认是随机值,要对他进行赋值为空指针。

完整的拷贝构造函数:

inline
Rectangle::Rectangle(const Rectangle& other)
:Shape(other),width(other.width),height(other.height){  // 注意对继承而来对象no的拷贝构造,通过调用父类的拷贝构造函数
    if(other.leftUp != NULL){                         //针对other.leftUp是否为空的讨论,空指针情况没必要在堆内生成对象
        this->leftUp = new Point(*other.leftUp);    //调用Point的拷贝构造函数,这样在Point内部有变化时,避免大量修改。
    }
    else{
        this->leftUp = NULL;                          //leftUp初始默认是随机值,要对他进行赋值为空指针。
    }
}

三 赋值操作符

1.赋值操作符往往需要首先判定自己给自己赋值的情况,避免内存泄露

if(this == &other){
        return *this;
}

2.调用父类的赋值操作符,完成对父类继承部分的赋值操作,方法如下:

    Shape::operator=(other);     //调用父类的赋值操作符,完成对父类继承部分的赋值操作 

3.需要对leftUp,other.leftUp是否为空进行讨论

other.leftUp为空时,直接释放this->leftUp空间,并将其赋为空即可;

other.leftUp不为空时,

若this->leftUp也不为空,则直接将other->leftUp指向的内容赋值给this->leftUp指向的内容即可;

若this->leftUp为空,创建新的Point对象

Rectangle& Rectangle::operator= (const Rectangle& other){
    if(this == &other){                 //赋值操作符往往需要首先判定自己给自己赋值的情况,避免内存泄露
        return *this;
    }
    Shape::operator=(other);     //调用父类的赋值操作符,完成对父类继承部分的赋值操作
    this->width = other.width;
    this->height = other.height;

    if(other.leftUp != NULL){
        if(leftUp != NULL) {
            *leftUp = *other.leftUp;             //不必删除leftUp再重新构建,直接进行赋值即可(解指针,调用point类的赋值操作符即可)
        }
        else{
            leftUp = new Point(*other.leftUp);      //leftUp为空,不能解指针,需要创建一个新对象
        }
    }
    else{
        delete leftUp;
        this->leftUp = NULL;
    }
    return *this;
} 

四 析构函数

Rectangle:: ~Rectangle(){
    delete leftUp;
} 

五 整体代码和其他注意事项

1.Rectangle赋值构造函数,构造顺序:先父类,后按照子类中声明的顺序,与初始化列表中的顺序无关。

2.正确区分拷贝构造函数和赋值操作符。

拷贝构造函数是构造函数,也就是创建新对象时,所以一个对象存在,一个对象尚未存在;

赋值操作符使用时,两个对象必然都是存在的,所以需要讨论的问题是是否自我赋值等等。

3 面对此类问题方法:

先忘掉语法,画内存模型

本例即

然后写的时候分析指针是否为空;

拷贝构造就是一边有,一边没有;赋值操作符就是两边都有;

结合指针是否为空,可以分析出上述的注意事项。

class Shape
{
     int no;
};

class Point
{
private:
     int x;
     int y;
public:
    Point(int x,int y):x(x),y(y){}
};

class Rectangle: public Shape
{
     int width;
     int height;
     Point* leftUp;
public:
     Rectangle(int width, int height, int x, int y);
     Rectangle(const Rectangle& other);
     Rectangle& operator=(const Rectangle& other);
     ~Rectangle();
};

inline
Rectangle::Rectangle(int width,int heigt, int x,int y):width(width),height(height),leftUp(new Point(x,y)){}
//尽量使用初始化列表,包括对leftUp的初始化  

inline
Rectangle::Rectangle(const Rectangle& other)
:Shape(other),width(other.width),height(other.height){  // 注意对继承而来对象no的拷贝构造,通过调用父类的拷贝构造函数
    if(other.leftUp != NULL){                         //针对other.leftUp是否为空的讨论,空指针情况没必要在堆内生成对象
        this->leftUp = new Point(*other.leftUp);    //调用Point的拷贝构造函数,这样在Point内部有变化时,避免大量修改。
    }
    else{
        this->leftUp = NULL;                          //leftUp初始默认是随机值,要对他进行赋值为空指针。
    }
}

Rectangle& Rectangle::operator= (const Rectangle& other){
    if(this == &other){                 //赋值操作符往往需要首先判定自己给自己赋值的情况,避免内存泄露
        return *this;
    }
    Shape::operator=(other);     //调用父类的赋值操作符,完成对父类继承部分的赋值操作
    this->width = other.width;
    this->height = other.height;

    if(other.leftUp != NULL){
        if(leftUp != NULL) {
            *leftUp = *other.leftUp;             //不必删除leftUp再重新构建,直接进行赋值即可(解指针,调用point类的赋值操作符即可)
        }
        else{
            leftUp = new Point(*other.leftUp);      //leftUp为空,不能解指针,需要创建一个新对象
        }
    }
    else{
        delete leftUp;
        this->leftUp = NULL;
    }
    return *this;
} 

Rectangle:: ~Rectangle(){
    delete leftUp;
} 
时间: 2024-10-05 05:02:12

作业分析 c++四大函数练习的相关文章

【E2LSH源码分析】E2LSH函数接口

上一小节,我们对E2LSH代码主要的源码框架和用到的数据结构作了简单介绍(http://blog.csdn.net/jasonding1354/article/details/38331229),这一节我将简单介绍一下E2LSH的主要函数接口. 1.构建R-NN数据结构(定义在NearNeighbors.h) 输入为1-δ.R.d和数据集P,利用下面函数来构建: PRNearNeighborStructT initSelfTunedRNearNeighborWithDataSet(RealT t

使用xdebug分析thinkphp框架函数调用图

开发中需要性能调优,使用xdebug分析thinkphp框架函数调用图. 关于xdebug的安装参考这2篇 NetBeans配置Xdebug 远程调试PHP php扩展xdebug安装以及用kcachegrind系统分析 1.安装xdebug 需要先去http://www.xdebug.org看看一些文档,xdebug作为php扩展安装 # http://www.xdebug.org/files/xdebug-2.3.3.tgz # tar -xzf xdebug-2.3.3.tgz# cd x

gcc源代码分析,debug_tree()函数,又一利器啊

gcc源代码分析,debug_rtx()函数,利器啊 print-tree.c #include "config.h" #include "tree.h" #include <stdio.h> /* Names of tree components. Used for printing out the tree and error messages.  */ #define DEFTREECODE(SYM, NAME, TYPE, LEN) NAME,

Yii2.0源码分析之——设置别名函数(setAlias)和获取别名函数(getAlias)

首先说说什么是别名.在Yii中有很多的路径,在开发的过程当前我们也会使用一些路径.一般来说都需要使用绝对路径,但绝对路径都很长.所以,为了方便的使用路径,可以在Yi中i给每个路径起个名称,这个名称就是别名.别名的格式: 别名必须以"@"字符开头,别名中还可以包含"/".如("@www"为根别名,"@www/test"就为子别名) 别名最后的目录分隔符("\"或者"/")都将去掉(如果有的

从string类的实现看C++类的四大函数

很久之前参加过一次面试,记得当时面试官问过我一个很基础的代码题:实现string类的四大基本函数! 一个C++类一般至少有四大函数,即构造函数.拷贝构造函数.析构函数和赋值函数,一般系统都会默认.但是往往系统默认的并不是我们所期望的,为此我们就有必要自己创造他们.在创造之前必须了解他们的作用和意义,做到有的放矢才能写出有效的函数. 1 #include <iostream> 2 3 class CString 4 { 5 friend std::ostream & operator&l

jQuery分析(2) - $工厂函数分析

jQuery分析(2) - $工厂函数分析 前言 从这节进入jQuery的世界,首先从jQuery的入口函数开始了解jQuery()或$是如何运作的,这里我给出了一个最小的例子来分析. 回忆 在进入分析代码前我们回想下jQuery的使用方法有哪些呢? selector,[context] selector:用来查找的字符串 context:作为待查找的 DOM 元素集.文档或 jQuery 对象. element 一个用于封装成jQuery对象的DOM元素 object 一个用于封装成jQuer

ViewPager 详解(二)---详解四大函数

前言:上篇中我们讲解了如何快速实现了一个滑动页面,但问题在于,PageAdapter必须要重写的四个函数,它们都各有什么意义,在上节的函数内部为什么要这么实现,下面我们就结合Android的API说明,详细讲解一下. 相关文章: 1.<ViewPager 详解(一)---基本入门> 2.<ViewPager 详解(二)---详解四大函数> 3.<ViewPager 详解(三)---PagerTabStrip与PagerTitleStrip添加标题栏的异同> 4.<

【P000-009】分析模块——四大类

仅仅将分析模块按实际惯例简单分成价值和价格两类看来并不准确.一家公司便宜绝对不等于能赚钱,公司所对应的股票也不会因为公司现在赚钱(亏钱)就一定涨(跌),当然长期来看,股票价格运行是与公司效益成正比的.也就是俗话说 (短期)市场的错误,和(长期)市场的正确. 不过这个展开来说又复杂了,抽象建模要的就是化繁为简,先打住. 公司战略,市场环境,投资倾向什么的全部屏蔽,我们现在要的只是一些数据 一.这些数据按来源可以分为两个部分 1.来自于公司 2.来自于(二级)市场(交易市场) 二.而按照时间连续性也

&lt;Spark&gt;&lt;Spark Streaming&gt;&lt;作业分析&gt;

Intro 这篇是对一个Spark (Streaming)作业的log进行分析.用来加深对Spark application运行过程,优化空间的各种理解. Here to Start 从我这个初学者写得一个Spark Streaming程序开始... package com.wttttt.spark /** * Created with IntelliJ IDEA. * Description: * Author: wttttt * Github: https://github.com/wttt