嵌入式中《C++之旅》阅读笔记

constexpr

constexpr的隐含意思是在编译阶段求值,对于一些求值操作,如果声明为constexpr,那么会编译器会尝试在编译阶段进行计算求值,如果求值成功,则用结果进行替换。

一个常用的例子是如下:

constexpr int factorial(int n) {return n <= 1 ? 1 : (n * factorial(n - 1));
}int main() {auto num = factorial(10); // 编译阶段求得值,可从汇编查看return 0;
}

如果一个变量声明为constexpr,那么意味着在编译阶段就要获得其值,比如如下这个例子:

#include <iostream>constexpr int factorial(int n) {return n <= 1 ? 1 : (n * factorial(n - 1));
}int main() {int i = 10;constexpr auto num = factorial(i);return 0;
}

因为num被声明为constexpr,所以正如前面所说,在编译阶段求值,又因为i是一个非常量表达式,所以编译器报错如下:

<source>: In function 'int main()':
<source>:10:35: error: the value of 'i' is not usable in a constant expression10 |    constexpr auto num = factorial(i);|                                   ^
<source>:9:8: note: 'int i' is not const9 |    int i = 10;|        ^
Compiler returned: 1

if语句初始化

这个是自C++17才支持的,可以在if语句中进行初始化,随后进行条件判断,如下:

if (int a = 0; a != 10);

也可以像下面这样:

std::vector<int> v;
// operation
if (auto size = v.size()); 

引用赋值不会改变其初始指向

示例:

int x = 2, y = 3;
int &r1 = x;
int &r2 = y;
r1 = r2;
std::cout << x  << " " << y << " " <<  r1 << " " << r2;

在上述代码中,虽然最后r1被赋值为r2,但是其仍然指向x,这样的结果就是x的值也被修改了。最后输出结果为3 3 3 3

enum vs enum类

传统的enum,如果像下面这样定义:

enum Colors {Red,Green,Blue};enum OtherColors {Yellow,Blue};

编译器会报如下错误:

<source>:16:5: error: 'Blue' conflicts with a previous declaration16 |     Blue|     ^~~~
<source>:11:5: note: previous declaration 'Colors Blue'11 |     Blue|     ^~~~

为了解决这个问题,引入了enum class,如下:

enum class Colors {Red,Green,Blue};enum class OtherColors {Yellow,Blue};

modules

示例如下:

// Vector.h:

class Vector {public:Vector(int s);double& operator[](int i);int size();private:double∗ elem; // elem points to an array of sz doublesint sz;
};

// Vector.cpp:

#include "Vector.h" // get Vector’s interface
Vector::Vector(int s) :elem{new double[s]}, sz{s} {
}
double& Vector::operator[](int i) {return elem[i];
}
int Vector::size() {return sz;
}

// user.cpp

#include "Vector.h" // get Vector’s interface
#include <cmath> // get the standard-librar y math function interface including sqrt()
double sqrt_sum(Vector& v) {double sum = 0;for (int i=0; i!=v.siz e(); ++i)sum+=std::sqr t(v[i]); // sum of square rootsreturn sum;
}

最终格式如下:

图片

可以单独对user.cpp 和 Vector.cpp编译,生成.o文件,这是因为上述示例使用了#include操作,预处理器在遇到#include的时候,会将其中的内容完整的拷贝一份到相应的文件,这就导致每个.cpp都有头文件Vector.h的一个副本,代码体积膨胀不说,还增加了编译时间。

为了解决上述问题,引入了modules,使用module优化上述代码,如下:

// Vector.cpp

module;
export module Vector; // defining the module called "Vector"
export class Vector {public:Vector(int s);double& operator[](int i);int size();private:double∗ elem; // elem points to an array of sz doublesint sz;
};Vector::Vector(int s) :elem{new double[s]}, sz{s} {
}
double& Vector::operator[](int i) {
return elem[i];
}
int Vector::siz e() {
return sz;
}
expor t int size(const Vector& v) { return v.siz e(); }

// user.cpp

import Vector; // get Vector’s interface
#include <cmath> // get the standard-librar y math function interface including sqrt()
double sqrt_sum(Vector& v) {double sum = 0;for (int i=0; i!=v.siz e(); ++i)sum+=std::sqr t(v[i]); // sum of square rootsreturn sum;
}

对于这块的内容,可以详细阅读之前的文章:

未来已来:C++ modules初探

纯虚函数

如果其中一个成员函数使用= 0,那么该函数为纯虚函数,继承于存在纯虚函数的子类,其必须实现该函数:

class Base {public:void fun() = 0;
};class Derived : public Base {public:void fun() {}
}

如果声明一个存在纯虚函数的类,那边编译器会报错如下:

<source>:21:8: error: initializer specified for non-virtual method 'void Base::fun()'21 |   void fun() = 0;|        ^~~

派生

判断一个类是否继承于另外一个类,可以使用如下方式:

