《c++语言的设计和演化》笔记(四)

2024-06-13 00:08
文章标签 语言 c++ 设计 笔记 演化

本文主要是介绍《c++语言的设计和演化》笔记(四),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

The Design and Evolution of C++

  • 13.2 抽象类
    • 13.2.4 虚函数和构造函数
      • 13.2.4.2 基类优先的构造
  • 13.3 const成员函数
    • 13.3.3 可变性与强制
  • 13.4 静态成员函数
    • 14.2.5 typeid()运算符
      • 14.2.5.1 类type_info
  • 15.1 引言
  • 15.3 类模板

13.2 抽象类

13.2.4 虚函数和构造函数

13.2.4.2 基类优先的构造

1.“构造函数就是要建立起一个环境,使其他成员函数在其中操作。”(p.251~252)
2.“考虑下面这个可能引起混乱的例子:

class B 
{
public:int b;virtual void f();void g();//...B();
};class D: public B
{
public:X x;void f();//...D();
};B::B()
{b++; //undefined: B::b isn't yet initialized.f(); //calls: B::f(); not D::f().
}

如果你真的希望调用B自己的f(),那就应该将它明确地写成B::f()。
这个构造函数的行为方式与写常规成员函数可能的方式成鲜明的对比,因为常规成员函数可以依靠构造函数的正确行为:

void B::g()
{b++; //fine, since B::b is a member, B::B should have initialized it.f(); //calls: D::f() if B::g() is called for a D.
}

当一个调用出自某个D的B部分时,在B::B()和B::g()里的f()调用的是不同函数。”(p.252)
3.“如果让构造函数去调用覆盖函数,构造函数的用途将受到严重的限制,以至于我们根本无法合理地编写覆盖函数了。
在这里,基本的设计要点是,直到对一个对象的构造函数的运行结束之前,这个对象就一直像一个正在建造之中的建筑物:你必须忍受结构没有完工所带来的各种不便,常常需要依靠临时性的脚手架,必须时时当心在与危险环境相处的各种问题。一旦构造函数返回,编译程序和用户就都可以假定构造完成的对象能使用了。”(p.253)

13.3 const成员函数

1.“我们需要一种方法,使程序员可以说明哪些成员函数将更新其对象的状态,而哪些并不更新:

class X	{int aa;
public:void update() { aa++; }int value() const {return aa;}void cheat() const { aa++; } //error: *this is const
}

声明为const的成员函数,如X::value(),被称为const成员函数,并保证不会修改对象的值。const成员函数可以用于const对象和非const对象,而非const成员函数,如X::update(),就只能用于非const对象…从技术上说,得到这种行为的方式就是要求X的非const成员函数里的this指针指向X,而让其const成员函数里的this指针只能指向const X。
”(p.253~254)
2.“将一个对象声明为const,就是认为它具有从其构造函数完成到析构函数的开始之间的不变性。”(p.255)

13.3.3 可变性与强制

1.“有些人还是特别讨厌强制去掉const,因为它是一个强制,甚至更因为这种东西并不保证对所有情况都能工作…应该能描述一种绝不应该被认为是const的成员,即使它是某个const对象的成员时也是这样…初始建议提出用‘~const’作为‘绝不能是const’的记法。甚至整个概念的一些拥护者也认为这个记法太难看,所以把关键字mutable引进建议里,被ANSI/ISO委员会接受:

class XXX {int a;mutable int cnt; //cnt will never be const
public:int f() const { cnt++; return a;}//...
};XXX var; //var.cnt is writable (of course)const XXX cnst; // cnst.cnt is writable because XXX::cnt is declared mutable

”(p.255~256)
2.“类的static数据成员是这样的一种成员,它只存在一个唯一的备份,而不像其他成员那样在每个对象中各有一个备份。因此,不需要引用特定对象就可以访问static成员。static成员可用于减少全局名称的数量,并且能把某个static成员在逻辑上属于哪个类的问题表述明确,还能实现对这些名称的访问控制。这种特性对于库的提供商都是非常重要的,因为它能够防止对全局名称空间的污染,并可以简化库代码的书写,也使同时使用多个库变得更加安全。”(p.256)

13.4 静态成员函数

1.“static成员函数并不关联任何特定对象,因而不需要用特定成员函数的语法进行调用。”(p.257)
2.“在某些情况下,类被简单地当作一种作用域来使用,把不放进来就是全局的名称放入其中,作为它的static成员,可以使这些名称不会污染全局的名称空间。”(p.257)

14.2.5 typeid()运算符

1.“可能需要确定一个对象的确切类型。也就是说,告诉说这个对象就是X类的对象,而不是只说,它是X类的或者某个由X类派生的类的对象。dynamic_cast做的是后一件事情。”(p.281)
2.“人们希望知道一个对象的确切类型,通常是他们因为想对这个对象的整体执行某种标准服务。”(p.281)

14.2.5.1 类type_info

1.“函数before()是为了使type_info信息能够排序,以便能通过散列表等方式访问它们。由before()定义的顺序关系和继承关系之间没有任何联系。进一步说,对不同的程序或者同一个程序的不同运行,我们都不能保证before()能产生同样的结果。在这个方面,before()与取地址运算符类似。”(p.282)

