初识c++

文章目录

  • 前言
  • 一、C++命名空间
    • 1、命名空间
    • 2、命名空间定义
  • 二、第一个c++程序
    • 1、c++的hello world
    • 2、std命名空间的使用惯例
  • 三、C++输入&输出
    • 1、c++输入&输出
  • 四、c++中缺省参数
    • 1、缺省参数概念
    • 2、缺省参数分类
    • 3、缺省参数应用
  • 五、c++中函数重载
    • 1、函数重载概念
    • 2、函数重载应用
  • 六、c++中的引用
    • 1、 引用概念
    • 2、引用特性
    • 3、引用的应用 -- 做参数
    • 4、引用的应用 -- 做返回值


前言

C语言是结构化和模块化的语言,适合处理较小规模的程序。对于复杂的问题,规模较大的程序,需要高度的抽象和建模时,C语言则不合适。为了解决软件危机, 20世纪80年代, 计算机界提出了OOP(object oriented programming:面向对象)思想,支持面向对象的程序设计语言应运而生。
1982年,Bjarne Stroustrup博士在C语言的基础上引入并扩充了面向对象的概念,发明了一种新的程序语言。为了表达该语言与C语言的渊源关系,命名为C++。因此:C++是基于C语言而产生的,它既可以进行C语言的过程化程序设计,又可以进行以抽象数据类型为特点的基于对象的程序设计,还可以进行面向对象的程序设计。

一、C++命名空间

1、命名空间

在C/C++中,变量、函数和后面要学到的类都是大量存在的,这些变量、函数和类的名称将都存在于全局作用域中,可能会导致很多冲突。使用命名空间的目的是对标识符的名称进行本地化,以避免命名冲突或名字污染,namespace关键字的出现就是针对这种问题的。
例如在如下代码中会报出rand重定义的错误,因为rand在stdlib库中为函数名。

#include<stdio.h>
#include<stdlib.h>int rand = 10;
int main()
{//因为stdlib库中定义了rand()函数,所以全局定义的rand会和库中的rand产生命名冲突//在c语言中没办法解决类似这样的命名冲突的问题,所以c++提出了namespace来解决printf("%d\n", rand);return 0;
}

c++的namespace就可以解决c语言中的命名冲突问题。

#include<stdio.h>
#include<stdlib.h>//命名空间域
namespace dong
{int rand = 0;
}int main()
{printf("%d\n", rand);  //此时rand为stdlib内的函数//  ::为域作用限定符printf("%d\n", dong::rand);  //此时dong::rand为dong命名空间域里面的rand变量return 0;
}

2、命名空间定义

定义命名空间,需要使用到namespace关键字,后面跟命名空间的名字,然后接一对{}即可,{}中即为命名空间的成员。

#include<stdio.h>
#include<stdlib.h>//命名空间域
namespace dong
{//命名空间中定义变量int rand = 0;//命名空间还可以嵌套定义//嵌套定义命名空间namespace inside{int tmp = 5;struct Stu{char name[20];int age;};}//命名空间中定义函数void func(){printf("func()\n");}//命名空间中定义结构体struct TreeNode{struct TreeNode* left;struct TreeNode* right;int val;};
}int main()
{printf("%d\n", rand);  //此时rand为stdlib内的函数//  ::为域作用限定符printf("%d\n", dong::rand);  //此时dong::rand为dong命名空间域里面的rand变量//此时会去全局域找func()函数,如果全局域没有定义func()就会报错//func();  //dong::指定了去命名空间域dong里面找func()的定义。dong::func();  //dong::指定了去命名空间域dong里面找struct TreeNode的定义。struct dong::TreeNode node;//访问嵌套定义的命名空间里面的变量printf("%d\n", dong::inside::tmp);struct dong::inside::Stu stu;return 0;
}

当我们在一个项目中,如果有多个模块都用到了类似栈和队列的操作,并且它们也定义了Stack和QueueNode结构等,此时我们可以直接使用一个namespace dong的命名空间将该模块的.h和.c文件都包含在namespace dong命名空间域中,此时在namespace dong里面的变量和结构体定义就不会和别的模块里面的变量和结构体产生冲突了。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
此时当我们想使用命名空间dong中定义的结构体和方法时,就不能再想下面图片中那样定义了。因为一个命名空间就定义了一个新的作用域,命名空间中的所有内容都局限于该命名空间中
在这里插入图片描述
此时使用dong命名空间的结构体和函数就可以如下图一样的形式使用。这样就可以避免每个模块之间命名冲突,比如这个模块使用这些操作的话就去dong命名空间中去找这些定义,这个模块再使用其他操作的话可以去另一个命名空间中去找这些定义,这两个命名空间中可以定义相同名字的结构体或变量或函数,只要在使用时标明去哪个命名空间去找这些结构体或变量或函数的定义即可。
在这里插入图片描述

二、第一个c++程序

1、c++的hello world

下面为c++语言写的第一个hello world程序。

#include<iostream>
using namespace std;
int main()
{cout << "hello world" << endl;return 0;
}

