侯捷课程笔记(一)(传统c++语法,类内容)

侯捷课程笔记(一)(传统c++语法,类内容)

2023-09-03更新:
本小节已经完结,只会进行小修改
埋下了一些坑,后面会单独讲或者起新章节讲

最近在学习侯捷的一些课程,虽然其中大部分内容之前也都已经了解过了,不过还是收获颇丰,特别是独立与所谓语法之外的,还有许多与设计相关的。

这一章内容比较简单,也就直接摆出内容吧, 相关重点内容就提一提,大部分都直接掠过了。然后会加入一些自己的侯捷没有讲到的内容。

那就直接开始吧


类设计时头文件防止重复包含

通常用法就是

#ifndef FOO_H__
#define FOO_H__
// ... 主体内容#endif // FOO_H__

因为这个很常用,但是每次都要写三行,也就有了简化版本

#pragma once

不过这个可能需要编译器支持(大部分肯定都是没问题的)


类内成员函数默认就是inline

inline就是内联,inline允许我们在头文件中定义函数重复包含时而不会发生重复定义问题
由于很多时候我们写类是在头文件中(声明在头文件),这时候,如果成员函数定义在类内,不也就是把函数放在了头文件吗,编译器比较智能,也就默认给类内定义的成员函数自动加上了inline属性
如果我们要把类的成员函数写在类外,就没有inline这个属性了,这时候如果还在头文件,我们就要手动加上inline


类使用初始化列表对成员进行初始化

初始化列表其实也就只有两个需要注意的点:

  1. 初始化的顺序不是按照写的顺序来的,而是按照成员变量定义的顺序来的
  2. 如果父类没有默认构造函数,也就是子类必须显式调用函数初始化父类,这时候也就必须要使用初始化列表初始化父类

类内对函数加上const(常函数)

正常的成员函数是可以更改类的,所以对于一个const属性的类,编译器考虑到这个类不可以更改,也就不允许它调用普通的成员函数,只能调用常函数(带const属性的函数)
一个良好的变成习惯是:对于一些不会更改成员变量/类属性的函数,都应该加上const属性,比如获取变量GetVal类似的函数

ps:使用const实例化一个对象,这个类一定要有用户定义的构造函数


对于第一点,比如下面的代码:

struct A {int a;int b;A() : b(1), a(2) {}
};

在这里初始化的顺序实际上是先执行a=2,再执行b=1,因为a是先定义的那个
看起来好像无所谓,但是下面的代码

struct A {int a;int b;A() : b(1), a(b) {}
};

就会出现a先初始化,但是这时候b还没有初始化的问题


对于第二点,比如

struct Base {int a;Base(int){}
};
struct Derive : public Base {Derive(){};
};

父类没有默认的构造函数,子类必须要也只能通过初始化列表初始化父类

struct Derive : public Base {Derive():Base(10){};
};

对于一个类来说,常用的构造函数或者基本架构是什么样的

在侯捷的课程中,常写的是

  • 构造函数
  • 析构函数
  • 拷贝构造函数
  • 拷贝赋值函数

其中如果类中包含指针,常常会自己写拷贝构造和拷贝复制,并会在析构函数中对指针进行处理

而在c++11后引入右值的概念
增加了

  • 移动构造函数
  • 移动复制函数
class MyClass {public:MyClass();MyClass(const MyClass &) = default;MyClass(MyClass &&) = default;MyClass &operator=(const MyClass &) = default;MyClass &operator=(MyClass &&) = default;~MyClass();
};

类中使用友元和重载cout

我们通常会这么写,并且也可以直接把重载函数写在类内

struct A {public:friend std::ostream& operator<<(std::ostream& os,const A& obj) {os << obj.a;return os;}private:int a;
};

可以注意几点:

  • 加上friend就可以访问类中的私有变量
  • 输入的os没有加上const,是因为os执行<<会更改内部内容,无法使用const
  • 输入的引用是为了避免不必要的拷贝,obj加上const是加上&后也可以传入右值
  • 返回引用是可以使用链式编程

带有指针的类的结

如果类内带有指针,比如类的构造函数里会new一块内存。

  • 需要在析构函数内释放对应的内存
  • 需要注意深拷贝和浅拷贝的问题
    • 拷贝构造函数和复制构造函数需要重写,重新开辟内存来进行深拷贝
  • 在赋值构造前需要注意是否是自身赋值(自己赋值给自己),需要判断这种特殊情况以防止bug

比如下面这个示例,注意关注一下前面提到的几点