15.1 引言

1.“模板概念植根于对描述参数化容器类的愿望:异常来自于渴望为运行时错误的处理提供一种标准化方式。”(p.298)

15.3 类模板

1.“一个C++的参数化类型被称为一个类模板。类模板描述了可以如何构造出一些个别的类,其方式很像在类里描述如何构造起个别的对象。一个向量的模板类可以像下面这样声明:

template<class T> 
class vector {T* V;int sz;
public:vector(int);T& operator[](int);T& elem(int i){return v[i];}
};

前缀template<class T>说明了这里声明的是一个模板,它有一个类型为T的参数类型将在声明中使用。将其引入后,在模板的作用域里,T就可以像其他类型名称一样使用了。向量模板可以像下面这样引用:

vector<int> v1(20);
vector<complex> v2(30);typedef vector<complex> cvec; //make cvec a synonym for vector<complex>.cvec v3(40); //v2 and v3 are of the same type.void f()
{v1[3]=7;v2[3]=v3.elem(4)=complex(7,8);
}


与类的声明相比,声明一个类模板并不复杂多少。关键字class用于指明类型参数的类型部分,一是因为它以很清楚的词的形式出现;二是因为这样可以节约一个关键字。在这个上下文环境里,class的意思是‘任意类型’,而不仅是‘某种用户定义类型’。
在这里使用尖括号<…>而不使用圆括号(…),是为了强调模板参数具有不同的性质(它们将在编译时求值),也因为圆括号在C++里已经过度使用了。
引进关键字template使模板声明很容易看清楚,无论是对人,还是对工具,同时也为模板类和模板函数提供了一种共有的语法形式。
模板是为生成类型提供的一种机制。它们本身并不是类型,也没有运行时的表示形式,因此它们对于对象的布局没有任何影响。”(p.301-302)
2.“除了类型参数之外,C++也允许非类型的模板参数。这种机制基本上被看做是为容器类提供大小和限界所需的信息。例如:

template<class T, int i>
class Buffer{T v[i];int sz;
public:Buffer():sz(i) {}//...
};

在那些运行时间、效率和紧凑性非常紧要的地方,为了能与C语言的数组和结构竞争,这样的模板就非常重要了。传递大小信息允许实现者不使用自由空间。”(p.303)
3.“在模板的初始设计中,不允许用名称空间或模板作为模板的参数,这一限制是过于谨慎的又一案例。我现在想不出任何理由去禁止这种参数,它们无疑是很有用的。以类模板作为模板参数,已在1994年3月圣迭戈会议上获得通过。”(p.303)

这篇关于《c++语言的设计和演化》笔记(四)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1055726

相关文章

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Go语言如何判断两张图片的相似度

《Go语言如何判断两张图片的相似度》这篇文章主要为大家详细介绍了Go语言如何中实现判断两张图片的相似度的两种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 在介绍技术细节前,我们先来看看图片对比在哪些场景下可以用得到:图片去重:自动删除重复图片,为存储空间"瘦身"。想象你是一个

Go语言中Recover机制的使用

《Go语言中Recover机制的使用》Go语言的recover机制通过defer函数捕获panic,实现异常恢复与程序稳定性,具有一定的参考价值,感兴趣的可以了解一下... 目录引言Recover 的基本概念基本代码示例简单的 Recover 示例嵌套函数中的 Recover项目场景中的应用Web 服务器中

MyBatis设计SQL返回布尔值(Boolean)的常见方法

《MyBatis设计SQL返回布尔值(Boolean)的常见方法》这篇文章主要为大家详细介绍了MyBatis设计SQL返回布尔值(Boolean)的几种常见方法,文中的示例代码讲解详细,感兴趣的小伙伴... 目录方案一:使用COUNT查询存在性(推荐)方案二:条件表达式直接返回布尔方案三:存在性检查(EXI

C/C++ chrono简单使用场景示例详解

《C/C++chrono简单使用场景示例详解》:本文主要介绍C/C++chrono简单使用场景示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友... 目录chrono使用场景举例1 输出格式化字符串chrono使用场景China编程举例1 输出格式化字符串示

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以

C++类和对象之默认成员函数的使用解读

《C++类和对象之默认成员函数的使用解读》:本文主要介绍C++类和对象之默认成员函数的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、默认成员函数有哪些二、各默认成员函数详解默认构造函数析构函数拷贝构造函数拷贝赋值运算符三、默认成员函数的注意事项总结一

C/C++中OpenCV 矩阵运算的实现

《C/C++中OpenCV矩阵运算的实现》本文主要介绍了C/C++中OpenCV矩阵运算的实现,包括基本算术运算(标量与矩阵)、矩阵乘法、转置、逆矩阵、行列式、迹、范数等操作,感兴趣的可以了解一下... 目录矩阵的创建与初始化创建矩阵访问矩阵元素基本的算术运算 ➕➖✖️➗矩阵与标量运算矩阵与矩阵运算 (逐元