【探索C++】C++对C语言的扩展

2023-09-20 15:03:01


一、 引用

1. 变量名         

        变量名本质上就是一段连续内存空间的名字,相当于一个标记,可以用变量名来申请并命名内存空间,可以借助变量名来使用这个空间。

int a = 10; // 声明一个整数变量a,分配内存空间

2. 引用的概念

        变量名本身为一块内存的引用,C++中的引用其实就是给一个已有的变量再取一个别名(alias)。

int x = 5; int &y = x; // 创建y作为x的引用

3. 语法格式

变量名
类型 &引用名 = 变量名

float f = 3.14;
//取别名的方式,给变量f取一个别名叫a
float &a = f;

4. 规则

        (1)引用没有定义,只是一种关系类型声明,所以不分配内存,声明它和原有某一变量的关系,所以类型要保持一致;

        (2)声明时必须初始化,一经声明,不可更改

        (3)可以对引用再次进行引用;

        (4)只有&前面有类型时,是引用,若没有类型,则是其他意思。

5. 引用作为函数参数

        普通引用在声明的时候必须用其他变量进行初始化,但是作为函数形参的时候,不需要进行初始化(因为这个初始化过程在函数调用的时候)实参给形参做初始化。

void func(int &a)
{
    cout << "a = " << a << endl;
}

6. 引用作为函数返回

        引用作为函数返回的时候,可以将这个函数作为赋值运算的左值(结果为赋值运算的值),需要保证函数返回的引用空间在函数退出的时候,没有被释放(全局变量,堆空间数据,静态局部变量,传入的指针数据)。

int &func(int &a)
{
    return a;
}

int main(int argc, char const *argv[])
{
    int a = 200;
    int b = func(a);
    //int b = func(a) = 300;  //可以作为左值
    cout << b << endl;
    
    return 0;
}

7. 引用的意义

        (1)在很多时候引用可以代替指针使用

        (2)引用在函数传参的使用中对于指针具有更好的可读性和实用性

8. 引用的本质         

        引用在C++中的内部实现是一个常指针:type &name <====> type * const name

        C++编译器在编译过程中使用常指针作为引用的内部实现,因此引用所占用的空间大小与指针相同。 从使用的角度,引用会让人误会其只是一个别名,没有自己的存储空间,这是C++为了实用性而做出的细节隐藏。

9. 指针引用

void func1(int *&a)
{
    a[2] = a[0] + a[4];

}



int main(int argc, char const *argv[])
{
    int *p = new int[5]{10,20,30,40,50};
    
    func1(p);
    
    for (int i = 0; i < 5; ++i)
    {
        cout << p[i] << " ";
    }
    cout << endl;
    
    return 0;
}

10. const引用

const引用的特征

  • const引用有较多使用,它可以防止对象的值被随意更改。
  • const对象的引用必须是const的,将普通引用绑定到const对象是不合法的。
const int a = 20;
// int &b = a;    //不合法的
const int &b = a;
  • const引用可以使用相关类型的对象(常量,非同类型的变量或表达式)初始化,这个是const引用与普通引用最大的区别。
int &m = 20;    //不合法的
const int &m = 20;

(1)const引用的原理:        

        const引用的目的是禁止通过修改引用的值来改变被引用的对象。const引用的初始化特性较为微妙,可以看如下代码:

double d = 20.34;
const int &a = d;
double &b = d;

cout << a << " " << b << endl;
d = 23.34;
cout << a << " " << b << endl;

        输出结果为:

20 20.34
20 23.34

        实际上,const引用使用相关类型对象初始化时发生如下过程:

double d = 20.34;
const int &a = d;
//int tmp = d;
//const int &a = tmp;

二、 默认参数和占位参数

1. 默认参数

        通常情况下,函数在调用的时候,形参从实参那里取得值,对于多次调用同一函数同一实参时,C++给出了更简单的处理方法,给形参给默认值,这样就不用从实参哪里取值了。

