C++学习全教程(Day2)

一、数组

在程序中为了处理方便,常常需要把具有相同类型的数据对象按有序的形式排列起来,形成"一组"数据,这就是"数组"(array)

数组中的数据,在内存中是连续存放的,每个元素占据相同大小的空间,就像排好队一样。

1、数组的定义

数组的定义形式如下:

首先需要声明类型r数组中所有元素必须具有相同的数据类型;

数组名是一个标识符;后面跟着中括号,里面定义了数组中元素的个数,也就是数组的"长度";

元素个数也是类型的一部分,所以必须是确定的;

需要注意的是:

对数组做初始化,要使用花括号括起来的数值序列;

如果做了初始化,数组定义时的元素个数可以省略,编译器可以根据初始化列表自动推断出来;

初始值的个数,不能超过指定的元素个数;

初始值的个数,如果小于元素个数,那么会用列表中的值初始化靠前的元素;剩余元素用默认值填充,整型的默认值就是0;

如果没有做初始化,数组中元素的值都是未定义的;这一点和普通的局部变量一致;|

2、数组的访问

(1)访问数组元素

数组元素在内存中是连续存放的,它们排好了队之后就会有一个队伍中的编号,称为"索引",也叫"下标";通过下标就可以快速访问每个元素了,具体形式为:

这里也是用了中括号来表示元素下标位置,被称为"下标运算符"。比如 a[2]就表示数组a中下标为2的元素,可以取它的值输出,也可以对它赋值。

#include<iostream>

using namespace std;

int main()
{
	int a[] = {1,2,3,4,5,6};
	cout << "a[1] = " << a[1] << endl;
	a[1] = 666;
	cout << "a[1] = " << a[1] << endl;
	cin.get();
}

运行结果:

需要注意的是:

数组的下标从o开始;

因此 a[2]访问的并不是数组 a的第2个元素,而是第三个元素;一个长度为10的数组,下标范围是o~9,而不是1~10;

合理的下标,不能小于o,也不能大于(数组长度-1);否则就会出现数组下标越界;

(2)数组的大小

所有的变量,都会在内存中占据一定大小的空间;而数据类型就决定了它具体的大小。而对于数组这样的"复合类型",由于每个元素类型相同,因此占据空间大小的计算遵循下面的简单公式:

#include<iostream>

using namespace std;

int main()
{
	int a[] = {1,2,3,4,5,6};
	cout << "a[1] = " << a[1] << endl;
	a[1] = 666;
	cout << "a[1] = " << a[1] << endl;

	//获取数组大小的长度
	cout << "数组a所占空间大小为:" << sizeof(a) << endl;
	cout << "数组a中每个元素的大小:" << sizeof(a[0]) << endl;

	int size;
	size = sizeof(a) / sizeof(a[0]);
	cout << "数组a的长度为:" << size << endl;
	cin.get();
}

运行结果:

3、多维数组

C++中本质上没有"多维数组"这种东西,所谓的"多维数组",其实就是"数组的数组"。

二维数组int arr[3][4]表示: arr是一个有三个元素的数组,其中的每个元素都是一个int 数组,包含4个元素;

三维数组int arr2[2][5][10]表示: arr2是一个长度为2的数组,其中每个元素都是一个二维数组;这个二维数组有5个元素,每个元素都是一个长度为10的int 数组;

一般最常见的就是二维数组。它有两个"维度",第一个维度表示数组本身的长度,第二个表示每个元素的长度,一般分别把它们叫做"行"和"列"。

(1)多维数组的初始化

和普通的"一维"数组一样,多维数组初始化时,也可以用花括号括起来的一组数。使用嵌套的花括号可以让不同的维度更清晰:

需要注意:

内嵌的花括号不是必需的,因为数组中的元素在内存中连续存放,可以用一个花括号将所有数据括在一起;

初始值的个数,可以小于数组定义的长度,其它元素初始化为0值;这一点对整个二维数组和每一行的一维数组都适用;

如果省略嵌套的花括号,当初始值个数小于总元素个数时,会按照顺序依次填充(填满第一行,才填第二行);其它元素初始化为О值;

多维数组的维度,可以省略第一个,由编译器自动推断;即二维数组可以省略行数,但不能省略列数。

#include<iostream>

using namespace std;

int main()
{
	//初始化
	int a[3][4] = {1,2,3,4,5,6,7,8,9,10,11,12};

	int b[3][4] = { 
		{1,2,3,4},
		{5,6,7,8},
		{9,10,11,12}
	};

	int c[3][4] = { 1,2,3,4 };

	int d[][4] = { 1,2,3,4,5,6 };

	//访问
	cout << "b[1][1] = " << b[1][1] << endl;
	b[1][1] = 666;
	cout << "b[1][1] = " << b[1][1] << endl;

	//遍历
	//计算二维数组中的行数和列数
	cout << "二维数组b的总大小:" << sizeof(b) << endl;  //大小说的是字节
	cout << "二维数组每一行的大小:" << sizeof(b[0]) << endl;
	cout << "二维数组中每一个元素的大小:" << sizeof(b[0][0]) << endl;

	int rowCnt = sizeof(b) / sizeof(b[0]);  //行数
	int colCnt = sizeof(b[0]) / sizeof(b[0][0]); //列数
	cout << "rowCnt = " << rowCnt << endl;
	cout << "colCnt = " << colCnt << endl;

	for (int i = 0; i < rowCnt; i++)
	{
		for (int j = 0; j < colCnt; j++)
		{
			cout << b[i][j] << "\t";
		}
		cout << endl;
	}

	cin.get();
}

