拷贝构造函数和const成员函数

实验原因

说明如何使用const描述保护类数据不会意外修改.

编译环境

vc6sp6 + win7x64

工程下载

copyConstruction_constMemberFunction.zip

使用非const成员函数,引起的拷贝构造函数报错

[cpp] view plain copy

  1. class CStudent
  2. {
  3. /// 常量定义
  4. public:
  5. enum {NAME_SIZE_MAX = 64};
  6. /// 构造, 拷贝构造, 析构函数
  7. public:
  8. CStudent();
  9. CStudent(const char* pcName, size_t nId = 1, int iAge = 20);
  10. CStudent(const CStudent& src);
  11. virtual ~CStudent();
  12. /// 成员函数
  13. public:
  14. void clear();
  15. private:
  16. void init(); ///< 类初始化
  17. void uninit(); ///< 类反初始化
  18. void copy(CStudent& src);
  19. /// setter, getter
  20. public:
  21. // m_nId
  22. void setter_m_nId(size_t nIn) {m_nId = nIn;}
  23. size_t getter_m_nId() {return m_nId;}
  24. // m_cName
  25. void setter_m_cName(const char* pcIn)
  26. {
  27. if (NULL == pcIn)
  28. return;
  29. memset(m_cName, ‘\0‘, NAME_SIZE_MAX);
  30. strncpy(m_cName, pcIn,
  31. (strlen(pcIn) < (NAME_SIZE_MAX - 1)) ?
  32. strlen(pcIn) : (NAME_SIZE_MAX - 1));
  33. }
  34. const char* getter_m_cName() {return m_cName;}
  35. // m_iAge
  36. void setter_m_iAge(int iIn) {m_iAge = iIn;}
  37. int getter_m_iAge() {return m_iAge;}
  38. /// 成员变量
  39. private:
  40. size_t m_nId; ///< 学号
  41. char m_cName[NAME_SIZE_MAX]; ///< 姓名
  42. int m_iAge; ///< 年龄
  43. };

[cpp] view plain copy

  1. CStudent::CStudent(const CStudent& src)
  2. {
  3. copy(src);
  4. }
  5. void CStudent::copy(CStudent& src)
  6. {
  7. setter_m_cName(src.getter_m_cName());
  8. setter_m_nId(src.getter_m_nId());
  9. setter_m_iAge(src.getter_m_iAge());
  10. }

拷贝构造函数声明为(const class&), 但是拷贝构造函数调用了非const成员函数, 会报错

error C2664: ‘copy‘ : cannot convert parameter 1 from ‘const class CStudent‘ to ‘class CStudent &‘

如果要查报错资料资料, 查一下关键字.

error C2664: cannot convert parameter 1 from ‘const class ‘ to ‘class &‘

需要将拷贝构造函数(直接, 简捷)调用的成员函数全部改成const成员函数,

需要将拷贝构造函数调用的成员函数入参为class& 或 class*全部改成 const class& 或 const class*

[cpp] view plain copy

  1. class CStudent
  2. {
  3. /// 常量定义
  4. public:
  5. enum {NAME_SIZE_MAX = 64};
  6. /// 构造, 拷贝构造, 析构函数
  7. public:
  8. CStudent();
  9. CStudent(const char* pcName, size_t nId = 1, int iAge = 20);
  10. CStudent(const CStudent& src);
  11. CStudent(const CStudent* psrc);
  12. virtual ~CStudent();
  13. /// 成员函数
  14. public:
  15. void clear();
  16. private:
  17. void init(); ///< 类初始化
  18. void uninit(); ///< 类反初始化
  19. void copy(const CStudent* psrc);
  20. /// setter, getter
  21. public:
  22. // m_nId
  23. void setter_m_nId(size_t nIn) {m_nId = nIn;}
  24. size_t getter_m_nId() const {return m_nId;}
  25. // m_cName
  26. void setter_m_cName(const char* pcIn)
  27. {
  28. if (NULL == pcIn)
  29. return;
  30. memset(m_cName, ‘\0‘, NAME_SIZE_MAX);
  31. strncpy(m_cName, pcIn,
  32. (strlen(pcIn) < (NAME_SIZE_MAX - 1)) ?
  33. strlen(pcIn) : (NAME_SIZE_MAX - 1));
  34. }
  35. const char* getter_m_cName() const {return m_cName;}
  36. // m_iAge
  37. void setter_m_iAge(int iIn) {m_iAge = iIn;}
  38. int getter_m_iAge() const {return m_iAge;}
  39. /// 成员变量
  40. private:
  41. size_t m_nId; ///< 学号
  42. char m_cName[NAME_SIZE_MAX]; ///< 姓名
  43. int m_iAge; ///< 年龄
  44. };