默认参数规则:

        (1)只有参数列表的后面部分才可以提供默认参数(从右往左依次带默认参数)。

        (2)一旦在一个函数调用中开始使用默认参数值,那么这个参数后的所有参数都必须使用默认参数值。

//int add(int a = 10, int b, int c = 100)    //语法错误,不能跳着带默认参
//int add(int a = 10, int b, int c)    //语法错误,不能从左开始带默认参
int add(int a, int b, int c = 100)    //形参带默认参数
{
    return a+b+c;    
}


int main(int argc, char const *argv[])
{
    cout << add(10, 20, 30) << endl;
    
    return 0;
}

2. 占位参数

        占位参数只有类型声明,没有参数名声明。占位参数在函数里面不能使用,为了以后函数做拓展留下线索。

/占位参数在函数里面能不能使用?不能使用
//占位参数为了以后函数做拓展留下线索
int add(int a, int b, int c, int = 0)    //形参带占位参数,占位参数还带有默认参数
{
    
    return a+b+c;
}

//int add(int a, int b, int = 0, int c = 0)
//占位参数,占位参数可以跟默认参数一起结合使用,前提是默认参要从右到左依次赋值

//int add(int a, int b, int, int c)
//占位参数可以放在任意位置



int main(int argc, char const *argv[])
{
    cout << add(10, 20, 30, 40) << endl;
    
    return 0;
}

三、C++函数重载

1. 函数重载

        函数重载是C++中一种允许你使用相同的函数名定义多个不同参数列表的函数的机制。函数重载的规则如下:

  • 函数名相同;
  • 参数列表不同,可以通过参数个数、参数类型和参数顺序的不同来区分;
  • 返回类型不同的函数不能构成重载。

以下是一些函数重载的示例(这些函数都具有相同的名称func,但参数列表不同,因此构成了函数重载):

        void func(int a) {}

        void func(float f) {}

        void func(int a, float f) {}

        void func(float a, float f) {}

        void func(float a, int f) {}

        void func(double a, char f) {}

        void func(float a, char f, int m) {}

        int func(char a, char f) {}

2. 调用准则

        当调用重载函数时,编译器会根据传递的参数选择最匹配的函数。它会按照以下准则来寻找可行的候选函数:

        (1)严格匹配,找到则调用;

        (2)通过隐式类型转换技术寻找匹配的函数,找到则调用;

        (3)一个函数不能即做重载,又作默认参数的函数,当你少写一个参数时,系统无法确认是重载函数默认参数。

3. 函数重载与函数指针结合

        函数指针可以指向不同参数列表的函数,但在赋值时要注意确保参数匹配。示例:

typedef void (*FUNC1)(int, int, int);
typedef void (*FUNC2)(int, int);

int main() {
    FUNC1 p1 = func; // 调用func(int, int, int)
    FUNC2 p2 = func; // 调用func(int, int)
    
    p1(10, 20, 30);
    p2(10, 20);
    
    return 0;
}

4. 函数重载总结

        函数重载是C++中的一种强大机制,它允许定义多个具有相同名称但参数列表不同的函数。以下是一些函数重载的总结:

  • 函数名相同,参数列表不同即可构成重载;
  • 函数返回值不能作为函数重载的依据;
  • 编译器根据传递的参数选择最匹配的函数。

四、运算符重载

        运算符重载是C++中另一个强大的特性,它允许你自定义数据类型的运算符行为。通过运算符重载,你可以使自定义类型像内置类型一样参与各种运算。

1. 运算符重载的基本原理

        运算符重载的基本原理是为自定义类型定义与运算符相关的函数。例如,如果你想要为自定义的Complex类实现+运算符,可以这样做:

class Complex {
public:
    Complex operator+(const Complex& other) {
        Complex result;
        result.real = this->real + other.real;
        result.imag = this->imag + other.imag;
        return result;
    }
private:
    double real;
    double imag;
};

2. 运算符重载的示例

class Complex {
public:
    Complex operator+(const Complex& other) {
        Complex result;
        result.real = this->real + other.real;
        result.imag = this->imag + other.imag;
        return result;
    }
    