template<typename Base, typename T>
inline bool IsDerivedOf(T *ptr) {return dynamic_cast<Base*>(ptr) != nullptr;
}

智能指针

智能指针可以避免内存泄漏,此处文章较多,可以参考:

智能指针-使用、避坑和实现

一次诡异的内存泄漏

显式构造

对于如下这种:

class Vector {public:Vector(int sz);
}

可以使用Vector v = 3;这种方式进行初始化,往往这种并不是我们希望看到的,所以可以使用关键字explicit来强制显式初始化:

class Vector {public:explicit Vector(int sz);
}

move语义

自C++11起引入了move语义,不过这个很容易引起初学者的误解,其实move()本身并不做任何操作,其只是进行了简单的类型转换,而真正的移动操作需要类实现者进行定义。

STL对其定义如下:

template<typename _Tp>constexpr typename std::remove_reference<_Tp>::type&&move(_Tp&& __t) noexcept{ return static_cast<typename std::remove_reference<_Tp>::type&&>(__t); }

如果需要深入了解其特性,可以参考文章:

【Modern C++】深入理解移动语义

CTAD

CTAD为Class Template Argument Deduction的缩写,中文称为类模板参数推导,在C++17之前,我们需要像下面这样写:

std::pair<int, double> p1 = {1, 3.41};

现在我们可以如下这样写:

std::pair              p2 = {1, 3.41};

可以参考文章【ModernCpp】新特性之CTAD

字面量

如果我们像下面这样写:

auto s = "hello";

s会被编译器推导为const char*,为了使得其为std::string类型,有以下几种方式:

auto s1 = std::string("hello");
std::string s2 = "hello";
auto s3 = "hello"s;

前两种方式比较常见,第三种方式是Modern cpp的新特性,俗称字面量。在这个语法中,"hello"是字符串字面值,而"s"是用户定义字面量的后缀,它将字符串字面值转换为std::string对象。

COW VS SSO

COW,想必大家都清楚其原理,这个机制很常用,比较常见的如fork等操作,在STL中也有用到这个,比如gcc5.1之前的string中,先看如下代码:

std::string s("str");std::string s1 = s;char *p = const_cast<char*>(s1.data());p[2] = '\0';std::cout << s << std::endl;std::cout << s1 << std::endl;

输出结果无非以下两种:

st
st

或者

str
st

第一种基于gcc5.1前的版本编译,第二种输出基于5.1之后的版本编译,这两个输出的不同正是源于gcc5.1之前的版本对于string的复制采用了COW操作。

自gcc5.1之后,字符串优化采用了新的机制,即SSO,其为Small String Optimization的简写,中文译为小字符串优化,基本原理是:当分配大小小于16个字节时候,从栈上进行分配,而如果大于等于16个字节,则在堆上进行内存分配

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

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

相关文章

深入探索C语言中的sizeof关键字

在C语言中&#xff0c;sizeof是一个运算符&#xff0c;用于获取数据类型或对象在内存中的大小&#xff08;以字节为单位&#xff09;。这是一个非常有用的工具&#xff0c;它可以帮助我们理解不同数据类型如何占用内存&#xff0c;以及如何在编程时更有效地管理内存。 基本数据…

深入解析 Spring 事务机制

当构建复杂的企业级应用程序时&#xff0c;数据一致性和可靠性是至关重要的。Spring 框架提供了强大而灵活的事务管理机制&#xff0c;成为开发者处理事务的首选工具。本文将深入探讨 Spring 事务的使用和原理&#xff0c;为大家提供全面的了解和实际应用的指导。 本文概览 首…

如何用 npm 运行本地 js 文件

安装好 npm 后&#xff0c;cd 到待运行 js 文件的文件夹&#xff0c;运行 npm init&#xff0c;一步步敲回车&#xff0c;可修改可不修改&#xff0c;会生成 package.json 文件 scripts 中一定得是 start&#xff0c;如果是别的名字 js 不运行 确保你已经安装了Node.js和npm。…

题目 1244: 破解简单密码

题目描述 密码是我们生活中非常重要的东东&#xff0c;我们的那么一点不能说的秘密就全靠它了。哇哈哈. 接下来渊子要在密码之上再加一套密码&#xff0c;虽然简单但也安全。 假设老王原来一个BBS上的密码为zvbo941987,为了方便记忆&#xff0c;他通过一种算法把这个密码变换…

ORM模型类

模型 创建两个表 创建模型类 from django.db import models# Create your models here. class BookInfo(models.Model):name models.CharField(max_length10, uniqueTrue) # 书名pub_date models.DateField(nullTrue) # 发布时间read_count models.IntegerField(default…

8.Swift条件语句

Swift 条件语句 在 Swift 中&#xff0c;条件语句用于根据特定条件执行不同的代码块。Swift 提供了 if、guard、switch 等条件语句来实现不同的条件逻辑。以下是 Swift 中常用的条件语句&#xff1a; 一、if 语句 if 语句用于根据条件执行代码块。语法如下&#xff1a; if …