[cpp] view plain copy

  1. CStudent::CStudent(const CStudent& src) ///< 入参改成const class&
  2. {
  3. copy(&src);
  4. }
  5. CStudent::CStudent(const CStudent* psrc)
  6. {
  7. copy(psrc);
  8. }
  9. void CStudent::copy(const CStudent* psrc)
  10. {
  11. setter_m_cName(psrc->getter_m_cName());
  12. setter_m_nId(psrc->getter_m_nId());
  13. setter_m_iAge(psrc->getter_m_iAge());
  14. }

const 成员函数的含义

摘录自 <<C++ const详解>>

http://blog.csdn.net/zhuanshenweiliu/article/details/38223907

3. const成员函数
任何不会修改数据成员(即函数中的变量)的函数都应该声明为const类型。

如果在编写const成员函数时,不慎修改了数据成员,或者调用了其它非const成员函数,编译器将指出错误,这无疑会提高程序的健壮性。

以下程序中,类stack的成员函数GetCount仅用于计数,从逻辑上讲GetCount应当为const函数。编译器将指出GetCount函数中的错误。
class Stack
{
public:
void Push(int elem);
int Pop(void);
int GetCount(void) const; // const成员函数
private:
int m_num;
int m_data[100];
} ;
int Stack::GetCount(void) const
{
++ m_num; //编译错误,企图修改数据成员m_num
Pop(); //编译错误,企图调用非const函数
return m_num;
}
const 成员函数的声明看起来怪怪的:const 关键字只能放在函数声明的尾部,大概是因为其它地方都已经被占用了。
关于Const函数的几点规则:
a. const对象只能访问const成员函数,而非const对象可以访问任意的成员函数,包括const成员函数.
b. const对象的成员是不可修改的,然而const对象通过指针维护的对象却是可以修改的.
c. const成员函数不可以修改对象的数据,不管对象是否具有const性质.它在编译时,以是否修改成员数据为依据,进行检查.

http://blog.csdn.net/lostspeed/article/details/50291749

时间: 2024-10-12 16:19:43

拷贝构造函数和const成员函数的相关文章

第五篇:明确拒绝不想编译器自动生成的拷贝构造函数和赋值运算符重载函数

前言 如果你不想要编译器帮你自动生成的拷贝机制 (参考前文),那么你应当明确的拒绝. 如何拒绝?这便是本文要解决的主要问题. 问题描述 当你定义了一个类,而这个类中各对象之间也是封装的 - 禁止同类对象之间的相互赋值以及复制,那么你需要屏蔽掉编译器帮你生成的拷贝构造函数以及赋值运算符. 在许多代码中,会看到通过将拷贝构造函数和赋值运算符重载函数声明为私有且不予实现来实现这个功能.然而,这不是最科学的做法. 因为这没有做到真正的屏蔽:你在自己的成员函数中,或者友元函数中仍然可以调用这两个私有函数,

明确拒绝不想编译器自动生成的拷贝构造函数和赋值运算符重载函数

前言 如果你不想要编译器帮你自动生成的拷贝机制 (参考前文),那么你应当明确的拒绝. 如何拒绝?这便是本文要解决的主要问题. 问题描述 当你定义了一个类,而这个类中各对象之间也是封装的 - 禁止同类对象之间的相互赋值以及复制,那么你需要屏蔽掉编译器帮你生成的拷贝构造函数以及赋值运算符. 在许多代码中,会看到通过将拷贝构造函数和赋值运算符重载函数声明为私有且不予实现来实现这个功能.然而,这不是最科学的做法. 因为这没有做到真正的屏蔽:你在自己的成员函数中,或者友元函数中仍然可以调用这两个私有函数,