    bool operator==(const Complex& other) {
        return (this->real == other.real) && (this->imag == other.imag);
    }
    
    Complex operator*(const Complex& other) {
        Complex result;
        result.real = this->real * other.real - this->imag * other.imag;
        result.imag = this->real * other.imag + this->imag * other.real;
        return result;
    }
    
    // 其他运算符重载也可以类似定义
private:
    double real;
    double imag;
};

3. 运算符重载的使用

        运算符重载使自定义类型的对象可以像内置类型一样使用运算符。例如:

Complex c1(1.0, 2.0);
Complex c2(3.0, 4.0);
Complex c3 = c1 + c2; // 使用+运算符重载
bool isEqual = (c1 == c2); // 使用==运算符重载
Complex c4 = c1 * c2; // 使用*运算符重载

五、命名空间

1.概念

        所谓namespace,是指标识符的各种可见范围,C++标准程序库当中所有的标识符都被定义在一个名为std的namespace中。

(1)<iostream>和<iostream.h>

        格式不一样,前者没有后缀,实际上,在你的编译器include文件夹里面可以看到,二者是两个文件,打开文件就会发现,里面的代码是不一样的,后缀为.h的头文件C++标准已经明确提出不支持了,早些的实现将标准库功能定义在全局空间里,声明在带.h的后缀文件里,C++标准为了和C区别开,也为了正确使用命名空间,规定头文件不使用后缀.h,因此:

        当<iostream.h>使用时,相当于在C中调用库函数,使用的是全局命名空间;

        当<iostream>使用时,该文件没有定义全局命名空间,必须使用using namespace std;这样才能正确使用cin,cout。

(2)由于namespace的概念,使用C++标准程序库中的任何标识符是,可以有三种选择:

用法1:在用的时候加上std:: 比较麻烦

        int a; std::cin >> a; // std::cout << a << std::endl; //

用法2:用的时候打开一部分,建议用的,安全性,命名空间的污染度不高

        using std::cin; using std::cout; using std::endl;

用法3:直接打开全部的,不安全,容易造成命名空间污染,出现重复定义,多次引用,解释如下:

        using namespace std;

        因为标准库非常的庞大,所以程序员在选择的类的名称或函数名 时就很有可能和标准库中的某个名字相同。所以为了避免这种情况所造成的名字冲突,就把标准库中的一切都被放在名字空间std中。但这又会带来了一个新问 题。无数原有的C++代码都依赖于使用了多年的伪标准库中的功能,他们都是在全局空间下的。所以就有了<iostream.h> 和<iostream>等等这样的头文件,一个是为了兼容以前的C++代码,一个是为了支持新的标准。命名空间std封装的是标准程序库的名称,标准程序库为了和以前的头文件区别,一般不加".h"

2. C++命名空间定义以及使用方法


        在C++中,名称可以是符号常量,变量,宏,函数,枚举,类和对象等等,为了避免在大规模程序的设计中,以及在程序员在使用各种的C++库时,这些标识符的命名发生冲突。
        标准C++引入了关键字namespace(命名空间/名字空间/名称空间/名域),可以更好地控制标识符的作用域。
        std是C++标准命名空间,C++标准程序库中所有标识符都被定义在std中,比如标准库中的类iostream、vector、map等都定义在该命名空间中,使用时要加上using声明(using namespace std)或者using指示(using std::cin using std::cout using std::endl using std::vector<int>  std::map<int, string>...)

C中的命名空间:
        在C语言中只有一个命名空间,全局作用域
        C语言中所有的全局标识符共享同一个作用域
        标识符之间可能发生冲突
C++中的命名空间:
        命名空间将全局作用域分成不同的部分
        不同命名空间中的标识符可以同名而且不会发生冲突
        命名空间可以相互嵌套
        全局作用域也叫做默认命名空间

(1)C++命名空间的定义
namespace name {...}
namespace {...}        //匿名命名空间,里面的内容只能在当前文件使用
(2)C++命名空间的使用
namespace A{int n;}
1、直接全部(打开)放出来       using namespace A;
2、只打开一部分                using A::n;
3、用到谁就使用谁              A::n;