template <typename T>
struct A {public:A(int n) {size = n;str = new T[n];}~A() { delete[] str; }A(const A& obj) { Copy(obj); }A& operator=(const A& obj) {if (&obj == this) return *this;Copy(obj);return *this;}void Copy(const A& obj) {if (str) delete[] str;size = obj.GetSize();str = new T[size];std::memcpy(str, obj.str, size * sizeof(T));}int GetSize() const { return size; }private:int size = 0;T* str = nullptr;
};

内存分配和管理

这一部分其实单独属于一块内容,后面会单独讲


设计模式(单例模式或者其它)

这一部分其实单独属于一块内容,后面会单独讲


类转换成标准类型(operator int())

类可以使用operator转换成一些类型,在编辑器认为转化可以通过编译时会进行转换,当然我们也可以使用static_cast显式转化

struct A {public:A(int a_) { a = a_; }int a;operator int() {return a;}operator float() {return a;}
}
void Test() {Aa(10);int b=10+static_cast<int>(a);int c=static_cast<float>(a)+100;
}

需要注意的是,这个operator函数并不需要返回值,默认返回值类型就是你写的要转换的类型
这里operator不光可以转为内置类型int/float,还可以转换成自己写的类等等

类通过单参数的构造函数自动转化/explicit

c++可以通过operator将类转化成一些类型,同样也支持反向转化,编译器可以自动根据单参数输入的构造函数,将对应的参数的自动执行构造函数构造对象
💡:这里说的单参数,只指可以输入是一个参数的函数,比如一个函数有三个三数,但是后面两个带了默认参数,也满足单参数;又或者只有一个参数,并且这个参数是默认参数,也属于单参数

struct A {public:A(int a_) { a = a_;}int a;A operator+(const A& obj) {return A(a+obj.a);}
};

比如我要执行: A a(10); A b=a+10; 这里重载了+运算符,c++会自动将后面的10调用构造函数,转换成一个类

如果是A b=10+a就不行
这里的+运算符属于内置的int,在这里就不会默认转换
如果要运行这句话,就要用前面的operator int()

operator int() {return a;
}

运行的顺序就是先把a转换成int,然后10+整数,然后将加之后的结果转换成A类类型赋值给b

如果我们同时写了单参数的构造函数和operator 类型(),就有可能出现冲突,就比如上面的例子:


struct A {public:A(int a_) { a = a_;}int a;A operator+(const A& obj) {return A(a+obj.a);}operator int() {return a;}
}void Test() {A a(10);A b=a+10;A c=10+a;
}

这里的A b=a+10;就会有问题,因为冲突了

  • 因为既可以把10调用构造函数转换成类然后执行operator +
  • 也可以把a转换成int,然后加了后再调用构造函数转换成类

这时候我们就可以使用explicit显式的禁止允许单参数的构造函数的默认转换,这样那个构造函数就只允许我们显式调用,而不允许转换了。在示例中也就是不允许10转换成类的类型了