运行结果:

4、数组的排序

数组排序指的是给定一个数组,要求把其中的元素按照从小到大(或从大到小)顺序排列。

这是一个非常经典的需求,有各种不同的算法可以实现。我们这里介绍两种最基本、最简单的排序算法。

(1)选择排序

选择排序是一种简单直观的排序算法。

它的工作原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后追加到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

#include<iostream>

using namespace std;

int main()
{
	int a[] = {2,5,8,4,9,1,6,3,7,0};

	//计算数组a的大小
	int size = sizeof(a) / sizeof(a[0]);

	//选择排序
	for (int i = 0; i < size; i++)
	{
		for (int j = i + 1; j < size; j++)
		{
			if (a[j] < a[i])
			{
				int temp = a[j];
				a[j] = a[i];
				a[i] = temp;
			}
		}
	}
	for (int num : a)
	{
		cout << num << "\t";
	}
	cout << endl;

	cin.get();
}

运行结果:

(2)冒泡排序

冒泡排序也是一种简单的排序算法。

它的基本原理是:重复地扫描要排序的数列,一次比较两个元素,如果它们的大小顺序错误,就把它们交换过来。这样,一次扫描结束,我们可以确保最大(小)的值被移动到序列末尾。这个算法的名字由来,就是因为越小的元素会经由交换,慢慢"浮"到数列的顶端。

#include<iostream>

using namespace std;

int main()
{
	int a[] = {2,5,8,4,9,1,6,3,7,0};

	//计算数组a的大小
	int size = sizeof(a) / sizeof(a[0]);

	//冒泡排序
	for (int i = 0; i < size; i++)
	{
		for (int j = 0; j < size - i - 1; j++)
		{
			if (a[j] > a[j + 1])
			{
				int temp = a[j + 1];
				a[j + 1] = a[j];
				a[j] = temp;
			}
		}
	}
	
	for (int num : a)
	{
		cout << num << "\t";
	}
	cout << endl;

	cin.get();
}

运行结果:

二、模板类vector

数组尽管很灵活,但使用起来还是很多不方便。为此,C++语言定义了扩展的"抽象数据类型"(Abstract Data Type,ADT),放在"标准库"中。

对数组功能进行扩展的一个标准库类型,就是"容器"vector。顾名思义,vector"容纳"着一堆数据对象,其实就是一组类型相同的数据对象的集合。

1、头文件和命名空间

vector是标准库的一部分。要想使用vector,必须在程序中包含<vector>头文件,并使用std命名空间。

在vector头文件中,对vector这种类型做了定义;使用#include引入它之后,并指定命名空间std之后,我们就可以在代码中直接使用vector了。

2、 vector的基本用法

vector其实是C++中的一个"类模板",是用来创建类的"模子"。所以在使用时还必须提供具体的类型信息,也就是说,这个容器中到底要容纳什么类型的数据对象;具体的形式是在vector后面跟一个尖括号<>,里面填入具体类型信息。

#include<iostream>
#include<vector>

using namespace std;

int main()
{
	//默认初始化
	vector<int> v1; //定义了整型的容器,容器名:v1

	//列表初始化
	vector<char> v2 = { 'a','b','c' };
	vector<char> v3{ 'a','b','c' };

	//直接初始化
	vector<short> v4(5);  //定义了大小为5的短整型容器
	vector<double> v5(5,100);//定义了大小为5的double容器,且每个值都为100

	//访问元素
	cout << v5[2] << endl;
	v5[2] = 666;
	cout << v5[2] << endl;

	//遍历所有元素
	for (int i = 0; i < v5.size(); i++)
	{
		cout << v5[i] << "\t";
	}
	cout << endl;

	//添加元素
	v5.push_back(69);

	for (int num : v5)
	{
		cout << num << "\t";
	}
	cout << endl;

	//向容器中添加倒序的元素
	for (int i = 10; i > 0; i--)
	{
		v1.push_back(i);
	}
	//遍历容器v1
	for (int num : v1)
	{
		cout << num << "\t";
	}
	cout << endl;


	cin.get();
}

运行结果:

2、vector和数组的区别

数组是更加底层的数据类型;长度固定,功能较少,安全性没有保证;但性能更好,运行更高效;

vector是模板类,是数组的上层抽象;长度不定,功能强大;缺点是运行效率较低;

除了vector之外,C++ 11还新增了一个array模板类,它跟数组更加类似,长度是固定的,但更加方便、更加安全。所以在实际应用中,一般推荐对于固定长度的数组使用array,不固定长度的数组使用vector。

三、字符串

字符串我们并不陌生。之前已经介绍过,一串字符连在一起就是一个"字符串",比如用双引号引起来的"Hello World !"就是一个字符串字面值。