namespace {int n}
使用默认匿名空间里面的变量      ::n
默认情况下,可以直接使用默认命名空间里面的所有标识符

3. C++命名空间代码实现

#include <iostream>

using namespace std;

namespace A
{
    int a;
}

namespace B
{    
    int a;
    int b;
    int c;
    namespace C     //可以嵌套命名空间
    {
        char name[20];
    }
    struct Stu         //可以嵌套结构体
    {
        int num;
    }d;
    void func(){cout << "123456" << endl;}        //可以嵌套函数
}

namespace     //匿名命名空间,简单点来讲就是全局命名空间
{
    int a;
    int b;
}

namespace     //匿名命名空间可以定义两个,但是里面不用出现相同的标识符
{    
    int d;    
}

//1、直接全部打开,全部裸露出来,但是会造成命名空间污染,不建议
// using namespace B;
// using namespace B::C;

//2、只放出一部分
using B::c;
// using B::C::name;

int main(int argc, char const *argv[])
{
    a = 200;    //默认情况下,可以直接使用默认命名空间里面的所有标识符
    cout << "a = " << a << endl;
    
    ::b = 300;    //使用匿名命名空间里面的标识符
    cout << "b = " << ::b <<  endl;
    
    c = 300;
    cout << "c = " << c << endl;
    
    //3、直接用命名空间::成员名
    B::func();
    
    cin >> B::C::name;
    cout << B::C::name << endl;
        
    return 0;
}

4. 总结

        引入命名空间:在使用某个命名空间中的成员之前,通常需要引入该命名空间。可以使用 using namespace name 来引入整个命名空间,也可以使用 using name::member 来引入特定的成员。

        命名空间的使用:在C++中,<iostream>等标准库头文件通常没有定义在全局命名空间中,因此使用其中的成员时,需要使用 using namespace std; 或者在使用时前缀以 std::,例如 std::cout

        头文件命名规范:为了与C语言区分开,C++标准规定头文件不使用后缀 .h,而是直接使用不带后缀的文件名。

        命名空间的定义:命名空间的定义使用 namespace name {...} 的形式,其中 name 是命名空间的名称,而 {...} 内包含了该命名空间的成员。

        命名空间的嵌套:C++允许嵌套命名空间,这意味着你可以在一个命名空间内再定义另一个命名空间,从而更好地组织和管理代码。

        命名空间是C++中用于避免命名冲突和组织代码的有用工具,它有助于将代码模块化,使代码更具可维护性和可读性。熟练掌握命名空间的使用是C++编程的重要一步。

六、内敛函数

        C++提供了内联函数(inline functions)这一重要特性,它可以帮助程序员优化代码执行效率,减少函数调用的开销。本文将详细介绍内联函数的特点、使用场景以及与宏替换的区别,帮助你更好地理解和利用这一功能,从而实现了真正的内嵌

#include <iostream>

using namespace std;

inline void func(int a)    //内敛函数   
{
    cout << "a = " << a << endl;
}

int main(int argc, char const *argv[])
{
    func(100);    //调用函数有 保护现场 和 恢复现场 这两个过程
    // {
    //     cout << "a = " << a << endl;
    // }
    
    return 0;
}

1.内联函数特点

        (1)内联函数使用 inline 关键字声明,通常与函数定义结合在一起;
        (2)内联函数的代码被编译器直接插入到函数调用的地方,类似于宏替换,减少了函数调用的开销;
        (3)内联函数具有普通函数的特征,包括参数检查和返回类型。


2. 内联函数的优点

        (1)提高程序运行效率,减少了函数调用时的入栈和出栈操作;
        (2)内联函数可以作为类的成员函数,可以访问所在类的保护成员和私有成员。


3. 内联函数的限制

        (1)内联函数的函数体应该相对较小,避免过于复杂的代码;
        (2)不适合包含循环或大量条件判断的函数体;
        (3)不能对内联函数进行取址操作。


