构造、析构、赋值运算
c++会为一个空类声明一个copy构造函数,一个copy assignment操作符和一个析构函数
如果没有声明构造函数,还会生成一个default构造函数
示例代码如下:
class Empty{...};
等同于
class Empty{
//default构造函数
Empty(){...}
//copy构造函数
Empty(const Empty& rhs){...}
//析构函数 non-virtual
~Empty(){...}
//copy assignment操作符
Empty& operator=(const Empty& rhs){...}
};
编译器会产生出
Empty el;//default构造函数
Empty e2(e1);//copy构造函数
e2=e1;//copy assignment操作符
copy构造函数和copy assignment操作符,编译器创建的版本只是单纯地将源对象的每一个
non-static成员变量拷贝到目标对象
示例代码如下:
template<typename T>
class NamedObject{
public:
//由于声明了构造函数,编译器不再生成default构造函数
NamedObject(const char* name,const T& value);
NamedObject(const std::string& name,const T& value);
...
private:
std::string nameValue;
T objectValue;
};
由于NamedObject函数没有声明copy构造函数和copy assignment操作符,编译器会自动创建
具体用法如下:
NamedObject<int> no1("Sample Prime Number",2);
//调用copy构造函数,以no1.nameValue和no1.objectValue为no2的两个属性符值
NamedObject<int> no2(no1);
copy assignment操作符
示例代码如下:
template<class T>
class NamedObject{
public:
NamedObject(std::string& name,const T& value);
...
private:
std::string& nameValue;
const T objectValue;
};
使用场景:
std::string newDog("persephone");
std::string oldDog("satch");
NamedObject<int> p(newDog,2);
NamedObject<int> s(oldDog,36);
p=s;
如果某对象是独一无二(不允许复制),就需要拒绝编译器自动生成的函数
即将copy构造器和copy assignment操作符声明为private
即使这样member函数和friend函数还是会调用private函数,因此还要不去定义private函数
可以从标准程序库实现代码中查看如ios_base basic_ios和sentry
示例代码如下:
class HomeForSale{
public:
...
//copy构造函数和copy assignment操作符
private:
...
HomeForSale(const HomeForSale&);
HomeForSale& operator=(const HomeForSale&);
};
如果想拷贝HomeForSale对象编译器会阻止,如果是在member函数和friend函数中调用
连接器会阻止
如果想在编译时阻止,可以用以下方法实现
class Uncopyable{
protected:
Uncopyable(){}
~Uncopyable(){}
private:
Uncopyable(const Uncopyable&);
Uncopyable& operator=(const Uncopyable&);
};
//目标函数继承Uncopyable函数
//此时不用声明copy构造函数和copy assignment操作符
class HomeForSal: private Uncopyable{...};
为多态基类声明virtual析构函数
示例代码如下:
class TimeKeeper{
public:
Timekeeper();
~TimeKeeper();
...
};
class AtomicClock:public TimeKeeper{...}; //原子钟
class WaterClock:public TimeKeeper{...}; //水钟
class WristWatch:public TimeKeeper{...};//腕表
需要一个factory函数返回一个计时对象
TimeKeeper* getTimeKeeper();//返回一个指向TimeKeeper派生类的动态对象
Timekeeper* ptk=getTimeKeeper();
delete ptk;//释放资源
注:getTimeKeeper返回的指针指向一个derived class(派生类)对象,这个对象由一个
base class指针被删除,而base class(TimeKeeper)有个non-virtual析构函数,执行时
对象的derived部分没有被销毁(derived class的析构函数未执行)
解决办法如下:
class TimeKeeper{
public:
TimeKeeper();
virtual ~TimeKeeper();
...
};
TimeKeeper* ptk=getTimeKeeper();
delete ptk;
如果一个函数不打算用作base class时,就不需要使其构造函数为virtual
避免析构函数中报异常
示例代码如下:
class Widget{
public:
...
~Widget(){...}//假设这里出现异常
};
void doSomething(){
std::vector<widget> v;
...
} //v在这里被自动销毁
//联接数据库
class DBConnection{
public:
...
static DBConnection create(); //返回DBConnection对象
void close();
}
//用来管理DBConnection
class DBConn{
public:
...
//关闭联接
~DBConn(){
db.close();
}
private:
...
DBConnection db;
};
为了防止在DBConn的析构函数中调用close()方法发生异常
处理方法如下:
DBConn::~DBConn(){
try{db.close();}
catch(...){
//制作运转记录,记录对close的调用失败
std::abort();
}
}
//采用独立函数处理异常
class DBConn{
public:
...
void close(){
db.close();
closed=true;
}
~DBConn(){
if(!close){
try{
db.close();
}
catch(){
//制作运转记录,记录close调用失败
}
}
}
private:
DBConnection db;
bool closed;
}
不在构造函数和析构函数中调用virtual函数,回为这类调用从不下降至derived class
示例代码如下:
//所有交易的base class
class Tranxaction{
public:
Transaction();
//区分类型的目志记录
virtual void logTransaction() const=0;
...
};
//base class 实现
Transaction::Transaction(){
...
logTransaction();
}
//子类
class BuyTransaction:public Transaction{
public:
//日志记录此类型交易
virtual void logTransaction() const;
...
};
//子类
class SellTransaction:public Transaction{
public:
//日志记录此类型交易
virtual void logTransaction() const;
};
当执行BuyTransaction b;时首先Transaction构造函数会更早被调用
替换方法
class Transaction{
public:
explicit Transaction(const std::string& logInfo);
void logTransaction(const std::string& logInfo) const;//non_virtual函数
...
};
Transaction::Transaction(const std::string& logInfo){
...
logTransaction(logInfo);
}
class BuyTransaction:public Transaction{
public:
//将log信息传递给base class 构造函数
BuyTransaction(parameters):Transaction(createLogString(parameters)){...}
...
private:
static std::string createLogString(parameters);
};
让operator=返回一个reference to *this
赋值
int x,y,z;
x=y=z=15;
x=(y=(z=15));
示例代码如下:
class Widget{
public:
...
//返回类型是个reference,指向当前类型
Widget& operator+=(const Widget& rhs){
...
return *this;
}
Widget& operator=(int rhs){
...
return *this;
}
}
在operator=中处理"自我赋值"
//对象在赋值给自已时
class Widget{...};
Widget w;
...
w=w;
示例代码如下:
//保存一个指针指向一块动态分配的内存
class Bitmap{...};
class Widget{
...
private:
Bitmap* pb;
};
//operator=实现代码
Widget& widget::operator=(const Widget& rhs){
//如果是自我赋值就不作任何事
if(this==&rhs) return *this;
delete pb;
pb=new Bitmap(*rhs.pb);
return *this
//以上三行代码的异常处理
Bitmap* pOrig=pb;
pb=new Bitmap(*ths.pb);
delete pOrig;
return *this;
}
更好的实现,即处理异常安全,又处理自我赋值
class Widget{
...
void swap(Widget& rhs);//交换*this 和rhs的数据
};
Widget& Widget::operator=(const Widget& rhs){
Widget temp(rhs);
swap(temp);
return *this;
}
复制对象时注意其每一成员
示例代码如下:
void logCall(const std::string& funcName);
class Customer{
public:
...
Customer(const Customer& rhs);
Customer& operator=(const Customer& rhs);
...
private:
std::string name;
};
Customer::Customer(const Customer& rhs):name(rhs.name){//复制rhs的数据
logCall("Customer copy constructor");
}
Customer& Customer::operator=(const Customer& rhs){
logCall("Customer copy assignment operator");
name=rhs.name; //复制rhs数据
return *this;
}
当customer 中添加一个成员时,copy构造函数和copy assignment都要添加
如果是子类
PriorityCustomer:public Customer{
public:
...
PriorityCustomer(const PrivrityCustomer& rhs);
PriorityCustomer& operator=(const PriorityCustomer& rhs);
...
private:
int priority;
};
PriorityCustomer::PriorityCustomer(const PriorityCustomer& rhs)
:Customer(rhs), //调用base class 的copy构造函数
priority(rhs.priority){
logCall("PriorityCustomer copy constructor");
}
PriorityCustomer& PriorityCustomer::operator=(const PriorityCustomer& rhs){
logCall("PriorityCustomer copy assignment operator");
Customer::operator=(rhs);//对base class成分进行赋值动作
priority=rhs.priority;
return *this;
}
- 浏览: 261396 次
- 性别:
- 来自: 济南
文章分类
- 全部博客 (303)
- c (31)
- c++ (16)
- java (18)
- c# (1)
- python (3)
- java web (6)
- oracle (7)
- sqlserver (2)
- mysql (2)
- android (24)
- android系统 (15)
- android多媒体部分 (15)
- android游戏 (12)
- linux (26)
- javaScript (1)
- ajax (1)
- node JS (2)
- html (5)
- apache (3)
- jboss (1)
- weblogic (0)
- 通信协议 (10)
- 云计算 (1)
- 分布式 (5)
- ejb (1)
- webservice (5)
- 设计模式 (16)
- JNI (6)
- swing (13)
- 版本控制 (1)
- UML (1)
- xml (4)
- spring (5)
- hibernate (5)
- struts1 (3)
- struts2 (4)
- ibatis (0)
- tomcat (2)
- 心得体会 (1)
- css (1)
- 嵌入式 (41)
- arm体系结构 (10)
发表评论
-
c++学习笔记十六
2013-05-14 21:50 820c with classes 尽量以const e ... -
c++学习笔记一
2012-09-03 15:25 606基本概念 第一个程序 h ... -
c++学习笔记二
2012-09-03 15:26 362处理基本数据类型 1 混合表达式 (向上转型) 2 赋 ... -
c++学习笔记三
2012-09-03 15:26 545选择和决策 比较数据值 1 关系运算符:< &g ... -
c++学习笔记五
2012-09-03 15:27 599while do-while for循环 循环和变 ... -
c++学习笔记六
2012-09-03 15:28 268指针 可以使用指针记 ... -
c++学习笔记七
2012-09-03 15:28 559使用函数编程 程序的分解 1 函数 定义函 ... -
c++学习笔记八
2012-09-03 15:29 595函数 1 函数的重载: 函数名相同,参数个数不同 ... -
c++学习笔记九
2012-09-20 23:14 698程序文件和预处理指令 1 程序文件 头文件:类型定 ... -
c++学习笔记十
2012-09-20 23:18 594创建自已的数据类型 1 对像的概念 2 c++中的结构 ... -
c++学习笔记十一
2012-09-20 23:20 240类 1 封装 继承 多 ... -
c++学习笔记十二
2012-09-20 23:20 689类的操作 1 类对象使用指针的三种情况: ... -
c++学习笔记十三
2012-09-20 23:31 668运算符重载 1 为自已的类实现运算符 运 ... -
c++学习笔记十四
2012-09-20 23:32 603继承 1 类和面向对象编程 2 类的继承 继承 ... -
c++学习笔记十五
2012-09-20 23:36 625虚函数和多态性 使用基类指针,示例代码如下(从Box派 ...
相关推荐
### C++ 学习笔记精华版 #### 一、C++ 语言概述 **1、历史背景** - **C++ 的江湖地位** - Java、C、C++、Python、C# 是当前主流的编程语言之一,而 C++ 在这些语言中以其高效性和灵活性著称。 - **C++ 之父 ...
C++学习笔记C++学习笔记C++学习笔记C++学习笔记C++学习笔记
C++学习笔记
这份"C++学习笔记"涵盖了C++的基础到高级概念,旨在帮助初学者和有一定经验的程序员深入理解并掌握C++。 在C++的学习过程中,以下几个关键知识点是不可或缺的: 1. **基础语法**:C++起源于C语言,因此它保留了...
C++是一种静态类型、编译式、通用的编程语言,它支持过程化编程、面向对象编程和泛型编程。C++广泛应用于软件开发领域,包括操作...以上内容覆盖了C++语言中一些基础知识点,对于学习和理解C++编程具有重要的指导意义。
【C++学习笔记】这份详尽的资源是为那些希望深入了解C++编程语言的人们精心准备的。C++是一种强大的、通用的编程语言,它的设计理念是面向对象,同时支持过程化编程,使得它在系统软件、应用软件、游戏开发、设备...
【C++ 学习笔记】深入理解编程与C++基础 C++是一种强大的、面向对象的编程语言,广泛应用于系统软件、游戏开发、嵌入式系统以及许多高性能应用。学习C++不仅仅是掌握语法,更重要的是理解编程的核心概念,以便更好...
C++ 学习笔记C++ 学习笔记C++ 学习笔记C++ 学习笔记002
C++ 学习笔记C++ 学习笔记C++ 学习笔记C++ 学习笔记C++ 学习笔记003
C++ 学习笔记C++ 学习笔记C++ 学习笔记C++ 学习笔记C++ 学习笔记004
【C++学习笔记概述】 C++是一门强大的编程语言,其在C语言的基础上进行了扩展,引入了许多现代化的特性。这份笔记主要涵盖了C++的基础知识,包括C++11和C++17的新特性,旨在帮助初学者理解C++的核心概念。 ### C++...
C++ 学习笔记C++ 学习笔记C++ 学习笔记C++ 学习笔记001
C++学习笔记 本文档提供了C++语言的基础知识,包括输入输出、变量、数据类型、运算符、内存管理、指针、流程控制语句等。 输入输出 C++语言提供了多种输入输出方式,包括使用cin和cout对象。cin对象用于从标准...
《新手C++学习笔记》是一份专为编程初学者打造的资源,旨在帮助那些刚刚踏入C++编程领域的“菜鸟”快速成长。这份笔记包含了前辈们的实践经验总结,具有很高的学习价值。文档以.doc格式存储,方便读者使用常见的文字...
这份"C++学习笔记"涵盖了C++的基础概念、语法结构、面向对象编程以及可能的高级主题,旨在帮助初学者或者有经验的程序员巩固C++知识。 首先,C++是从C语言发展而来,它保留了C语言的效率,同时引入了类和对象的概念...
C++是一种强大的面向对象编程语言,它源自C语言并扩展了其...学习这些概念是成为C++程序员的第一步。在实际编程中,还需要理解类、对象、继承、多态等面向对象编程的概念,以及函数、模板、文件操作等更高级的主题。
《C++学习笔记经典(与C比较)》这份资料应该会详细讲解这些知识点,并通过实例来帮助读者深入理解C++与C的差异,以及如何在实际编程中应用C++的特性和功能。这份资料可能会涵盖基本语法、类和对象、模板、STL的使用...
"黑马C++学习笔记" 本笔记主要记录了C++的基础知识和一些重要的概念,包括变量的声明、赋值、输出、引用、指针、结构体等。 变量声明和赋值 在C++中,变量的声明和赋值是非常重要的。变量可以是整数、浮点数、...