字符串其实就是所谓的"纯文本",就是各种文字、数字、符号在一起表达的一串信息;所以字符串就是C++中用来表达和处理文本信息的数据类型。

1、标准库类型string

C++的标准库中,提供了一种用来表示字符串的数据类型string,这种类型能够表示长度可变的字符序列。和 vector类似,Istring 类型也定义在命名空间std中,使用它必须包含string头文件。

(1)定义和初始化 string

我们已经接触过C++中几种不同的初始化方式, string 也是一个标准库类型,它的初始化与vector非常相似。

#include<iostream>
#include<string>

using namespace std;

int main()
{
	//初始化
	string s1;

	//拷贝初始化
	string s2 = s1;
	string s3 = "hello world";

	//直接初始化
	string s4("hello world");
	string s5(8, 'h');

	cout << s5 << endl;

	//访问字符
	cout << "s4[1] = " << s4[1] << endl;
	s4[0] = 'H';
	cout << s4 << endl;

	cin.get();
}

运行结果:

字符串内字符的访问,跟vector内元素的访问类似,需要注意:

string内字符的索引,也是从o开始;

string 同样有一个成员函数size,可以获取字符串的长度;

索引最大值为字符串长度- 1),不能越界访问;如果直接越界访问并赋值,有可能导致非常严重的后果,出现安全问题;

如果希望遍历字符串的元素,也可以使用普通for 循环和范围for 循环,依次获取每个字符

#include<iostream>
#include<string>

using namespace std;

int main()
{
	//初始化
	string s1;

	//拷贝初始化
	string s2 = s1;
	string s3 = "hello world";

	//直接初始化
	string s4("hello world");
	string s5(8, 'h');

	cout << s5 << endl;

	//访问字符
	cout << "s4[1] = " << s4[1] << endl;
	s4[0] = 'H';
	cout << s4 << endl;

	//遍历
	for (int i = 0; i < s4.size(); i++)
	{
		s4[i] = toupper(s4[i]);  //toupper:小写转换大写
	}
	cout << s4 << endl;

	cin.get();
}

运行结果:

(2)字符串相加

string本身的长度是不定的,可以通过"相加"的方式扩展一个字符串。

需要注意:

字符串相加使用加号"E"来表示,这是算术运算符"+"的运算符重载,含义是"字符串拼接";

两个string对象,可以直接进行字符串相加;结果是将两个字符串拼接在一起,得到一个新的string对象返回;

一个string对象和一个字符串字面值常量,可以进行字符串相加,同样是得到一个拼接后的string对象返回;

两个字符串字面值常量,不能相加;

多个string 对象和多个字符串字面值常量,可以连续相加;前提是按照左结合律,每次相加必须保证至少有一个string对象;

(3)比较字符串

字符串比较的规则为:

如果两个字符串长度相同,每个位置包含的字符也都相同,那么两者"相等";否则"不相等";

如果两个字符串在某一位置上开始不同,那么就比较这两个字符的 ASCIl码,比较结果就代表两个字符串的大小关系

如果两个字符串在某一位置上开始不同,那么就比较这两个字符的 ASCIl码,比较结果就代表两个字符串的大小关系

2、字符数组

通过对string的介绍可以发现,字符串就是一串字符的集合,本质上其实就是一个"字符的数组"。

在C语言中,确实是用char[]类型来表示字符串的;不过为了区分纯粹的"字符数组"和"字符串",C语言规定:字符串必须以空字符结束。空字符的ASClI码为o,专门用来标记字符串的结尾,在程序中写作\0'。

#include<iostream>

using namespace std;

int main()
{
	char str1[5] = { 'g','y','s','z','s' };//str1不是一个字符串
	char str2[6] = { 'g','y','s','z','s','\0'};//str2是一个字符串

	cout << "str1 = " << str1 << endl;
	cout << "str2 = " << str2 << endl;

	cin.get();
}

运行结果:

3、读取输入的字符串

(1)使用输入操作符读取单词

标准库中提供了iostream,可以使用内置的 cin对象,调用重载的输入操作符>>来读取键盘输入。

这种方式的特点是:忽略开始的空白符,遇到下一个空白符(空格、回车、制表等)就会停止。所以如果我们输入""hello world",那么读取给str的只有"hello"这相当于读取了一个"单词"。

剩下的内容"world"其实也没有丢,而是保存在了输入流的"输入队列"里。如果我们想读取更多的输入信息,就需要使用更多的string对象来获取:

(2〉使用getline读取一行

如果希望直接读取一整行输入信息,可以使用getline函数来替代输入操作符。

getline函数有两个参数:一个是输入流对象cin,另一个是保存字符串的string 对象;它会一直读取输入流中的内容,直到遇到换行符为止,然后把所有内容保存到string 对象中。所以现在可以完整读取一整行信息了。

#include<iostream>
#include<string>

using namespace std;

int main()
{
	//使用getline读取一行信息
	string str1;

	getline(cin, str1);
	cin.get();
	cout << "str1 = " << str1 << endl;

	cin.get();
}

运行结果:

(3)使用get读取字符

还有一种方法,是调用cin 的 get函数读取一个字符。

有两种方式:

调用cin.get()函数,不传参数,得到一个字符赋给char类型变量;

将char类型变量作为参数传入,将捕获的字符赋值给它,返回的是istream对象

(4)读写文件

C++的lo库中提供了专门用于文件输入的 ifstream类和用于文件输出的ofstream类,要使用它们需要引入头文件 fstream。ifstream 用于读取文件内容,跟istream 的用法类似;也可以通过输入操作符>>来读"单词"(空格分隔),通过getline 函数来读取一行,通过get函数来读取一个字符:

#include<iostream>
#include<fstream>
#include<string>

using namespace std;

int main()
{
	ifstream input("1.txt"); //通过input对象来从文件中读取数据
	ofstream output("2.txt");

	//1.按照单词逐个读取
	/*string word;
	while (input >> word)
	{
		cout << word << endl;
	}*/
	//2.逐行读取
	string line;
	while (getline(input, line))
	{
		cout << line << endl;
		//output << line << endl;
	}

	cin.get();
}

运行结果:

四、结构体

C/C++中提供了另一种更加灵活的数据结构------结构体。结构体是用户自定义的复合数据结构,里面可以包含多个不同类型的数据对象。

1、结构体的定义和声明

声明一个结构体需要使用struct关键字,具体形式如下:

#include<iostream>

using namespace std;

//定义1个结构体
struct  student
{
	string name;
	int age;
	double score;
};

int main()
{
	//创建对象并初始化
	student s1 = { "ljl",18,99 };

	cin.get();
}

2、访问结构体中数据

访问结构体变量中的数据成员,可以使用成员运算符点号.),后面跟上数据成员的名称。例如stu.name就可以访问stu对象的name成员。

#include<iostream>

using namespace std;

//定义1个结构体
struct  student
{
	string name;
	int age;
	double score;
};

//输出一个数据对象的完整信息
void pintinfo(student s)
{
	//访问数据
	cout << "学生姓名:" << s.name << "\t年龄:" << s.age << "\t成绩:" << s.score << endl;
}

int main()
{
	//创建对象并初始化
	student s1 = { "ljl",18,99 };
	student s2;

	s2.age = 20;
	s2.name = "xiaoming";
	s2.score = 80;

	pintinfo(s1);
	pintinfo(s2);

	cin.get();
}

运行结果:

3、结构体数组

可以把结构体和数组结合起来,创建结构体的数组。顾名思义,结构体数组就是元素为结构体的数组,它的定义和访问跟普通的数组完全一样。

#include<iostream>

using namespace std;

//定义1个结构体
struct  student
{
	string name;
	int age;
	double score;
};

//输出一个数据对象的完整信息
void pintinfo(student s)
{
	//访问数据
	cout << "学生姓名:" << s.name << "\t年龄:" << s.age << "\t成绩:" << s.score << endl;
}

int main()
{
	//创建对象并初始化
	student s1 = { "ljl",18,99 };
	student s2;

	s2.age = 20;
	s2.name = "xiaoming";
	s2.score = 80;

	pintinfo(s1);
	pintinfo(s2);

	//结构体数组
	student s3[3] = {
		{"小王",18,88},
		{"小张",19,66},
		{"小钱",20,99}
	};
	pintinfo(s3[0]);

	cin.get();
}

运行结果:

五、枚举

枚举类型的定义和结构体非常像,需要使用enum关键字。

与结构体不同的是,枚举类型内只有有限个名字,它们都各自代表一个常量,被称为"枚举量"。

需要注意的是:

默认情况下,会将整数值赋给枚举量;

枚举量默认从0开始,每个枚举量依次加1;所以上面week枚举类型中,一周七天枚举量分别对应着0~6的常量值;

可以通过对枚举量赋值,显式地设置每个枚举量的值;

#include<iostream>

using namespace std;

//定义1个枚举类型
enum Week {
	Mon, Tue, Wed, Thu, Fri, Sta, Sun
};



int main()
{
	Week w1 = Mon, w2 = Tue;
	cout << "w1 = " << w1 << endl;
	cout << "w2 = " << w2 << endl;

	cin.get();
}

运行结果:

六、指针 ---存放是地址

指针顾名思义,是"指向"另外一种数据类型的复合类型。指针是C/C++中一种特殊的数据类型,它所保存的信息,其实是另外一个数据对象在内存中的"地址"。通过指针可以访问到指向的那个数据对象,所以这是一种间接访问对象的方法。

1、指针的定义

这里的类型就是指针所指向的数据类型,后面加上星号"*",然后跟指针变量的名称。指针在定义的时候可以不做初始化。相比一般的变量声明,看起来指针只是多了一个星号"*"而已。例如:

#include<iostream>

using namespace std;

int main()
{
	int* p;
	long* p1;
	long long* p2;

	cout << "p在内存中的长度为:" << sizeof(p) << endl;
	cout << "p1在内存中的长度为:" << sizeof(p1) << endl;
	cout << "p2在内存中的长度为:" << sizeof(p2) << endl;

	cin.get();
}

运行结果:

指针在64位系统下占据8个字节,在32位系统下占4个字节。

2、指针的用法

(1)获取对象地址给指针赋值