4.内联函数与宏替换的区别

        (1)内联在编绎时展开,宏在预编译时展开。 展开的时间不同。

        (2)编译内联函数可以嵌入到目标代码,宏只是简单文本替换。

        (3)内联会做类型,语法检查,而宏不具这样功能。

        (4)宏不是函数,inline函数是函数

        (5)宏定义小心处理宏参数(一般参数要括号起来),否则易出现二义性,而内联定义不会出现。

#include <iostream>

using namespace std;

#define SQ(x) ((x)*(x))

inline int sq(int x)
{
    return x*x;
}


int main(int argc, char const *argv[])
{
    int i = 0;
    for (int i = 0; i < 5; ++i)
    {
        // cout << sq(i+1) << endl;//1 4 9 16 25
        cout << SQ(i+1) << endl;// 1 4 9 16 25
    }
    
    return 0;
}


5.适用场景

        (1)内联函数适用于函数体相对简单且被频繁调用的情况;
        (2)当函数体较大或包含复杂的逻辑时,内联可能不会带来明显的性能提升;
        内联函数是C++中一种用于提高程序性能的重要技术,但需要注意适用场景和代码复杂度,以充分发挥其优势。熟练使用内联函数可以有效地减少函数调用带来的开销,提高程序的运行效率。

七、C++头文件和extern "C"

(1)标准C++的头文件引用

#include //C++的标准输入输出头文件

(2)如果要在C++程序里面用标准C的头文件

//#include <stdio.h>
//#include <stdlib.h>
#include  <cstdio>
#include  <cstdlib>

        注意:不是所有的.h的头文件都是以c开头,去.h尾。

(3)使用C语言的相关函数的时候,头文件的引用方式

extern "C"
{
    #include <sys/types.h>    
    #include <sys/stat.h>      
    #include <fcntl.h>
    #include <stdio.h>
    #include <stdlib.h>
}

        写一个练习吧:设计一个计算圆柱体体积的函数,表面积函数,P值和圆柱体的高都有默认参数。

#include <iostream>

using namespace std;

void func1(int r, int h = 10, float PI = 3.14)	//计算圆柱体体积
{
	cout << "圆柱体的体积:" << r*r*PI*h << endl;
}

void func2(int r, int h = 10, float PI = 3.14)	//计算圆柱体表面积
{
	cout << "圆柱体的表面积:" << (2*r*r*PI)+(2*PI*r*h) << endl;
}

int main(int argc, char const *argv[])
{
	cout << "输入圆柱体的半径跟高" << endl;
	int r, h;
	cin >> r >> h;
	if (h > 0)
	{
		func1(r, h);
		func2(r, h);
	}
	else
	{
		func1(r);
		func2(r);
	}

	return 0;
}

        更多C/C++语言Linux系统数据结构ARM板实战相关文章,关注专栏:

   手撕C语言

            玩转linux

                    脚踢数据结构

                            系统、网络编程

                                     探索C++

                                             6818(ARM)开发板实战

📢写在最后

  • 今天的分享就到这啦~
  • 觉得博主写的还不错的烦劳 一键三连喔~
  • 🎉🎉🎉感谢关注🎉🎉🎉
更多推荐

OpenSergo & Spring Cloud Alibaba 带来的服务治理能力

博主介绍:✌全网粉丝3W+,全栈开发工程师,从事多年软件开发,在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战,博主也曾写过优秀论文,查重率极低,在这方面有丰富的经验✌博主作品:《Java项目案例》主要基于SpringBoot+MyBatis/MyBatis-plus+MySQL+Vue等前后端

云原生微服务 第四章 Spring Cloud Netflix 之 Eureka

系列文章目录第一章Java线程池技术应用第二章CountDownLatch和Semaphone的应用第三章SpringCloud简介第四章SpringCloudNetflix之Eureka文章目录系列文章目录@[TOC](文章目录)前言1、Eureka两大组件2、Eureka服务注册与发现3、案例3.1、创建主工程3.