C++ Primer 学习笔记_24_类与数据抽象(10)--static 与单例模式、auto_ptr与单例模式、const成员函数、const 对象、mutable修饰符

C++ Primer 学习笔记_24_类与数据抽象(10)--static 与单例模式.auto_ptr与单例模式.const成员函数.const 对象.mutable修饰符 前言 [例]写出面向对象的五个基本原则? 解答:单一职责原则,开放封闭原则,依赖倒置原则,接口隔离原则和里氏替换原则 里氏替换原则:子类型必须能够替换他们的基类型. 设计模式分为三种类型:创建型模式.结构型模式和行为型模式 一.static 与单例模式 1.单例模式 单例模式的意图:保证一个类仅有一个实例,并提供一个访问它

C++基础知识--static成员和const成员函数

我们可以用static修饰一个类的成员函数,也可以用const修饰类的成员函数(写在函数的最后表示不能修改成员变量,不是指写在前面表示返回值为常量).能不能同时用static和const修饰类的称冠函数? 不能.C++编译器在实现const成员函数的时候为了确保该函数不能修改类的实例的状态,会在函数中添加一个隐式的参数const this*.但当一个成员为static的时候,该函数是没有this指针的.也就是说此时static的用法和static是冲突的. Static的作用是表示该函数只那个在

拷贝构造函数与赋值运算符重载函数要点

拷贝构造函数 一个小例子 最近在<剑指Offer>上看到了一道题(程序如下),要求我们分析编译运行的结果,并提供3个选项: A. 编译错误: B. 编译成功,运行时程序崩溃:C. 编译运行正常,输出10. 1 #include <iostream> 2 using namespace std; 3 4 class A 5 { 6 private: 7 int value; 8 9 public: 10 A(int n) { value = n; } 11 A(A other) {

const成员函数,内联函数,友元函数,static成员

原文链接:https://blog.csdn.net/qq_38646470/article/details/79795538专栏链接:https://blog.csdn.net/column/details/20026.html[TOC]#1.const成员函数const修饰成员函数在成员函数后面加const,const修饰this指针所指向的对象,也就是保证调用这个const成员函数的对象在函数内不会被改变. class Date { public : void Display () { c

(转)函数后面加const--C++ const成员函数

类的成员函数后面加 const,表明这个函数不会对这个类对象的数据成员(准确地说是非静态数据成员)作任何改变. 在设计类的时候,一个原则就是对于不改变数据成员的成员函数都要在后面加 const,而对于改变数据成员的成员函数不能加 const.所以 const 关键字对成员函数的行为作了更加明确的限定:有 const 修饰的成员函数(指 const 放在函数参数表的后面,而不是在函数前面或者参数表内),只能读取数据成员,不能改变数据成员:没有 const 修饰的成员函数,对数据成员则是可读可写的.

c++中的const参数,const变量,const指针,const对象,以及const成员函数

const 是constant 的缩写,“恒定不变”的意思.被const 修饰的东西都受到强制保护,可以预防意外的变动,能提高程序的健壮性.所以很多C++程序设计书籍建议:“Use const whenever you need”. 1.用const 修饰函数的参数 如果参数作输出用,不论它是什么数据类型,也不论它采用“指针传递”还是“引用传递”,都不能加const 修饰,否则该参数将失去输出功能.const 只能修饰输入参数: 如果输入参数采用“指针传递”,那么加const 修饰可以防止意外地

c++ const: const 变量, const 输入参数, const 返回值, const 成员函数

看到const 关键字,C++程序员首先想到的可能是const 常量.这可不是良好的条件反射.如果只知道用const 定义常量,那么相当于把火药仅用于制作鞭炮.const 更大的魅力是它可以修饰函数的参数.返回值,甚至函数的定义体. const 是constant 的缩写,“恒定不变”的意思.被const 修饰的东西都受到强制保护,可以预防意外的变动,能提高程序的健壮性.所以很多C++程序设计书籍建议:“Use const whenever you need”. 1.用const 修饰函数的参数