指针保存的是数据对象的内存地址,所以可以用地址给指针赋值;获取对象地址的方式是使用"取地址操作符"((&)。

#include<iostream>

using namespace std;

int main()
{
	int* p;
	long* p1;
	long long* p2;

	cout << "p在内存中的长度为:" << sizeof(p) << endl;
	cout << "p1在内存中的长度为:" << sizeof(p1) << endl;
	cout << "p2在内存中的长度为:" << sizeof(p2) << endl;

	//指针的使用
	int a = 10;
	long b = 20;
	
	p = &a;
	p1 = &b;

	cout << "a的地址为:" << &a << endl;
	cout << "b的地址为:" << &b << endl;
	cout << "p:" << p << endl;
	cout << "p1:" << p1 << endl;

	cin.get();
}

运行结果:

(2)通过指针访问对象

指针指向数据对象后,可以通过指针来访问对象。访问方式是使用"解引用操作符"(*):

在这里由于p指向了a,所以*p可以等同于a。

#include<iostream>

using namespace std;

int main()
{
	int* p;
	long* p1;
	long long* p2;

	cout << "p在内存中的长度为:" << sizeof(p) << endl;
	cout << "p1在内存中的长度为:" << sizeof(p1) << endl;
	cout << "p2在内存中的长度为:" << sizeof(p2) << endl;

	//指针的使用
	int a = 10;
	long b = 20;
	
	p = &a;
	p1 = &b;

	cout << "a的地址为:" << &a << endl;
	cout << "b的地址为:" << &b << endl;
	cout << "p:" << p << endl;
	cout << "p1:" << p1 << endl;

	*p = 666;
	cout << "a = " << a << endl;

	cin.get();
}

运行结果:

3、无效指针、空指针和 void*指针

(1)无效指针

定义一个指针之后,如果不进行初始化,那么它的内容是不确定的(比如Oxcccc)。如果这时把它的内容当成一个地址去访问,就可能访问的是不存在的对象;更可怕的是,如果访问到的是系统核心内存区域,修改其中内容会导致系统崩溃。这样的指针就是"无效指针",也被叫做"野指针"。

指针非常灵活非常强大,但野指针非常危险。所以建议使用指针的时候,一定要先初始化,让它指向真实的对象。

(2)空指针

如果先定义了一个指针,但确实还不知道它要指向哪个对象,这时可以把它初始化为"空指针"。空指针不指向任何对象。

(3)void* 指针

一般来说,指针的类型必须和指向的对象类型匹配,否则就会报错。不过有一种指针比较特殊,可以用来存放任意对象的地址,这种指针的类型是void*。

4、指向指针的指针

指针本身也是一个数据对象,也有自己的内存地址。所以可以让一个指针保存另一个指针的地址,这就是"指向指针的指针",有时也叫"二级指针";形式上可以用连续两个的星号**来表示。类似地,如果是三级指针就是***,表示"指向二级指针的指针"。

#include<iostream>

using namespace std;

int main()
{
	int* p;
	long* p1;
	long long* p2;

	cout << "p在内存中的长度为:" << sizeof(p) << endl;
	cout << "p1在内存中的长度为:" << sizeof(p1) << endl;
	cout << "p2在内存中的长度为:" << sizeof(p2) << endl;

	//指针的使用
	int a = 10;
	long b = 20;
	
	p = &a;
	p1 = &b;

	cout << "a的地址为:" << &a << endl;
	cout << "b的地址为:" << &b << endl;
	cout << "p:" << p << endl;
	cout << "p1:" << p1 << endl;

	*p = 666;
	cout << "a = " << a << endl;
	cout << "--------------------------------" << endl;
	//二级指针
	int i = 100;
	int* pi = &i;
	int** ppi = &pi;

	cout << "i = " << i << endl;
	cout << "pi = " << pi << endl;
	cout << "ppi = " << ppi << endl;
	cout << "*pi = " << *pi << endl;
	cout << "*ppi = " << *ppi << endl;
	cout << "**ppi = " << **ppi << endl;

	cin.get();
}

运行结果:

5、指针和const

指针可以和 const 修饰符结合,这可以有两种形式:一种是指针指向的是一个常量;另一种是指针本身是一个常量。

(1)指向常量的指针

指针指向的是一个常量,所以只能访问数据,不能通过指针对数据进行修改。不过指针本身是变量,可以指向另外的数据对象。这时应该把const加在类型前。

这里发现,pc是一个指向常量的指针,但其实把一个变量i的地址赋给它也是可以的;编译器只是不允许通过指针pc去间接更改数据对象。

(2)指针常量(const 指针)

指针本身是一个数据对象,所以也可以区分变量和常量。如果指针本身是一个常量,就意味它保存的地址不能更改,也就是它永远指向同一个对象;而数据对象的内容是可以通过指针改变的。这种指针一般叫做"指针常量"。

指针常量在定义的时候,需要在星号*后、标识符前加上 const。

6、指针和数组

(1)数组名

用到数组名时,编译器一般都会把它转换成指针,这个指针就指向数组的第一个元素。所以我们也可以用数组名来给指针赋值。

运行结果:

(2)指针运算

如果对指针pia 做加1操作,我们会发现它保存的地址直接加了4,这其实是指向了下一个int类型数据对象:

所谓的"指针运算",就是直接对一个指针加/减一个整数值,得到的结果仍然是指针。新指针指向的数据元素,跟原指针指向的相比移动了对应个数据单位。

(3)指针数组和数组指针

指针和数组这两种类型可以结合在一起,这就是"指针数组"和"数组指针"。

指针数组:一个数组,它的所有元素都是相同类型的指针;

数组指针:一个指针,指向一个数组的指针;

运行结果:

七、引用

1、引用的用法

在做声明时,我们可以在变量名前加上"&"符号,表示它是另一个变量的引用。引用必须被初始化。

#include<iostream>

using namespace std;

int main()
{
	int a = 888, b = 666;
	int& ref = a;

	cout << "ref = " << ref << endl;
	cout << "a的地址为: " << &a << endl;
	cout << "ref的地址为: " << &ref << endl;

	cin.get();
}

运行结果:

引用本质上就是一个"别名",它本身不是数据对象,所以本身不会存储数据,而是和初始值"绑定"(bind)在一起,绑定之后就不能再绑定别的对象了。

2、对常量的引用

可以把引用绑定到一个常量上,这就是"对常量的引用"。很显然,对常量的引用是常量的别名,绑定的对象不能修改,所以也不能做赋值操作:

对常量的引用有时也会直接简称"常量引用"。因为引用只是别名,本身不是数据对象;所以这只能代表"对一个常量的引用",而不会像"常量指针"那样引起混淆。

常量引用和普通变量的引用不同,它的初始化要求宽松很多,只要是可以转换成它指定类型的所有表达式,都可以用来做初始化。

3、指针和引用

(1)引用和指针常量

事实上,引用的行为,非常类似于"指针常量",也就是只能指向唯一的对象、不能更改的指针。

#include<iostream>

using namespace std;

int main()
{
	int a = 66;
	//引用和指针常量
	int& ref = a;
	int* const p = &a;

	ref = 666;
	cout << "a = " << a << endl;
	*p = 888;
	cout << "a = " << a << endl;

	cout << "a的地址为:" << &a << endl;
	cout << "ref的地址为:" << &ref << endl;
	cout << "p的地址为:" << &p << endl;
	cout << "p的值为: " << p << endl;

	//绑定指针的引用
	int* ptr = &a;
	int*& pref = ptr;  //pref是ptr指针的别名

	*ptr = 100;
	cout << "a = " << a << endl;

	cin.get();
}

运行结果:

八、函数

8.1、函数的定义

一个完整的函数定义主要包括以下部分:

返回类型:调用函数之后,返回结果的数据类型;

函数名:用来命名代码块的标识符,在当前作用域内唯一;

参数列表:参数表示函数调用时需要传入的数据,一般叫做"形参";放在函数名后的小括号里,可以有o个或多个,用逗号隔开;

函数体:函数要执行的语句块,用花括号括起来。

#include<iostream>

using namespace std;

//定义一个平方函数
int square(int x)
{
	int y = x * x;
	return y;
}

int main()
{
	cout << "请输入一个整数:";
	int num;
	cin >> num;
	cin.get();
	cout << "num的平方为:" << square(num) << endl;

	cin.get();
}

运行结果:

8.1.2、局部变量的生命周期

对于花括号内定义的变量,具有"块作用域",在花括号外就不可见了。函数体都是语句块,而主函数main本身也是一个函数;所以在main 中定义的所有变量、所有函数形参和在函数体内部定义的变量,都具有块作用域,统称为"局部变量"。局部变量仅在函数作用域内部可见。

在C++中,作用域指的是变量名字的可见范围;变量不可见,并不代表变量所指代的数据对象就销毁了。这是两个不同的概念:

作用域:针对名字而言,是程序文本中的一部分,名字在这部分可见;

生命周期:针对数据对象而言,是程序在执行过程中,对象从创建到销毁的时间段

基于作用域,变量可以分为"局部变量"和"全局变量"。对于全局变量而言,名字全局可见,对象也只有在程序结束时才销毁。

而对于局部变量代表的数据对象,基于生命周期,又可以分为"自动对象"和"静态对象"。

(1)自动对象

平常代码中定义的普通局部变量,生命周期为:在程序执行到变量定义语句时创建,在程序运行到当前块末尾时销毁。这样的对象称为"自动对象"。

形参也是一种自动对象。形参定义在函数体作用域内,一旦函数终止,形参也就被销毁了。

对于自动对象来说,它的生命周期和作用域是一致的。

(2)静态对象

如果希望延长一个局部变量的生命周期,让它在作用域外依然保留,可以在定义局部变量时加上static关键字;这样的对象叫做"局部静态对象"。

局部静态对象只有局部的作用域,在块外依然是不可见的;但是它的生命周期贯穿整个程序运行过程,只有在程序结束时才被销毁,这一点与全局变量类似。

8.1.3、分离式编译和头文件

(1)分离式编译

当程序越来越复杂,我们就会希望代码分散到不同的文件中来做管理。C++支持分离式编译,这就可以把函数单独放在一个文件,独立编译之后链接运行。

比如可以把复制字符串的函数单独保存成一个文件 copy_string.cpp:

(2)编写头文件

对于一个项目而言,有些定义可能是所有文件共用的,比如一些常量、结构体/类,以及功能性的函数。于是每次需要引入时,都得做一堆声明------这显然太麻烦了。

一个好方法是,把它们定义在同一个文件中,需要时用一句#include 统一引入就可以了,就像使用库一样。这样的文件以.h作为后缀,被称为"头文件"。

8.2、参数传递

参数传递和变量的初始化类似,根据形参的类型可以分为两种方式:传值(value)和传引用(reference)。

8.2.1、传引用参数

(1)传引用方便函数调用

C++新增了引用的概念,可以替换必须使用指针的场景。采用引用作为函数形参,可以使函数调用更加方便。这种传参方式叫做"传引用参数"。之前的例子就可以改写成:

#include<iostream>

using namespace std;

//使用引用作为形参
void increase(int& x)
{
	++x;
}

int main()
{
	int num = 0;
	increase(num);

	cout << "num = " << num << endl;

	cin.get();
}

运行结果:

(2)传引用避免拷贝

使用引用还有一个非常重要的场景,就是不希望进行值拷贝的时候。实际应用中,很多时候函数要操作的对象可能非常庞大,如果做值拷贝会使得效率大大降低;这时使用引用就是一个好方法。

比如,想要定义一个函数比较两个字符串的长度,需要将两个字符串作为参数传入。因为字符串有可能非常长,直接做值拷贝并不是一个好选择,最好的方式就是传递引用:

8.2.2、数组形参

数组是不允许做直接拷贝的,所以如果想要把数组作为函数的形参,使用值传递的方式是不可行的。与此同时,数组名可以解析成一个指针,所以可以用传递指针的方式来处理数组。|

比如一个简单的函数,需要遍历int类型数组所有元素并输出,就可以这样声明:

#include<iostream>

using namespace std;

//遍历数组
void printArray(const int* arr, int size)
{
	for (int i = 0; i < size; i++)
	{
		cout << arr[i] << "\t";
	}
	cout << endl;
}

int main()
{
	int arr[5] = { 1,2,3,4,5 };
	printArray(arr, 5);

	cin.get();
}

运行结果:

8.2.3、数组引用作为形参

#include<iostream>

using namespace std;

//遍历数组
void printArray(const int* arr, int size)
{
	for (int i = 0; i < size; i++)
	{
		cout << arr[i] << "\t";
	}
	cout << endl;
}

void printArray(const int(& arr)[5])
{
	for (int num : arr)
	{
		cout << num << "\t";
	}
	cout << endl;
}

int main()
{
	int arr[5] = { 1,2,3,4,5 };
	printArray(arr, 5);
	printArray(arr);

	cin.get();
}

运行结果:

8.3.1、无返回值

#include<iostream>

using namespace std;

void swap(int& x, int& y)
{
	int temp = x;
	x = y;
	y = temp;
}

int main()
{
	int a = 66, b = 88;
	swap(a, b);
	cout << "a = " << a << endl;
	cout << "b = " << b << endl;
	
	cin.get();
}

运行结果:

8.3.2、有返回值

#include<iostream>

using namespace std;

void swap(int& x, int& y)
{
	int temp = x;
	x = y;
	y = temp;
}

//有返回值的函数,返回较长的字符串
string longstr(const string& str1, const string& str2)
{
	return str1.size() >= str2.size() ? str1 : str2;
}

int main()
{
	int a = 66, b = 88;
	swap(a, b);
	cout << "a = " << a << endl;
	cout << "b = " << b << endl;

	string str1 = "hello";
	string str2 = "hello world";

	cout << longstr(str1, str2) << endl;
	
	cin.get();
}

运行结果:

8.3.3、返回数组指针

这里对于函数fun的声明,我们可以进行层层解析:

fun(int x):函数名为fun,形参为int类型的x;

(*fun(int 炳 ):函数返回的结果,可以执行解引用操作,说明是一个指针;

( *fun(int x))[5]:函数返回结果解引用之后是一个长度为5的数组,说明返回类型是数组指针;

int ( * fun(int x) )[5]:数组中元素类型为int

九、递归

如果一个函数调用了自身,这样的函数就叫做"递归函数"(recursivefunction)。

1、递归的实现

递归是调用自身,如果不加限制,这个过程是不会结束的;函数永远调用自己下去,最终会导致程序栈空间耗尽。所以在递归函数中,一定会有某种"基准情况",这个时候不会调用自身,而是直接返回结果。基准情况的处理保证了递归能够结束。|

递归是不断地自我重复,这一点和循环有相似之处。事实上,递归和循环往往可以实现同样的功能。

#include<iostream>

using namespace std;

//用递归实现阶乘
int factorial(int n)
{
	if (n == 1)
		return 1;
	return factorial(n - 1) * n;
}

int main()
{
	cout << "5! = " << factorial(5) << endl;

	cin.get();
}

运行结果:

这里我们的基准情况是n ==1,也就是当n不断减小,直到1时就结束递归直接返回。5的阶乘具体计算流程如下:

因为递归至少需要额外的栈空间开销,所以递归的效率往往会比循环低一些。不过在很多数学问题上,递归可以让代码非常简洁。

2、经典递归-------斐波那契数列

#include<iostream>

using namespace std;

//用递归实现阶乘
int factorial(int n)
{
	if (n == 1)
		return 1;
	return factorial(n - 1) * n;
}

//斐波那契数列
int fib(int n)
{
	if (n == 1 || n == 2)
		return 1;
	return fib(n - 1) + fib(n - 2);
}

int main()
{
	cout << "5! = " << factorial(5) << endl;
	cout << "fib(5) = " << fib(5) << endl;

	cin.get();
}

运行结果:

十、函数高阶

1、内联函数

内联函数是C++为了提高运行速度做的一项优化。

函数让代码更加模块化,可重用性、可读性大大提高;不过函数也有一个缺点:函数调用需要执行一系列额外操作,会降低程序运行效率。

为了解决这个问题,C++引入了"内联函数"的概念。使用内联函数时,编译器不再去做常规的函数调用,而是把它在调用点上"内联"展开,也就是直接用函数代码替换了函数调用。

1.1内联函数的定义

定义内联函数,只需要在函数声明或者函数定义前加上 inline关键字。

1.2内联函数和宏

内联函数是C++新增的特性。在C语言中,类似功能是通过预处理语句和#define定义"宏"来实现的。

然而c中的宏本身并不是函数,无法进行值传递;它的本质是文本替换,我们一般只用宏来定义常量。用宏实现函数的功能会比较麻烦,而且可读性较差。所以在C++中,一般都会用内联函数来取代C中的宏。

2、函数重载

在C++中,同一作用域下,同一个函数名是可以定义多次的,前提是形参列表不同。这种名字相同但形参列表不同的函数,叫做"重载函数"。这是C++相对c语言的重大改进,也是面向对象的基础。

2.1定义重装函数

重载的函数,应该在形参的数量或者类型上有所不同;

形参的名称在类型中可以省略,所以只有形参名不同的函数是一样的;

调用函数时,编译器会根据传递的实参个数和类型,自动推断使用哪个函数;

主函数不能重载

2.2函数匹配

确定到底调用哪个函数的过程,叫做"函数匹配"。

#include<iostream>

using namespace std;

void f() { cout << "1" << endl; }
void f(int x) { cout << "2" << endl; }
void f(int x, int y) { cout << "3" << endl; }
void f(double x, double y = 1.5) { cout << "4" << endl; }


int main()
{
	f(3.14);

	cin.get();
}

运行结果:

2.3函数重装和作用域

重载是否生效,跟作用域是有关系的。如果在内层、外层作用域分别声明了同名的函数,那么内层作用域中的函数会覆盖外层的同名实体,让它隐藏起来。

不同的作用域中,是无法重载函数名的。

3、函数指针

一类特殊的指针,指向的不是数据对象而是函数,这就是"函数指针"

3.1声明函数指针

函数指针本质还是指针,它的类型和所指向的对象类型有关。现在指向的是函数,函数的类型是由它的返回类型和形参类型共同决定的,跟函数名、形参名都没有关系。

3.2使用函数指针

当一个函数名后面跟调用操作符(小括号),表示函数调用;而单独使用函数名作为一个值时,函数会自动转换成指针。这一点跟数组名类似。

所以我们可以直接使用函数名给函数指针赋值:

也可以加上取地址符&,这和不加&是等价的。

3.3函数指针作为形参

有了指向函数的指针,就给函数带来了更加丰富灵活的用法。比如,可以将函数指针作为形参,定义在另一个函数中。也就是说,可以定义一个函数,它以另一个函数类型作为形参。当然,函数本身不能作为形参,不过函数指针完美地填补了这个空缺。这一点上,函数跟数组非常类似。

3.4函数指针作为返回值

相关推荐
我是谁??36 分钟前
C/C++使用AddressSanitizer检测内存错误
c语言·c++
Mephisto.java39 分钟前
【大数据学习 | kafka高级部分】kafka中的选举机制
大数据·学习·kafka
南宫生1 小时前
贪心算法习题其三【力扣】【算法学习day.20】
java·数据结构·学习·算法·leetcode·贪心算法
发霉的闲鱼1 小时前
MFC 重写了listControl类(类名为A),并把双击事件的处理函数定义在A中,主窗口如何接收表格是否被双击
c++·mfc
小c君tt1 小时前
MFC中Excel的导入以及使用步骤
c++·excel·mfc
xiaoxiao涛1 小时前
协程6 --- HOOK
c++·协程
武子康2 小时前
大数据-212 数据挖掘 机器学习理论 - 无监督学习算法 KMeans 基本原理 簇内误差平方和
大数据·人工智能·学习·算法·机器学习·数据挖掘
使者大牙2 小时前
【大语言模型学习笔记】第一篇:LLM大规模语言模型介绍
笔记·学习·语言模型
As977_3 小时前
前端学习Day12 CSS盒子的定位(相对定位篇“附练习”)
前端·css·学习
ajsbxi3 小时前
苍穹外卖学习记录
java·笔记·后端·学习·nginx·spring·servlet