jq命令安装与使用

目录一、简介二、下载及安装1.Linux安装2.Windows安装3.测试安装结果三、jq用法1.基本语法2.常见用法1)格式化JSON2)获取属性3)属性不存在情况处理4)数组遍历、截取、展开5)管道、逗号、加号6)数据构造7)基础函数8)过滤、排序、分组函数9)字符串操作函数10)日期函数11)高级用法官网地址:h

网络安全(黑客技术)学习笔记

1.网络安全是什么网络安全可以基于攻击和防御视角来分类,我们经常听到的“红队”、“渗透测试”等就是研究攻击技术,而“蓝队”、“安全运营”、“安全运维”则研究防御技术。2.网络安全市场一、是市场需求量高;二、则是发展相对成熟入门比较容易。3.所需要的技术水平需要掌握的知识点偏多(举例):外围打点能力渗透漏洞挖掘流量分析代

LLM系列 | 20 : Llama2 实战(下篇)-中文语料微调(附完整代码)

简介紧接前文:万字长文细说ChatGPT的前世今生Llama2实战(上篇):本地部署(附代码)上篇主要介绍Llama2的基本情况和基于官方模型实测Llama2在中英上的效果,包括单轮和多轮对话。今天这篇小作文作为Llama2的下篇,主要介绍如何用中文语料对Llama2的基座模型(7B版)进行微调并实测微调后模型的效果。

八大排序(一)冒泡排序,选择排序,插入排序,希尔排序

一、冒泡排序冒泡排序的原理是:从左到右,相邻元素进行比较。每次比较一轮,就会找到序列中最大的一个或最小的一个。这个数就会从序列的最右边冒出来。以从小到大排序为例,第一轮比较后,所有数中最大的那个数就会浮到最右边;第二轮比较后,所有数中第二大的那个数就会浮到倒数第二个位置……就这样一轮一轮地比较,最后实现从小到大排序。代

面试题五:computed的使用

题记大部分的工作中使用computed的频次很低的,所以今天拿出来一文对于computed进行详细的介绍,因为Vue的灵魂之一就是computed。模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护,所以,对于复杂逻辑,vue提倡使用计算属性。需要特别说明:计算属性

python基础语法回顾

1/1pythoneval函数1/1eval()函数是Python中的一个内置函数,它用于执行一个字符串形式的Python表达式,并返回表达式的结果。这个函数接受一个字符串参数,然后将其解析并执行,将结果返回。通常情况下,eval()用于动态地执行Python代码。以下是eval()函数的基本语法:pythonresu

共享股东:让你的连锁店铺更有竞争力

你是否有过这样的困扰:你经营了一家或多家连锁店铺,但是客流量不稳定,收入不理想,资金周转困难,竞争对手层出不穷,你怎么才能让你的店铺更有吸引力,更有利润,更有发展前景呢?如果你有这样的困扰,那么你可能需要了解一下共享股东这个新兴的商业模式。共享股东是一种利用互联网平台,将线下实体店铺的资源和利润与消费者和投资者进行共享

Mysql

视频链接黑马Mysql基础篇通用语法及分类DDL:数据定义语言,用来定义数据库对象(数据库、表、字段)DML:数据操作语言,用来对数据库表中的数据进行增删改DQL:数据查询语言,用来查询数据库中表的记录DCL:数据控制语言,用来创建数据库用户、控制数据库的控制权限DDL(数据定义语言)数据定义语言数据库操作查询所有数据

企业工程项目管理系统源码(三控:进度组织、质量安全、预算资金成本、二平台:招采、设计管理)

工程项目管理软件(工程项目管理系统)对建设工程项目管理组织建设、项目策划决策、规划设计、施工建设到竣工交付、总结评估、运维运营,全过程、全方位的对项目进行综合管理工程项目各模块及其功能点清单一、系统管理1、数据字典:实现对数据字典标签的增删改查操作2、编码管理:实现对系统编码的增删改查操作3、用户管理:管理和查看用户角

热文推荐