其中该语句说明引入了c++里面的io流的库。

#include<iostream>

该语句说明将std命名空间里面的成员都引入,c++中标准库的东西都放到了std中,即如果有了该语句,在下面的代码中使用std里面的变量时就不需要写成std::cout,而是可以直接写成cout,省略前面的std::。因为此时std命名空间的成员已经都引入到全局域了,所以编译器可以在全局域里面找到这些成员的定义。

using namespace std;

如果我们将using namespace std;注释掉的话,再使用命名空间std里面的成员时,就需要像下面这样写。

#include<iostream>
//using namespace std;
int main()
{std::cout << "hello world" << std::endl;return 0;
}

我们还可以只引入std库里面部分的变量名称。

#include<iostream>
//using namespace std;
//只引入我们需要的成员即可
using std::cout;
using std::cin;
using std::endl;
int main()
{cout << "hello world" << endl;return 0;
}

2、std命名空间的使用惯例

1.在日常练习中,建议直接using namespace std即可,这样就很方便。
2 using namespace std展开,标准库就全部暴露出来了,如果我们定义跟库重名的类型/对象/函数,就存在冲突问题。该问题在日常练习中很少出现,但是项目开发中代码较多、规模大,就很容易出现。所以建议在项目开发中使用,像std::cout这样使用时指定命名空间 或 using std::cout展开常用的库对象/类型等方式。

#include<iostream>
#include<vector>
//引入常用的库对象/类型
using std::cout;
using std::endl;
int main()
{//指定命名空间访问std::vector<int> v;v.push_back(1);v.push_back(2);v.push_back(3);v.push_back(4);cout << "hello world!" << endl;cout << "hello world!" << endl;cout << "hello world!" << endl;cout << "hello world!" << endl;return 0;
}

三、C++输入&输出

1、c++输入&输出

1.使用cout标准输出对象(控制台)和cin标准输入对象(键盘)时,必须包含< iostream >头文件以及按命名空间使用方法使用std。
2. cout和cin是全局的流对象,endl是特殊的C++符号,表示换行输出,他们都包含在包含< iostream >头文件中。
3. << 是流插入运算符,>> 是流提取运算符。
4. 使用C++输入输出更方便,不需要像printf/scanf输入输出时那样,需要手动控制格式。C++的输入输出可以自动识别变量类型。
在c语言中,当我们需要在标准输出流stdout中打印数据时,使用printf函数,当我们要读取数据时,需要使用scanf从标准输入流stdin中获取数据,并且这两个函数在打印和获取数据时,都要标明数据的类型。
但是c++中不需要像c语言那样标明数据类型。

#include<iostream>
//using namespace std;
int main()
{//  << 是流插入运算符std::cout << "hello world" << std::endl;//std命名空间中的endl就类似于换行符,即 std::endl 等价于 "\n"std::cout << "hello world" << "\n";//c++中自动识别类型int i = 11;double d = 11.11;printf("%d %lf\n", i, d);//c++中可以自动识别类型std::cout << i << "," << d << std::endl;//c语言中读取数据scanf("%d %lf", &i, &d);printf("%d,%lf\n", i, d);//c++中读取数据//  >> 为流提取std::cin >> i >> d;std::cout << i << "," << d << std::endl;//当需要精度控制时,因为c++实现精度控制比较麻烦,所以还可以使用c语言的printf()printf("%.2lf\n", d);return 0;
}

四、c++中缺省参数

1、缺省参数概念

缺省参数是声明或定义函数时为函数的参数指定一个缺省值。在调用该函数时,如果没有指定实参则采用该形参的缺省值,否则使用指定的实参。缺省值必须是常量或者全局变量。

//缺省参数
#include<iostream>
using namespace std;
void Func(int a = 0)
{cout << a << endl;
}
int main()
{//当调用函数Func()时,传递的有实参,则缺省参数的值不起作用Func(1);Func(2);Func(3);//当调用函数Func()时,没有传递实参,此时缺省参数的值就起作用Func();return 0;
}

2、缺省参数分类

全缺省参数:全缺省参数需要注意的就是在函数调用时,不能直接跳过前面的形参,然后传值给后面的形参。

//全缺省参数
#include<iostream>
using namespace std;
void TestFunc(int a = 10, int b = 20, int c = 30)
{cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl << endl;
}
int main()
{TestFunc();//传入的实参是按照从左向右的顺序赋值的,只有一个实参就赋给形参aTestFunc(1);//有两个实参就赋给a和bTestFunc(1,2);//有三个实参就赋给a、b、cTestFunc(1,2,3);//这里需要注意的是,不能直接跳过前面的a,然后直接传值给b//TestFunc(, 1, );  //这样是错误的,语法不允许return 0;
}

半缺省参数:这里需要注意的是半缺省参数必须从右往左依次来给出,不能间隔着给。