让类表现出指针形式(重载*->

我们可以重载*->让类表现出类似于指针的形式,使用示例比如说智能指针和容器的迭代器

struct A {public:int& operator*() const {return *p;}int* operator->() const {return &(this->operator*());// return p;}int* p;
};

在这里,*用来表现解引用,->用来表现指针的成员内容,一般来说->在函数返回值表现形式后还会有一个潜在的->(设计如此)


让类表现出函数形式(重载括号运算符)

这个其实非常常用,比如在ceres库中用于传递代价函数,平时也把这种函数叫做仿函数(模仿函数?)

struct A {public:template <typename T>T operator()(const T& a,const T& b) {return a>b?a:b;}
};
void Test() {A a;std::cout << a(10,20);
}

函数模板/类模板/模板模板参数

对于模板来说,关键字classtypename是一样的,为啥有两个?历史原因,不重要了
函数模板:

template <typename T>
void Foo(T t) {}

类模板:

template <typename T>
struct Foo {Foo(T t) {}
}

模板模板参数:

template <typename T>
struct Foo {T foo;
};template <typename T,template <typename> class my_class>
struct A {my_class<T> class_a;
};template <typename T1,typename T2>
struct Goo {T1 goo1;T1 goo2;
};
template <typename T1,typename T2,template <typename,typename> class my_class>
struct B {my_class<T1,T2> class_b;
};void Test() {A<int,Foo> a;B<int,float,Goo> b;
}

其中:template <typename> class my_class是一个示例,表示输入的是带有一个模板参数的类

  • template表示这是一个模板
  • typename的个数表示对应类的模板个数,需要和传入的模板类对应
  • class 也是个关键词,和typename一样,换成typename也是可以的
  • my_class是这个模板名

在上面的示例中,分别给了一个参数和两个参数的模板类传参示例


模板特化和偏特化

其实特化也就是指定模板的某一个或者任意个参数。
所以特化也就分成全特化和偏特化,全特化就是所有模板参数都指定的特化,偏特化就是只指定部分的特化

有一条规则是函数模板只允许全特化,不允许偏特化,类模板允许偏特化

比如在c++标准库中判断一个参数是否为整数的源码:

// Integer types
template <typename _Tp>
struct __is_integer {enum { __value = 0 };
};
template <>
struct __is_integer<bool> {enum { __value = 1 };
};
template <>
struct __is_integer<char> {enum { __value = 1 };
};
template <>
struct __is_integer<signed char> {enum { __value = 1 };
};
template <>
struct __is_integer<unsigned char> {enum { __value = 1 };
};
...
后面有其它整数类型包括了intlong,都是和前面一样的
...

在这里列出所有整数类型,只要是整数就会进入到特化的版本,这样只需要根据__value的值就可以判断了
上面这种形式就是全特化

再给个函数模板全特化的例子:

template <typename T1,typename T2>
void Foo(T1 a,T2 b) {std::cout << "Test1" << std::endl;
}template<>
void Foo<int,float>(int a,float b) {std::cout << "Test2" << std::endl;
}void Test() {Foo<int,int>(10,20);Foo<int,float>(10,20);Foo(10,20);Foo(10,20.0f);
}

下面演示下类模板偏特化和全特化

template <typename T1,typename T2>
struct A {A(T1 a,T2 b) {std::cout << "A" << std::endl;}
};template <typename T>
struct A<int,T> {A(int a,T b) {std::cout << "A<int,T>" << std::endl;}
};template <typename T>
struct A<float,T> {A(int a,T b) {std::cout << "A<float,T>" << std::endl;}
};template <>
struct A<int,int> {A(int a,int b) {std::cout << "A<int,int>" << std::endl;}
};template <>
struct A<int,float> {A(int a,float b) {std::cout << "A<int,float>" << std::endl;}
};void Test() {A("10","20");A(10,"20");A(10.0f,"20");A(10,20);A(10,20.0f);
}

在上面的例子中就是给了两个偏特化和两个全特化

通过前面的全特化都可以看出来,全特化一般会伴随template <>出现(因为全都指定了,也就没有T类型了)


auto/右值引用/变长模板 等c++11的内容

这些内容后面都会单独讲,而且其中很多内容在c++14/c++17会有变化和增强
比如右值在c++17后定义更加成熟,变长模板增加了一些展开方式
后面单独讲比在这里粗浅的讲要好


继承/虚函数/虚表

这一块最好也是单独开一块内容来讲,关于虚的内容还是很多的。底层实现也很值得研究。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/65175.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

19 Linux之Python定制篇-apt软件管理和远程登录

19 Linux之Python定制篇-apt软件管理和远程登录 文章目录 19 Linux之Python定制篇-apt软件管理和远程登录19.1 apt软件管理19.1.1 apt介绍19.1.2 更新软件下载地址-阿里源19.1.3 使用apt完成安装和卸载vim 19.2 远程登录Ubuntu 学习视频来自于B站【小白入门 通俗易懂】2021韩顺…

Kafka常用参数

文章目录 概要broker端参数producer端参数consumer端参数 概要 kafka broker、consumer、和producer都有很多可配置的参数。本文主要总结日常开发中常用到的参数。其中producer端可以在org.apache.kafka.clients.producer.ProducerConfig 中找到配置项&#xff0c;consumer端可…

Linux x86_64 C语言实现gdb断点机制

文章目录 前言一、trap指令简介二、调用ptrace三、创建breakpoints四、CONT 和 SINGLESTEP五、完整代码演示六、增加参数检测参考资料 前言 本文参考文章&#xff1a;Implementing breakpoints on x86 Linux 一、trap指令简介 将通过在断点地址向目标进程的内存中插入一条新…

多线程专栏------多线程的实现方式(二)

目录 1、继承Thread类1.1 实现步骤1.2 代码演示 2、实现Runnable接口2.1 实现步骤2.2 代码演示 3、实现Callable接口3.1 实现步骤3.2 代码演示 1、继承Thread类 优点&#xff1a; 编码简单 缺点&#xff1a; 无法继承其他类了&#xff0c;不利于扩展 资源共享: 不能资源共享 1.…

面试中的商业思维:如何展示你对业务的理解

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

Vue中el-table表格的拖拽排序

el-table实现拖拽 element-ui 表格没有拖拽排序的功能&#xff0c;只能使用sortable.js插件实现拖拽排序&#xff0c;当然也可以应用到其他的组件里面&#xff0c;用法类似&#xff0c;这里只说表格。 实现步骤&#xff1a; 1、安装sortable.js npm install sortablejs --s…

SourceTree 使用技巧

参考资料 SourceTree使用教程&#xff08;一&#xff09;—克隆、提交、推送SourceTree的软合并、混合合并、强合并区别SourceTree 合并分支上的多个提交&#xff0c;一次性合并分支的多次提交至另一分支&#xff0c;主分支前进时的合并冲突解决 目录 一. 基础设置1.1 用户信息…

VR司法法治教育平台,沉浸式课堂教学培养刑侦思维和能力

VR司法法治教育平台提供了多种沉浸式体验&#xff0c;通过虚拟现实(Virtual Reality&#xff0c;简称VR)技术让用户深度参与和体验法治知识。以下是一些常见的沉浸式体验&#xff1a; 1.罪案重现 VR司法法治教育平台可以通过重现真实案例的方式&#xff0c;让用户亲眼目睹罪案发…

一文了解气象站是什么,作用有哪些?

气象站被广泛应用于气象、农业、交通、能源、水利、环保等领域&#xff0c;是一种用于收集、分析和处理气象数据的设备&#xff0c;能够为人们提供及时、准确的气象数据和决策支持。 气象站一般由传感器、环境监控主机和监控平台组成。传感器能够测量各种气象要素&#xff0c;…

Spring 中存取 Bean 的相关注解

目录 一、五大类注解 1、五大类注解存储Bean对象 1.1Controller(控制器储存) 1.2Service(服务存储) 1.3Repository(仓库存储) 1.4Component(组件存储) 1.5Configuration(配置存储) 2、五大类注解小结 2.1为什么要这么多类注解 2.2 五大类注解之间的关系 二、方法注解 1.方法注…

SQL-子查询

SQL 子查询 是指将一个SELECT查询&#xff08;子查询&#xff09;的结果用括号括起来作为另一个SQL语句的数据来源或者判断条件

【Bug】Ubuntu 有线设置打不开无反应

前言&#xff1a; 突然有线设置就没法启用了&#xff0c;但是能联网&#xff0c;能查看ip 解决&#xff1a; 最后安装了一个新的依赖包&#xff1a; sudo apt install gnome-control-center 然后就可以了 还有一个方法&#xff0c;没试过&#xff0c;但感觉有点道理的&#…

在抖音中使用语聚AI,实现自动回复用户视频评论、私信问答

您可以通过集简云数据流程&#xff0c;将语聚AI助手集成到抖音视频评论、抖音私信&#xff0c;实现自动回复用户视频评论、私信问答&#xff0c;大大提升账号互动与运营效率。 效果如下&#xff1a; 自动化流程&#xff1a; ● 抖音普通号评论对接语聚AI&#xff08;点击可一…

宏观经济和风电预测误差分析(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

pytorch异常——RuntimeError:Given groups=1, weight of size..., expected of...

文章目录 省流异常报错异常截图异常代码原因解释修正代码执行结果 省流 nn.Conv2d 需要的输入张量格式为 (batch_size, channels, height, width)&#xff0c;但您的示例输入张量 x 是 (batch_size, height, width, channels)。因此&#xff0c;需要对输入张量进行转置。 注意…

Java的guava 限流写法

第一步先引入 maven <dependency> <groupId>com.google.guava</groupId> <artifactId>guava</artifactId> <version>32.0.1-jre</version> </dependency> 然后上方法 private final double rateLimiter10 1.0 / 10.0; // 每…

多线程学习之多线程的案例

练习一&#xff1a;赠送礼物 需求&#xff1a;有100份礼品,两人同时发送&#xff0c;当剩下的礼品小于10份的时候则不再送出。利用多线程模拟该过程并将线程的名字和礼物的剩余数量打印出来. 示例&#xff1a; public class MyRunable implements Runnable {//第二种方式实现…

LLM学习笔记(1)

学习链接 ChatGPT Prompt Engineering for Developers - DeepLearning.AI 一、prompt engineering for developer 1、原则 prompting principles and iterative pattern 2、用于summarize 环境与helper functions import openai import osfrom dotenv import load_dotenv…

[C++] STL_list常用接口的模拟实现

文章目录 1、list的介绍与使用1.1 list的介绍1.2 list的使用 2、list迭代器3、list的构造4、list常用接口的实现4.1 list capacity4.2 插入删除、交换、清理4.2.1 insert任意位置插入4.2.2 push_front头插4.2.3 push_back尾插4.2.4 erase任意位置删除4.2.5 pop_front头删4.2.6 …

Redis之管道解读

目录 基本介绍 使用例子 管道对比 管道与原生批量命令对比 管道与事务对比 使用pipeline注意事项 基准测试 基本介绍 Redis是一种基于客户端-服务端模型以及请求/响应协议的TCP服务器。 这意味着请求通常按如下步骤处理&#xff1a; 客户端发送一个请求到服务器&am…