【JSON2WEB】04 amis低代码前端框架介绍

1 什么是 amis amis 是一个低代码前端框架&#xff0c;它使用 JSON 配置来生成页面&#xff0c;可以减少页面开发工作量&#xff0c;极大提升效率。 看到amis一句话的介绍&#xff0c;感觉就是JSON2WEB要找的前端框架。 amis是百度开源的框架&#xff0c;毕竟是大厂&#xff0c…

新概念英语第二册(55)

【New words and expressions】生词和短语&#xff08;19&#xff09; gold n. 金子 mine n. 矿 treasure n. 财宝 revealer n. 探测器 invent …

即插即用、简单有效的大语言模型推荐算法!港大联合百度推出RLMRec

论文链接&#xff1a; https://arxiv.org/abs/2310.15950 论文代码&#xff1a; https://github.com/HKUDS/RLMRec 实验室主页&#xff1a; https://sites.google.com/view/chaoh/group-join-us?authuser0 TLDR 本文从互信息最大化的理论角度出发&#xff0c;通过引入文本信号…

c# 正则表达式 帮助类

public class RegexHelper { /// <summary> /// 验证输入字符串是否与模式字符串匹配&#xff0c;匹配返回true /// </summary> /// <param name"input">输入字符串</param> /// <param nam…

使用No-SQL数据库支持连接查询用例的讨论

简介 在本文中&#xff0c;我们将简单介绍什么是No-SQL数据库。然后我们会讨论一种使用关系数据库比较容易实现的查询&#xff0c;即连接查询&#xff0c;怎么样使用No-SQL来实现。 什么是No-SQL数据库 与No-SQL数据库相对应的是传统的关系数据库&#xff08;RDBMS&#xff…

JRT监听程序

本次设计避免以往设计缺陷&#xff0c;老的主要为了保持兼容性&#xff0c;在用的设计就不好调了。 首先&#xff0c;接口抽象时候就不在给参数放仪器ID和处理类了&#xff0c;直接放仪器配置实体&#xff0c;接口实现想用什么属性就用什么属性&#xff0c;避免老方式要扩参数时…

java的excel列行合并模版

1.效果 2.模版 <tableborder"1"cellpadding"0"cellspacing"0"class"tablebor"id"TABLE"><tr align"center" class"bg217"><td style"background-color: #008000; color: #ffffff;p…

archlinux 使用 electron-ssr 代理 socks5

提前下载好 pacman 包 https://github.com/shadowsocksrr/electron-ssr/releases/download/v0.2.7/electron-ssr-0.2.7.pacman 首先要有 yay 和 aur 源&#xff0c;这个可以参考我之前的博客 虚拟机内使用 archinstall 安装 arch linux 2024.01.01 安装依赖 yay 安装的&#…

[力扣 Hot100]Day26 环形链表 II

题目描述 给定一个链表的头节点 head &#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内…

面试题:如何使用 Redis,打造一款 Feed 流系统?

文章目录 背景相关概念什么是 feed 流feed 流分类设计Feed 流初始化推送发布/删除 Feed 流程推拉结合模式推模式两种模式总结:实现初始化 Feed 流关注的用户发布/删除新的 feed用户新增关注/取消关注最后

WebGL+Three.js入门与实战——绘制水平移动的点、通过鼠标控制绘制(点击绘制、移动绘制、模拟画笔)

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;正逐渐往全干发展 &#x1f4c3;个人状态&#xff1a; 研发工程师&#xff0c;现效力于中国工业软件事业 &#x1f680;人生格言&#xff1a; 积跬步…

JVM 性能调优 - 常用的垃圾回收器(6)

垃圾收集器 在 JVM(Java虚拟机)中,垃圾收集器(Garbage Collector)是负责自动管理内存的组件。它的主要任务是在程序运行过程中,自动回收不再使用的对象所占用的内存空间,以便为新的对象提供足够的内存。 JVM中的垃圾收集器使用不同的算法和策略来实现垃圾收集过程,以…

VisaulStudio2022下用VB.net实现socket与西门子PLC进行通讯案例(优化版)

前言 对于电气工程师来说,不仅要会PLC,还要会上位机。 此前,我写过一个VB.net下雨西门子PLC通讯案例的博文: VisaulStudio2019下用VB.net实现socket与西门子PLC进行通讯案例 但当时很多东西都理解不深,博文也写的比较浅,但我看有不少收藏,也有些朋友在底下询问,所以,…

第4章——深度学习入门(鱼书)

第4章 神经网络的学习 本章的主题是神经网络的学习。这里所说的“学习”是指从训练数据中自动获取最优权重参数的过程。本章中&#xff0c;为了使神经网络能进行学习&#xff0c;将导入损失函数这一指标。而学习的目的就是以该损失函数为基准&#xff0c;找出能使它的值达到最…