//半缺省参数
#include<iostream>
using namespace std;//这里函数TestFunc只定义了两个缺省参数,所以在调用时必须要将形参a的值通过实参传递过来
//半缺省参数必须从右往左依次来给出,不能间隔着给
//   void TestFunc(int a = 10, int b, int c = 30)  //这样定义的就是错误的
void TestFunc(int a, int b = 20, int c = 30)
{cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl << endl;
}
int main()
{//此时必须要将形参a的值通过实参传递过去//TestFunc();  //错误// //传入的实参是按照从左向右的顺序赋值的,只有一个实参就赋给形参aTestFunc(1);//有两个实参就赋给a和bTestFunc(1, 2);//有三个实参就赋给a、b、cTestFunc(1, 2, 3);return 0;
}

3、缺省参数应用

前面我们使用c语言实现了栈的创建和初始化,当我们初始化栈时会为栈的容量开辟4个空间,但是当我们知道需要向栈中插入多少数据时,此时如果还每次只开辟capacity*2的空间,然后一次一次开辟到我们需要的空间时,这样程序的开销就大了,因为每次开辟空间还有可能需要将数据转移。而此时缺省参数就可以解决上面的问题,我们将栈容量定义为缺省参数,当在初始化栈时,如果已经提前知道了需要多少空间,我们就可以给capacity传入值,而当不知道需要多少空间时,就可以不用给capacity传值,此时capacity就为4。

#include<iostream>
#include<stdlib.h>
struct Stack
{int* a;int top;int capacity;
};
//将栈的容量capacity设为缺省参数
void StackInit(struct Stack* ps, int capacity = 4)
{//当调用StackInit()没有给capacity传参时,缺省参数的值就起作用,此时capacity为4//当调用StackInit()给capacity传参时,此时缺省参数的值就不起作用,capacity的值就为传进来的值ps->a = (int*)malloc(sizeof(int) * capacity);//ps->top = 0;ps->capacity = capacity;}int main()
{//当我们在使用栈时,已经提前知道了一定会插入100个数据,所以可以直接将capacity的值当作实参传入//这样可以提前开好空间,插入数据时就避免了扩容,程序开销就会小一些struct Stack st1;StackInit(&st1, 100);//当我们不知道需要插入多少数据时,就可以不传入capacity的值。struct Stack st2;StackInit(&st2);
}

上面的代码中我们是直接将StackInit(struct Stack* ps,int capacity=4)函数写出来了,而在项目中,我们需要在.h文件中先将StackInit(struct Stack* ps,int capacity=4)声明一下,然后在.c文件中完成StackInit(struct Stack* ps,int capacity=4)函数的定义。此时就会产生一个问题,如果StackInit()函数的声明和定义中给缺省参数capacity的值不一样时,那么编译器会以哪个为准呢?所以缺省参数不能在函数声明和定义中同时出现
此时编译器就不知道该以哪一个为准了。
在这里插入图片描述
在这里插入图片描述
所以都是在函数声明中给缺省参数值,然后函数定义时就不需要给缺省参数值了。
在这里插入图片描述
在这里插入图片描述
那如果在函数定义时给了缺省参数,而在函数声明时没有给缺省参数,那么此时缺省参数的值不会起作用。因为已经在函数声明时说明了该函数没有缺省参数,但是调用时按缺省参数的形式调用,就会出现错误。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

五、c++中函数重载

1、函数重载概念

函数重载:是函数的一种特殊情况,C++允许在同一作用域中声明几个功能类似的同名函数,这些同名函数的形参列表(参数个数 或 类型 或 类型顺序)不同,常用来处理实现功能类似数据类型不同的问题。
函数重载 – 参数个数不同

//函数重载 参数个数不同
#include<iostream>
using namespace std;
int Add(int a, int b, int c)
{return a + b + c;
}
int Add(int a, int b)
{return a + b;
}int main()
{cout << Add(2, 2, 2) << endl;cout << Add(2, 2) << endl;return 0;
}

函数重载 – 参数类型不同

//函数重载 参数类型不同
#include<iostream>
using namespace std;
int Add(int a, int b)
{return a + b;
}
double Add(double a, double b)
{return a + b;
}int main()
{cout << Add(2, 2) << endl;cout << Add(1.1, 2.2) << endl;return 0;
}

函数重载 – 参数类型顺序不同

//函数重载 参数类型顺序不同
#include<iostream>
using namespace std;
void func(int i, char ch)
{cout << "void func(int i, char ch)" << endl;
}
void func(char ch, int i)
{cout << "void func(char ch, int i)" << endl;
}
int main()
{func(1, 'a');func('a', 1);return 0;
}

当函数返回值不同时不构成函数重载,只有参数不同才构成重载。因为函数返回值不同,函数调用时编译器无法区分去调用哪一个函数。
在这里插入图片描述

2、函数重载应用

在c语言中,当我们要交换两个int型变量的值时,需要写一个函数;当我们要交换两个double类型的值时,又要写一个函数,而且两个函数的名字还不能相同。

#include<stdio.h>
void Swapi(int* p1, int* p2)
{int tmp = *p1;*p1 = *p2;*p2 = tmp;
}void Swapd(double* p1, double* p2)
{double tmp = *p1;*p1 = *p2;*p2 = tmp;
}int main()
{int a = 1;int b = 2;double c = 1.1;double d = 2.2;Swapi(&a, &b);Swapd(&c, &d);printf("%d %d\n", a, b);printf("%lf %lf\n", c, d);return 0;
}

但是c++中的函数重载就可以解决这个问题,只要两个名称相同的函数的参数数量、类型或顺序不同,就可以构成函数重载。

#include<iostream>
using namespace std;
void Swap(int* p1, int* p2)
{int tmp = *p1;*p1 = *p2;*p2 = tmp;
}void Swap(double* p1, double* p2)
{double tmp = *p1;*p1 = *p2;*p2 = tmp;
}int main()
{int a = 1;int b = 2;double c = 1.1;double d = 2.2;Swap(&a, &b);Swap(&c, &d);cout << a << " " << b << endl;cout << c << " " << d << endl;return 0;
}

通过上述代码我们可以发现c++中的cout和cin可以自动识别类型,其实也是用到了函数重载,这样cout和cin才可以在各种类型时都可以用。
在这里插入图片描述

六、c++中的引用

1、 引用概念

引用不是新定义一个变量,而是给已存在变量取了一个别名,编译器不会为引用变量开辟内存空间,它和它引用的变量共用同一块内存空间。
引用变量的定义: 类型& 引用变量名(对象名) = 引用实体;

//引用
#include<iostream>
using namespace std;int main()
{int a = 0;//b为a变量的别名,a和b共用一块内存空间,不管修改a或b的值,这片空间的值都会改变int& b = a;//a和b共用一块内存空间,所以a和b的地址一样cout << &b << endl;cout << &a << endl;a++;b++;cout << a << endl;cout << b << endl;return 0;
}

在这里插入图片描述

2、引用特性

1.引用在定义时必须初始化
2.一个变量可以有多个引用
3.引用一旦引用一个实体,再不能引用其他实体

//引用特性
#include<iostream>
using namespace std;int main()
{int a = 1;//1.引用在定义时必须初始化,不然就不知道该引用是哪个变量的别名//int& b; //2.一个变量可以有多个引用,即这些变量都和a共享一片内存空间,只要其中一个引用的值改变了,该内存空间的值就改变了int& b = a;int& c = a;//还可以给a的别名c再起一个别名,此时d也和c一样,与a共享一片内存空间。int& d = c;++a;++b;++c;++d;cout << a << endl;cout << b << endl;cout << c << endl;cout << d << endl;int x = 10;//3.引用一旦引用一个实体,就不能再引用其他实体b = x;  //因为此时是给b赋值为x,并不是将b变为x的别名cout << a << endl;cout << b << endl;cout << c << endl;cout << d << endl;return 0;
}

常引用

//常引用
#include<iostream>
using namespace std;
int main()
{const int a = 10;//下列语句编译时会出错,因为a被const修饰,a为常变量//int& ra = a; //而将ra用const修饰就不会出错const int& ra = a;//下列语句编译时会出错,因为b为常量//int& b = 10;const int& b = 10;double d = 12.34;//下列语句编译时会出错,因为类型不同//int& rd = d; const int& rd = d;return 0;
}

3、引用的应用 – 做参数

(1)输出型参数
引用的一个使用场景就是作为输出型参数,即在函数定义形参时,将形参定义为实参的引用,然后就可以通过引用来改变实参的值。

//引用应用
#include<iostream>
using namespace std;
//形参是实参的别名,即引用,所以可以通过形参来改变实参的值
void Swap(int& r1, int& r2)
{int tmp = r1;r1 = r2;r2 = tmp;
}
int main()
{int a = 0;int b = 2;cout << "a = " << a << endl;cout << "b = " << b << endl;Swap(a, b);cout << "a = " << a << endl;cout << "b = " << b << endl;return 0;
}

所以我们可以使用引用来修改一下之前写的顺序表,我们还记得之前写的顺序表代码如下。

typedef int SLDataType;
typedef struct SeqList
{SLDataType* data;  //指向动态开辟的数组int size;  //有效数据个数int capacity;  //顺序表的容量
}SL;void SLInit(SL* ps)
{assert(ps);SLDataType* tmp = (SLDataType*)calloc(4, sizeof(SLDataType));  //初始化时自动开辟4个空间if (tmp == NULL){perror("SLInit");exit(-1);  //退出程序}ps->data = tmp;ps->size = 0;ps->capacity = 4;
}int main()
{//创建一个SL类型的结构体变量SL s;//需要将s地址传入函数,这样函数中才能通过s的地址来改变s结构体变量的值SLInit(&s);return 0;
}

当我们了解了引用之后,我们就可以将上面的代码改为使用引用来写,而不需要使用指针。

#include<iostream>
#include<stdlib.h>
#include<assert.h>
using namespace std;typedef int SLDataType;
typedef struct SeqList
{SLDataType* data;  //指向动态开辟的数组int size;  //有效数据个数int capacity;  //顺序表的容量
}SL;//形参是实参的别名,即引用,所以可以通过形参来改变实参的内容。
void SLInit(SL& ps)
{assert(&ps);SLDataType* tmp = (SLDataType*)calloc(4, sizeof(SLDataType));  //初始化时自动开辟4个空间if (tmp == NULL){perror("SLInit");exit(-1);  //退出程序}ps.data = tmp;ps.size = 0;ps.capacity = 4;
}int main()
{//创建一个SL类型的结构体变量SL s;//此时直接将该变量传入函数中SLInit(s);return 0;
}

那么我们前面写的单链表也可以改成使用引用的版本,我们之前写的单链表代码如下。在实现单链表的一些操作时,我们使用到了二级指针。

#include<stdlib.h>
#include<assert.h>
#include<stdio.h>typedef int SLTDataType;
typedef struct SListNode
{SLTDataType data;struct SListNode* next;
}SLTNode;void SListPushBack(SLTNode** pphead, SLTDataType x)
{assert(pphead);SLTNode* newNode = (SLTNode*)malloc(sizeof(SLTNode)); //创建一个新结点assert(newNode);newNode->data = x;newNode->next = NULL;if (*pphead == NULL)  //如果单链表为空,就使新结点为单链表的首结点。{*pphead = newNode;}else{//找单链表的尾结点SLTNode* tail = *pphead;while (tail->next != NULL){tail = tail->next;}tail->next = newNode;}
}int main()
{//创建一个单链表结点指针,并将该指针赋值为NULL,代表该单链表为空SLTNode* s = NULL;//此时如果函数SListPushBack()中想要改变s的值不为NULL,就需要将s的地址传入函数中,//而s本来就为指针,所以SListPushBack()函数的形参要定义一个二级指针,用来接收指针s的地址。SListPushBack(&s, 1);SListPushBack(&s, 2);SListPushBack(&s, 3);SListPushBack(&s, 4);return 0;
}

现在我们可以使用引用来改变上面的代码,这样我们就不用使用二级指针了。

#include<stdlib.h>
#include<assert.h>
#include<stdio.h>typedef int SLTDataType;
typedef struct SListNode
{SLTDataType data;struct SListNode* next;
}SLTNode;//将函数的形参定义为一个SLTNode*类型的指针变量的引用
//此时pphead就是list的别名,phead的改变也会影响list
void SListPushBack(SLTNode*& pphead, SLTDataType x)
{assert(&pphead);SLTNode* newNode = (SLTNode*)malloc(sizeof(SLTNode)); //创建一个新结点assert(newNode);newNode->data = x;newNode->next = NULL;if (pphead == NULL)  //如果单链表为空,就使新结点为单链表的首结点。{pphead = newNode;}else{//找单链表的尾结点SLTNode* tail = pphead;while (tail->next != NULL){tail = tail->next;}tail->next = newNode;}
}int main()
{//创建一个单链表结点指针,即代表该单链表为空SLTNode* list = NULL;//此时SListPushBack(SLTNode*& pphead, SLTDataType x)的形参为list指针变量的引用//所以在函数中通过该引用就可以改变list的内容SListPushBack(list, 1);SListPushBack(list, 2);SListPushBack(list, 3);SListPushBack(list, 4);return 0;
}

在一些数据结构的书上也会这样定义,即将上面的代码再次简化。

#include<stdlib.h>
#include<assert.h>
#include<stdio.h>typedef int SLTDataType;
typedef struct SListNode
{SLTDataType data;struct SListNode* next;
}SLTNode,*PSLTNode;//*PSLTNode等价于下面的语句
//typedef struct SListNode* PSLTNode;//将函数的形参定义为一个SLTNode*类型的指针变量的引用
//void SListPushBack(SLTNode*& pphead, SLTDataType x)
//上面的语句等价于下面的语句
void SListPushBack(PSLTNode& pphead, SLTDataType x)
{assert(&pphead);SLTNode* newNode = (SLTNode*)malloc(sizeof(SLTNode)); //创建一个新结点assert(newNode);newNode->data = x;newNode->next = NULL;if (pphead == NULL)  //如果单链表为空,就使新结点为单链表的首结点。{pphead = newNode;}else{//找单链表的尾结点SLTNode* tail = pphead;while (tail->next != NULL){tail = tail->next;}tail->next = newNode;}
}int main()
{//等价于SLTNode* s;//通过PSLTNode创建一个单链表结点指针,该指针为空,就说明该单链表为空PSLTNode s = NULL;SListPushBack(s, 1);SListPushBack(s, 2);SListPushBack(s, 3);SListPushBack(s, 4);return 0;
}

(2)大对象传参,提高效率

#include<time.h>
#include<iostream>
using namespace std;
struct A
{int a[10000];
};void TestFunc1(A aa)
{}void TestFunc2(A& aa)
{}int main()
{A a;//以值作为函数参数int begin1 = clock();for (int i = 0; i < 10000; ++i){//每次调用TestFunc1函数都会在该函数中重新创建一个A结构体,并且该结构体的数据都拷贝实参a的值TestFunc1(a);}int end1 = clock();//以引用作为函数参数int begin2 = clock();for (int i = 0; i < 10000; ++i){//调用函数TestFunc2不会重新创建A结构体,所以该循环会效率更高TestFunc2(a);}int end2 = clock();cout << end1 - begin1 << endl;cout << end2 - begin2 << endl;return 0;
}

4、引用的应用 – 做返回值

(1)输出型返回对象
我们知道c语言中每当有一个函数调用时,编译器都会在栈区中为该次函数调用开辟一片空间,例如有如下一个程序。Count()函数的返回值为int型,那么函数调用时是如何将返回值带回到main函数内的呢?

我们先看下面两个传值返回的例子。
(1)我们可以看到当Count函数执行完后,编译器为该函数分配的空间就会回收,此时存储返回值n的空间也销毁,那么正确的n的值是怎么回到main函数中的呢?其实当Count函数返回值较小时,此时系统会直接将Count函数的返回值存在寄存器中,然后main函数去该寄存器取返回值即可。当Count函数返回值较大时,会先在main的栈帧中创建一个临时变量,然后将Count栈帧里面的n拷贝到main的临时变量中,不会直接取Count函数的栈帧中取n,因为此时Count函数的栈帧已经销毁,n的内存空间中已经存的不是原来的值了。
在这里插入图片描述
(2)下图中的n为static修饰的静态变量,所以此时n存储在静态区。此时虽然函数Count的空间销毁后,n还是存在静态区中没有被销毁,但是在main函数中获取Count函数的返回值时也不会去静态区去取n的值。因为编译器不知道此时静态区中有Count函数的返回值。而是还会将Count的返回值和上面例子的处理情况一样,即将Count函数的返回值拷贝一份到main函数的临时变量中。
所以不管n是在栈区还是静态区,编译器都会生成一个函数返回对象的拷贝,用来作为函数调用返回值,这样调用函数的栈帧销毁了,返回值也还有拷贝的一份,不会让函数的返回值丢失。
在这里插入图片描述

我们再看下面的两个传引用返回的例子。
下面的例子中main函数中的ret的值是不确定的,所以这种情况不能使用传引用返回。只能使用传值返回。
在这里插入图片描述
下面的例子中就可以使用传引用返回,因为返回值n在静态区中,Count函数执行时的空间被系统销毁后,n的值还在。所以当出了函数作用域,返回对象就销毁了时,那么一定不能用传引用返回,一定要用传值返回,只有在出了函数作用域后,返回对象还没有销毁的情况时才可以使用传引用返回。而传值返回不管什么情况都可以使用,因为传值返回会将返回值拷贝一份。传引用返回就是比传值返回少了一次返回值的拷贝。
在这里插入图片描述
知道了上面的知识后,我们又可以将以前写的关于顺序表的修改数据的函数改一下,改为传引用返回的函数。原来我们写的顺序表的修改数据的代码如下。

#include<assert.h>
#include<stdlib.h>
#include<stdio.h>
typedef int SLDataType;
typedef struct SeqList
{SLDataType* data;  //指向动态开辟的数组。int size;  //有效数据的个数int capacity;  // 容量空间的大小
}SL;
void SLInit(SL* ps)
{assert(ps);SLDataType* tmp = (SLDataType*)calloc(4, sizeof(SLDataType));  //初始化时自动开辟4个空间if (tmp == NULL){perror("SLInit");exit(-1);  //退出程序}ps->data = tmp;ps->size = 0;ps->capacity = 4;
}void SLCheckCapacity(SL* ps)
{assert(ps);if (ps->size == ps->capacity && ps->size != 0){SLDataType* tmp = (SLDataType*)realloc(ps->data, (2 * ps->capacity) * sizeof(SLDataType));if (tmp == NULL){perror("SLCheckCapacity");exit(-1);}ps->data = tmp;ps->capacity = ps->capacity * 2;}
}void SLPushBack(SL* ps, SLDataType x)
{assert(ps);SLCheckCapacity(ps);  //先检查是否需要扩容ps->data[ps->size] = x;  //插入数据ps->size++;
}void SLModify(SL* ps, int pos, SLDataType x)
{assert(ps && pos >= 0 && pos <= ps->size);ps->data[pos] = x;
}int main()
{SL s;SLInit(&s);SLPushBack(&s, 1);SLPushBack(&s, 2);SLPushBack(&s, 3);SLPushBack(&s, 4);SLModify(&s, 2, 6);return 0;
}

然后我们可以将SLModify函数改为一个传引用返回的函数,这样我们就可以根据返回的别名,来修改顺序表中元素的值。

#include<assert.h>
#include<stdlib.h>
#include<stdio.h>
#include<iostream>
using namespace std;
typedef int SLDataType;
typedef struct SeqList
{SLDataType* data;  //指向动态开辟的数组。int size;  //有效数据的个数int capacity;  // 容量空间的大小
}SL;
void SLInit(SL& ps)
{assert(&ps);SLDataType* tmp = (SLDataType*)calloc(4, sizeof(SLDataType));  //初始化时自动开辟4个空间if (tmp == NULL){perror("SLInit");exit(-1);  //退出程序}ps.data = tmp;ps.size = 0;ps.capacity = 4;
}void SLCheckCapacity(SL& ps)
{assert(&ps);if (ps.size == ps.capacity && ps.size != 0){SLDataType* tmp = (SLDataType*)realloc(ps.data, (2 * ps.capacity) * sizeof(SLDataType));if (tmp == NULL){perror("SLCheckCapacity");exit(-1);}ps.data = tmp;ps.capacity = ps.capacity * 2;}
}void SLPushBack(SL& ps, SLDataType x)
{assert(&ps);SLCheckCapacity(ps);  //先检查是否需要扩容ps.data[ps.size] = x;  //插入数据ps.size++;
}int& SLModify(SL& ps, int pos)
{assert(&ps && pos >= 0 && pos <= ps.size);return ps.data[pos];
}int main()
{SL s;SLInit(s);SLPushBack(s, 1);SLPushBack(s, 2);SLPushBack(s, 3);SLPushBack(s, 4);for (int i = 0; i < s.size; ++i){cout << SLModify(s, i) << " " ;}cout << endl;//此时返回的就是s.data[2]的别名,通过该别名就可以修改s.data[2]的值SLModify(s, 2)++;SLModify(s, 3) = 10;for (int i = 0; i < s.size; ++i){cout << SLModify(s, i) << " ";}cout << endl;//虽然也可以直接s.data[2]来修改,但是这个例子就是举例说明一下传引用返回的应用s.data[2]++;for (int i = 0; i < s.size; ++i){cout << SLModify(s, i) << " ";}cout << endl;return 0;
}

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

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

相关文章

数据分析之面试题目汇总

1、解释数据清洗的过程及常见的清洗方法。 数据清洗是指在数据分析过程中对数据进行检查、处理和纠正的过程&#xff1b;是数据预处理的一步&#xff0c;用于处理数据集中的无效、错误、缺失或冗余数据 常见的清洗方法包括&#xff1a;处理缺失值、处理异常值、去除重复值、统一…

强化学习算法总结 (1)

强化学习算法总结 (1) 1.综述 强化学习是通过与环境进行交互&#xff0c;来实现目标的一种计算方法。 s − a 1 − r − s ′ s - a_1 - r- s s−a1​−r−s′ 1.1强化学习优化目标 p o l i c y a r g m a x p o l i c y E ( a , s ) [ r e w a r d ( s , a ) ] policy ar…

CSS实现隐藏滚动条但可以滚动

场景 隐藏滚动条&#xff0c;但可以滚动 解决 全局样式 /* 隐藏滚动条 */ .outer-container::-webkit-scrollbar {width: 0; /* 设置滚动条的宽度为0 */background-color: transparent; /* 设置滚动条背景为透明 */ }/* 自定义滚动条轨道样式 */ .outer-container::-webkit…

day37:网编day4,多点通信和并发服务器

一、广播接收方&#xff1a; #include <myhead.h>#define ERR_MSG(msg) do{\ fprintf(stderr,"__%d__\n",__LINE__);\ perror(msg);\ }while(0)#define BRD_IP "192.168.114.255" #define BRD_PORT 8888int main(int argc, const char *argv[]) {//…

three.js 场景中如何彻底删除模型和性能优化

three.js 场景中如何彻底删除模型和性能优化 删除外部模型 在three.js场景中&#xff0c;要彻底删除外部模型&#xff0c;需要执行以下几个步骤&#xff1a; 从场景中移除模型 你可以使用 scene.remove(model) 或者 scene.remove(model.children[0]) 将模型从场景中移除。如果…

【⑰MySQL】 变量 | 循环 | 游标 | 处理程序

前言 ✨欢迎来到小K的MySQL专栏&#xff0c;本节将为大家带来MySQL变量 | 循环 | 游标 | 处理程序的分享✨ 目录 前言1. 变量1.1系统变量1.2 用户变量 2. 定义条件与处理程序2.1 案例分析2.2 定义条件2.3 定义处理程序2.4 案例解决 3. 流程控制3.1 分支结构3.2 循环结构3.3 跳转…

ELK安装、部署、调试 (八)logstash配置语法详解

input {#输入插件 }filter {#过滤插件 }output {#输出插件 } 1.读取文件。 使用filewatch的ruby gem库来监听文件变化&#xff0c;并通过.sincedb的数据库文件记录被监听日志we年的读取进度&#xff08;时间 搓&#xff09; 。sincedb数据文件的默认路径为<path.data>/…

sentinel-core

引入依赖<dependencies><dependency><groupId>com.alibaba.csp</groupId><artifactId>sentinel-core</artifactId></dependency><dependency><groupId>com.alibaba.csp</groupId><artifactId>sentinel-anno…

Revit SDK 介绍:DWGFamilyCreation 导入DWG

前言 这个例子介绍了如何导入 DWG。 内容 核心逻辑代码&#xff1a; // 设置导入选项 DWGImportOptions options new DWGImportOptions(); options.Placement Autodesk.Revit.DB.ImportPlacement.Origin; options.OrientToView true; ElementId elementId null; // 导入…

【Vue2.0源码学习】生命周期篇-初始化阶段(initInjections)

文章目录 1. 前言2. initInjections函数分析resolveInject函数分析 3. 总结 1. 前言 本篇文章介绍生命周期初始化阶段所调用的第四个初始化函数——initInjections。从函数名字上来看&#xff0c;该函数是用来初始化实例中的inject选项的。说到inject选项&#xff0c;那必然离…

labview卸载重装碰到的问题

目录 labeiw卸载重装过程当中总是碰到一些问题&#xff0c;记录一下解决办法&#xff0c;碰到后好查找。 个人推荐卸载 用laview自带的卸载软件进行卸载&#xff0c;卸载的比较干净。 卸载完全第一步 启动 labview自带的卸载软件进行卸载。一般进行完这一步&#xff0c;就…

设计模式-9--迭代器模式(Iterator Pattern)

一、什么是迭代器模式 迭代器模式&#xff08;Iterator Pattern&#xff09;是一种行为型设计模式&#xff0c;用于提供一种统一的方式来访问一个聚合对象中的各个元素&#xff0c;而不需要暴露该聚合对象的内部结构。迭代器模式将遍历集合的责任从集合对象中分离出来&#xf…

【深入解析spring cloud gateway】07 自定义异常返回报文

Servlet的HttpResponse对象&#xff0c;返回响应报文&#xff0c;一般是这么写的&#xff0c;通过输出流直接就可以将返回报文输出。 OutputStream out response.getOutputStream(); out.write("输出的内容"); out.flush();在filter中如果发生异常&#xff08;例如…

Java中的网络编程------基于Socket的TCP编程和基于UDP的网络编程,netstat指令

Socket 在Java中&#xff0c;Socket是一种用于网络通信的编程接口&#xff0c;它允许不同计算机之间的程序进行数据交换和通信。Socket使得网络应用程序能够通过TCP或UDP协议在不同主机之间建立连接、发送数据和接收数据。以下是Socket的基本介绍&#xff1a; Socket类型&…

1775_树莓派3B键盘映射错误解决

全部学习汇总&#xff1a; GitHub - GreyZhang/little_bits_of_raspberry_pi: my hacking trip about raspberry pi. 入手树莓派3B之后用了没有多长时间&#xff0c;最初的这段时间感觉想让它代替我的PC机是不肯能的。性能先不说&#xff0c;我完全没有找到当初在我的笔记本上使…

css网格布局

css网格布局 常用属性 display: grid; //开启网格grid-template-columns: 2fr 1fr 1fr 1fr 1fr; //设置多少列每列宽度grid-gap: 10px; // 设置表格之间间距grid-template-rows: 50px 50px 50px 50px; // 设置多少行 每行的高度grid-column : 1 //占据位置 占据1格grid-colu…

ChatGPT分析日本排放核污水对世界的影响

文章目录 1 背景2 环境影响3 健康影响4 国际关系影响5 应对措施 近段时间被日本排放核污水到海里的消息刷屏了&#xff0c;这一举措引发了广泛的关注和担忧。本文结合ChatGPT来分析这件事的前因后果、会对世界造成的影响、以及应对措施。 1 背景 受2011年发生的大地震及海啸影响…

机器人中的数值优化(九)——拟牛顿方法(下)、BB方法

本系列文章主要是我在学习《数值优化》过程中的一些笔记和相关思考&#xff0c;主要的学习资料是深蓝学院的课程《机器人中的数值优化》和高立编著的《数值最优化方法》等&#xff0c;本系列文章篇数较多&#xff0c;不定期更新&#xff0c;上半部分介绍无约束优化&#xff0c;…

某物联网数智化园区行业基于 KubeSphere 的云原生实践

公司简介 作为物联网 数智化园区一体化解决方案提供商&#xff0c;我们致力于为大中型园区、停车场提供软硬件平台&#xff0c;帮助园区运营者实现数字化、智能化运营。 在使用 K8s 之前我们使用传统的方式部署上线&#xff0c;使用 spug&#xff08;一款轻量级无 Agent 的自…

网络版五子棋C++实现

目录 1.项目介绍 2.开发环境 3.核心技术 4.环境搭建 5.WebSocketpp介绍 5.1WebSocketpp是什么 5.2为什么使用WebSocketpp 5.3原理解析&#xff1a; 5.4WebSocketpp主要特性 6.WebSocketpp使用 7.JsonCpp使用 8.MySQL API 9.项目模块设计以及流程图 10.封装日志宏…