多维数组的地址,通过指针引用多维数组详解

news2025/1/21 21:56:52

通过指针引用一维数组可以参考这篇文章:

通过指针引用数组的几种方法的原理和差异;以及利用指针引用数组元素的技巧_juechen333的博客-CSDN博客一个数组包含若干元素,每个数组元素都占用存储单元,所以他们都有相应的地址,所谓数组元素的指针就是数组元素的地址。https://blog.csdn.net/qq_57342311/article/details/129262120

目录

一、多维数组的地址

1.1引入

1.2地址举例说明

1.3地址类型详解

1.4实例说明

二、指向多维数组元素的指针变量

2.1指向数组元素的指针变量

2.2指向由 m 个元素组成的一维数组的指针变量

三、用指向数组元素的指针作函数参数

3.1举例说明1

3.2举例说明2


 指针变量可以指向一维数组中的元素,也可以指向多维数组中的元素;但在概念和使用方法上,多维数组的指针比一维数组的指针要复杂一些。

一、多维数组的地址

1.1引入

先定义一个二维数组

int a[3][4] = { {1,3,5,7},{9,11,13,15},{17,19,21,23} };

a 是二维数组名,a 数组包含 3 行,即 3 个行元素:a[0],a[1],a[2]。而每一个行元素又是一个一维数组,它包含 4 个元素(即 4 个列元素)。例如, a[0] 所代表的一维数组又包含 4 个元素:a[0][0],a[0][1],a[0][2],a[0][3]。可以认为二维数组是 “数组的数组”,即二维数组 a 是由 3 个一维数组所组成的。

从二维数组的角度来看,a 代表二维数组首元素的地址,现在的首元素不是一个简单的整型元素,而是由 4 个整型元素所组成的一维数组,因此 a 代表的是首行(即序号为 0 的行)的起始地址。a+1代表序号为 1 的行的起始地址。如果二维数组的首行的起始地址为2000,一个整型数据占 4 个字节,则 a+1 的值应该是 2000+4×4=2016 (因为第 0 行有 4 个整型数据)。a+1 指向a[1],或者说,a+1 的值是 a[1] 的起始地址。a+2 代表 a[2] 的起始地址,它的值是2032。

a[0],a[1],a[2] 既然是一维数组名,从前面已知,数组名代表数组首元素地址,因此 a[0] 代表一维数组 a[0] 中第 0 列元素的地址,即 &a[0][0]。同理,a[1] 的值是 &a[1][0],a[2] 的值是 &a[2][0]。

a[0] 是一维数组名,该一维数组中序号为 1 的元素的地址显然应该用 a[0]+1 来表示。此时 “a[0]+1” 中的 1 代表 1 个列元素的字节数,即4个字节。a[0] 的值是2000,a[0]+1 的值是2004(而不是2016)。这是因为现在是在一维数组范围内讨论问题,正如有一个一维数组x,x+1 是其第 1 个元素x[1] 的地址一样。a[0]+0,a[0]+1,a[0]+2,a[0]+3 分别是a[0][0],a[0][1],a[0][2],a[0][3] 元素的地址(即 &a[0][0],&a[0][1],&a[0][2],&[0][3])。

从上一篇文章可知,a[0] 和 *(a+0) 等价,a[1] 和 *(a+1) 等价,a[i] 和 *(a+i) 等价。

因此,a[0]+1和 *(a+0)+1 都是 &a[0][1](即地址都为2004)。a[1]+2 和 *(a+1)+2 的值都是 &a[1][2](即地址为2024)。请注意不要将*(a+1)+2 错写成 *(a+1+2),后者变成 *(a+3) 了,相当于 a[3] 。进一步分析,欲得到 a[0][1] 的值,用地址法怎么表示呢?既然 a[0]+1 和 *(a+0)+1 是 a[0][1] 的地址,那么,*(a[o]+1) 就是 a[0][1] 的值。同理,*(*(a+0)+1) 或 *(*a+1) 也是 a[0][1] 的值。*(a[i]+j) 或 *(*(a+i)+j)是 a[i][j] 的值。务请记住 *(a+i) 和 a[i] 是等价的。

注意:a[i] 从形式上看是 a 数组中序号为 i 的元素。如果a是一维数组名,则 a[i] 代表 a 数组序号为 i 的元素的存储单元。a[i] 是一个有确定地址的存储单元。但如果 a 是二维数组,则 a[i] 是一维数组名,它只是一个地址,并不代表一个存储单元,也不代表存储单元中的值(如同一维数组名只是一个指针常量一样)。a,a+i,a[i],* (a+i),* (a+i)+j,a[i]+j 都是地址。而 *(a[i]+j) 和 *(*(a+i)+j) 是二维数组元素 a[i][j] 的值。

1.2地址举例说明

表现形式含义
a二维数组名,指向行,即一维数组a[0],即0行的起始地址2000
a[0]一维数组名,指向列,即整型常量a[0][0],即0行0列元素的地址2000
*(a+0),*a0 行 0 列元素的地址2000
a+1,&a[1]第 1 行 (a[1]) 的起始地址2016
a[1],*(a+1)第 1 行 0 列元素 a[1][0] 的地址2016
a[1]+2,*(a+1)+2,&a[1][2]第 1 行 2 列元素 a[1][2] 的地址2024
*(a[1]+2),*(*(a+1)+2),a[1][2]第 1 行 2 列元素 a[1][2] 的值13

a+1 是二维数组 a 中序号为 1 的行的起始地址(序号从 0 起算),而 *(a+1) 并不是 a+1 单元的内容(值),因为 a+1 并不是一个数组元素的地址,也就谈不上存储单元的内容了。*(a+1) 就是 a[1],而a[1] 是一维数组名,所以也是地址,它指向a[1][0]。a[1]和 *(a+1) 都是二维数组元素 a[1][0] 的地址的不同的表示形式。

1.3地址类型详解

C 语言的地址信息中既包含位置信息(如内存编号2000),还包含它所指向的数据的类型信息。现在a[0] 是一维数组名,它是一维数组中起始元素的地址,a 是二维数组名,它是二维数组的首行起始地址,二者的纯地址是相同的;即2000,但它们的基类型不同,即它们指向的数据的类型不同,前者是整型数据,后者是一维数组。

如果用一个指针变量 pt 来指向一维数组,应当这样定义:

int (*pt)[4];

int(*pt)[4];表示 pt 指向由 4 个整型元素组成的一维数组,此时指针变量 pt 的基类型是由 4 个整型元素组成的一维数组。

a+1 与 a[0]+1 是不同的,a+1 是序号为 1 的行的起始地址,a+1 指向序号为 1 的行,而 *(a+1)或 a[1] 或 a[1]+0 都指向 1 行 0 列元素,二者地址虽相同,但指向的数据类型不同。a 和 a[0] 的值虽然相同(等于2000),但是由于指针的基类型不同,a 指向一维数组a[0],而 a[0] 指向列元素a[0][0]。因此,对不同的指针进行加 1 的运算,得到的结果是不同的。

注意:

二维数组名(如 a )是指向行(一维数组)的。因此 a+1 中的 “1” 代表一行中全部元素所占的字节数。一维数组名(如 a[0],a[1] )是指向列元素的。a[0]+1 中的 1 代表一个 a 元素所占的字节数。在指向行的指针前面加一个 *,就转换为指向列的指针。例如,a 和 a+1 是指向行的指针,在它们前面加一个 * 就是 *a和 *(a+1),它们就成为指向列的指针,分别指向 a 数组 0 行 0 列的元素和 1 行 0 列的元素。反之,在指向列的指针前面加 &,就成为指向行的指针。例如 a[0] 是指向 0 行 0 列元素的指针,在它前面加一个&,得 &a[0],由于 a[0] 与 *(a+0)等价,因此 &a[0] 与 &*a 等价,也就是与 a 等价,它指向二维数组的 0 行。

注意:

不要把 &a[i] 简单地理解为 a[i] 元素的存储单元的地址,因为并不存在 a[i] 这样一个实际的数据存储单元。它只是一种地址的计算方法,能得到第 i 行的起始地址,&a[i] 和 a[i] 的值是一样的,但它们的基类型是不同的。&a[i] 或 a+i 指向行,而 a[i] 或 *(a+i) 指向列。当列下标 j 为 0 时,&a[i] 和 a[i] (即 a[i]+j )值相等,即它们的纯地址相同,但应注意它们所指向的对象的类型是不同的,即指针的基类型是不同的。*(a+i) 只是 a[i] 的另一种表示形式,不要简单地认为 *(a+i) 是 “ a+i 所指单元中的内容 ” 。在一维数组中 a+i 所指的是一个数组元素的存储单元,在该单元中有具体值。而对二维数组,a+i 不是指向具体存储单元而是指向行(即指向一维数组)。在二维数组中,a+i,a[i],*(a+i),&a[i],&a[i][0] 的值相等,即它们都代表同一地址,但基类型不同。

1.4实例说明

#include<stdio.h>
int main()
{
	int a[3][4] = { {1,3,5,7},{9,11,13,15},{17,19,21,23} };
	printf("%p,%p\n", a, *a);			//0行起始地址和0行0列元素的地址
	printf("%p,%p\n", a[0], *(a + 0));	//0行0列元素的起始地址
	printf("%p,%p\n", &a[0], &a[0][0]);	//0行起始地址和0行0列元素的地址
	printf("%p,%p\n", a[1], a + 1);		//1行0列元素的地址和1行起始地址
	printf("%p,%p\n", &a[1][0], *(a + 1) + 0);	//1行0列元素的地址
	printf("%p,%p\n", a[2], *(a + 2));			//2行0列元素的地址
	printf("%p,%p\n", &a[2], a + 2);			//2行起始地址
	printf("%d,%d\n", a[1][0], *(*(a + 1) + 0));//1行0列元素的值
	printf("%d,%d\n", *a[2], *(*(a + 2) + 0));	//2行0列元素的值
}

程序分析:

数组 a 的起始地址是 00EFFC74 (16进制) 。上面是在 VS2019 环境下的一次运行记录。在不同的计算机、不同的编译环境、不同的时间运行以上程序时,由于分配内存情况不同,所显示的地址可能是不同的。但是上面显示的地址是有共同规律的,如上面显示 0 行起始地址和 0 行 0 列元素地址为 00EFFC74,前 3 行显示的地址是相同的。第 4,5 行是 1 行 0 列元素地址和 1 行起始地址,它的值应当比上面显示的 0 行起始地址和 0 行 0 列元素地址大 16 个字节(一行有 4 个元素,每个元素 4 个字节),00EFFC84 和 00EFFC74 之差是16。同样,第 6,7 行是 2 行 0 列元素地址和 ⒉行起始地址,它的值应当比 1 行起始地址和 1 行 0 列元素地址大 16 个字节,00EFFC94 和 00EFFC84 之差是16。最后两行显示的是 a[1][0] 和 a[2][0] 的值。

二、指向多维数组元素的指针变量

2.1指向数组元素的指针变量

实例:有一个3×4的二维数组,要求用指向元素的指针变量输出二维数组各元素的值。

#include<stdio.h>
int main()
{
	int a[3][4] = { 1,3,5,7,9,11,13,15,17,19,21,23 };
	int* p;		//p是int* 型的指针
	for (p = a[0]; p < a[0] + 12; p++)	//使p依次指向下一个元素
	{
		if (p != a[0] && ((p - a[0]) % 4 == 0)) printf("\n");	//p移动4次后换行
		printf("%4d ", *p);		//输出p所指向元素的值
	}
}

运行结果:

程序分析:

p 是一个 int* 型(指向整型数据)的指针变量,它可以指向一般的整型变量,也可以指向整型的数组元素。每次使 p 值加 1,使 p 指向下一元素,比较简单。如果要输出某个指定的数值元素(例如 a[1][2] ),则应事先计算该元素在数组中的相对位置(即相对于数组起始位置的相对位移量)。计算 a[i][j] 在数组中的相对位置的计算公式为:i * m + j

其中,m 为二维数组的列数(二维数组大小为 n×m )。例如,对上述 3×4 的二维数组,它的 2 行 3 列元素 a[2][3] 对 a[0][0] 的相对位移量为 2×4+3=11 元素。如果一个元素占 4 个字节,则 a[2][3] 对 a[0][0]的地址差为 11×4=44 字节。若开始时指针变量 p 指向 a[0][0],a[i][j] 的地址为 “&a[0][0]+(i*m+j)” 或 “p+(i*m+j)” 。a[2][3] 的地址是 (p+2*4+3),即 (p+11)。a[2][3] 的值为 *(p+11)。

从图中可以看到在 a[i][j] 元素之前有 i 行元素(每行有 m 个元素),在 a[i][j] 所在行,a[i][j] 的前面还有 j 个元素,因此 a[i][j] 之前共有 i×m+j 个元素。例如,a[2][3]的前面有两行,共 2×4=8 个元素,在它本行内还有 3 个元素在它前面,故共有 8+3=11 个元素在它之前,可用 p+11 表示其相对位置。

2.2指向由 m 个元素组成的一维数组的指针变量

上面实例的指针变量 p 是用 “int* p;” 定义的,它是指向整型数据的,p+1 所指向的元素是 p 所指向的列元素的下一元素(按在内存中存储的下一个整型元素)。可以改用另一方法,使 p 不是指向整型变量,而是指向一个包含 m 个元素的一维数组。这时,如果 p 先指向 a[0](即 p=&a[0] ),则 p+1 不是指向 a[0][1],而是指向 a[1],p 的增值以一维数组的长度为单位。

实例:输出二维数组任一行任一列元素的值

#include<stdio.h>
int main()
{
	int a[3][4] = { 1,3,5,7,9,11,13,15,17,19,21,23 };
	int(*p)[4] = a;		//p是指向包含4个整型元素的一维数组的指针变量
	int i, j;
	printf("请输入要输出元素的行列号:");
	scanf_s("%d,%d", &i, &j);		//获得所需要的行列号
	printf("a[%d,%d] = %d", i, j, *(*(p + i) + j));	//输出a[i][j]的值
}

运行结果:

程序分析:

程序第 4 行中 “int(*p)[4];” 表示定义 p 为一个指针变量,它指向包含 4 个整型元素的一维数组。注意,*p 两侧的括号不可缺少,如果写成 *p[4],由于方括号 [ ] 运算级别高,因此 p 先与 [4] 结合,p[4] 是定义数组的形式,然后再与前面的 * 结合,*p[4] 就是指针数组。

int(*p)[4]详解:

表示 (*p) 有 4 个元素,每个元素为整型。也就是 p 所指的对象是有 4 个整型元素的数组,即 p 是指向一维数组的指针。应该记住,此时 p 只能指向一个包含 4 个元素的一维数组,不能指向一维数组中的某一元素。p 的值是该一维数组的起始地址。虽然这个地址(指纯地址)与该一维数组首元素的地址相同,但它们的基类型是不同的。不要混淆。

实例:

#include<stdio.h>
int main()
{
	int a[4] = { 1,3,5,7 };
	int(*p)[4];			//p是指向包含4个整型元素的一维数组的指针变量
	p = &a;				//使p指向一维数组
	printf("%d", (*p)[3]);	//输出a[3]
}

注意第 6 行不应写成 “p=a;” ,因为这样写表示 p 的值是 &a[0],指向首元素a[0]。“p=&a;” 表示 p 指向一维数组(行),(*p)[3] 是 p 所指向的行中列为 3 的元素。

注意:

要注意指针变量的类型,从 “int (*p)[4];” 可以看到,p 的类型不是 int* 型,而是 int(*)[4] 型,p 被定义为指向一维整型数组的指针变量,一维数组有 4 个元素,因此 p 的基类型是一维数组,其长度是16字节。

三、用指向数组元素的指针作函数参数

一维数组名可以作为函数参数,多维数组名也可作函数参数。用指针变量作形参,以接受实参数组名传递来的地址。可以有两种方法:

①用指向变量的指针变量

②用指向一维数组的指针变量

3.1举例说明1

实例:3 个学生,4 门课,计算总平均分数以及第 n 个学生的成绩

#include<stdio.h>
int main()
{
	void average(float* p, int n);		//average函数声明
	void search(float(*p)[4], int n);	//search函数声明
	float score[3][4] = { {66,85,79,94},{88,98,76,84},{98,88,72,83} };
	average(*score, 12);	//求平均分
	search(score, 2);		//求序号为2的学生的成绩
	return 0;
}

void average(float* p, int n)
{
	float* q = p;	//指针变量为int*型
	float sum = 0, avg;
	for (; q < p + n; q++)
		sum += *q;		//求和
	avg = sum / n;		//求平均值
	printf("average = %5.2f\n", avg);
}

void search(float(*p)[4], int n)
{
	printf("第%d位同学的成绩为:", n);
	for (int i = 0; i < 4; i++)
		printf("%5.2f ", *(*(p + n) + i));	//指针变量为int (*p)[4]
	printf("\n");
}

运行结果:

程序分析:

在 main 函数中,先调用 average 函数以求总平均值。在函数 average 中形参 p 被声明为 float* 类型(指向 float 型变量)的指针变量。它的基类型为 float 型,实参用 *score,即 score[0],也就是&score[0][0],即 score[0][0] 的地址。把 score[0][0] 的地址传给 p,使 p 指向score[0][0]。然后在average 函数中使 p 先后指向二维数组的各个元素,p 每加 1 就改为指向 score 数组的下一个元素。形参 n 代表需要求平均值的元素的个数,实参 12 表示要求 12 个元素值的平均值。

函数search的形参 p 的类型是 float(*)[4],它不是指向整型变量的指针变量,而是指向包含 4 个元素的一维数组的指针变量。函数调用开始时,将实参 score 的值(代表该数组 0 行起始地址)传给p,使 p 也指向 score[0]。p+n 是 score[n] 的起始地址,*(p+n)+i 是score [n][i] 的地址,*(*(p+n)+i) 是 score[n][i] 的值。现在实参传给形参 n 的值是 2,即想找序号为 2 的学生的成绩。

调用 search 函数时,实参是 score(二维数组名,代表该数组中 0 行起始地址)传给 p,使 p 也指向score[0]。p+n 是 score[n] 的起始地址,*(p+n)+i 是 score[n][i] 的地址,*(*(p+n)+i) 是 score[n][i] 的值。现在 n=2,i 由 0 变到 3,for 循环输出 score[2][0] 到 score[2][3] 的值。

注意:

实参与形参如果是指针类型,应当注意它们的基类型必须一致。不应把 int* 型的指针(即数组元素的地址)传给 int(*)[4] 型(指向一维数组)的指针变量,反之亦然。

3.2举例说明2

在3.1实例的基础上,查找有一门以上课程不及格的学生,输出他们的全部成绩

#include<stdio.h>
int main()
{
	void search(float(*p)[4], int n);	//search函数声明
	float score[3][4] = { {66,85,55,94},{88,98,43,84},{98,88,72,83} };
	search(score, 3);		//查找存在不及格同学的成绩
	return 0;
}

void search(float(*p)[4], int n)
{
	for (int i = 0; i < n; i++)		//遍历每位同学
	{
		int  flag = 0;	//默认此同学不存在挂科情况
		for (int j = 0; j < 4; j++)
			if (*(*(p + i) + j) < 60)
			{
				flag = 1;	//遍历一个同学的每科成绩,如果存在小于60分的,将flag标记为1
			}
		if (flag == 1)	//如果flag==1,则此同学存在挂科,输出他的所有成绩
		{
			printf("第%d位同学存在挂科现象:", i + 1);
			for (int j = 0; j < 4; j++)
				printf("%5.2f ", *(*(p + i) + j));
			printf("\n");
		}
	}
}

运行结果:

程序分析:

实参 score 和形参 p 的类型是相同的。在调用 search 函数时,p 得到实参 score 的值,即 score[0] 的起始地址,也就是说 p 也指向 score 数组的第 1 行。然后 p 先后指向各行(每行包括该学生几门课的成绩)。p+i 是 score 数组第 i 行的起始地址,*(p+i) 是 score[i][0] 元素的地址,即 &score[i][0],*(p+i)+j 是 score[i][j] 的地址,即 &score[i][j] ,search函数中的 *(*(p+i)+j)就是 score[i][j]。先后检查各学生每门课的成绩,如有不及格的就记录下来。

在函数 search 中,变量 flag 用来表示有无不及格的课程。若flag 的值为 1 表示有不及格的课程,若 flag 的值为 0 表示没有不及格的课程。开始时先使 flag=0,若发现某一学生有一门不及格,就使 flag 变为 1。最后用 if 语句检查 flag,如为 1,则表示该学生有不及格的记录,输出该学生全部课程成绩。变量 i 代表学生号,j 代表课程号。score[i][j] 是序号为 i 的学生第 i 门课的成绩。

通过指针变量存取数组元素速度快,程序简明。用指针变量作形参,所处理的数组大小可以改变。因此数组与指针常常是紧密联系的,使用熟练的话可以使程序质量提高,编写程序方便灵活。

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

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

相关文章

《Ansible模块篇:debug模块详解》

一、简介 平时我们在使用ansible执行playbook时&#xff0c;经常可能会遇到一些错误&#xff0c;有的时候不知道问题在哪里 &#xff0c;这个时候可以使用-vvv参数打印出来详细信息&#xff0c;不过很多时候-vvv参数里很多东西并不是我们想要的&#xff0c;这时候就可以使用官方…

python第四天作业~函数练习

目录 作业4、判断以下哪些不能作为标识符 A、a B、&#xffe5;a C、_12 D、$a12 E、false F、False 作业5&#xff1a; 输入数&#xff0c;判断这个数是否是质数&#xff08;要求使用函数 for循环&#xff09; 作业6&#xff1a;求50~150之间的质数是…

ReentrantLock 源码解读

一、ReentrantLock ReentrantLock 是 java JUC 中的一个可重入锁&#xff0c;在上篇文章讲解 AQS 源码的时候提到 ReentrantLock 锁是基于 AQS 实现的&#xff0c;那是如何使用的 AQS 呢&#xff0c;本篇文章一起带大家看下 ReentrantLock 的源码。 在 AQS 中&#xff0c;如果…

linux安装influxdb-rpmyum方式

一、influxdb的安装InfluxDB简介时序数据库InfluxDB版是一款专门处理高写入和查询负载的时序数据库&#xff0c;用于存储大规模的时序数据并进行实时分析&#xff0c;包括来自DevOps监控、应用指标和IoT传感器上的数据主要特点&#xff1a;专为时间序列数据量身订造高性能数据存…

uniapp——ucharts的使用

ucharts是一个类似于echarts的图表框架 引入 在Hbuilder的插件下载仓库中搜索ucharts &#xff01;&#xff01;值得一提的是&#xff0c;Hbuilder的版本必须是大于3.1.0的&#xff01;&#xff01; 这样就导入到了项目的uni_modules里了 ucharts下载完成后&#xff0c;可以…

Java——异常机制

前言 随着对java的不断深入学习&#xff0c;在对语法以及编程思想有了一定的了解之后&#xff0c;在编程的过程中有可能会因为用户的输入不正确或者逻辑错误而出现异常或者错误&#xff0c;因此如何去捕捉与避免不应该出现的异常或者错误就变得十分重要。本文就介绍了java的异…

C++实现的二叉树创建和遍历,超入门邻家小女也懂了

目录 二叉树 特点 性质 二叉树的创建 声明 创建 -> 成员运算符 批量创建 二叉树的遍历 先序遍历 中序遍历 后序遍历 层序遍历 树的相关术语 特殊二叉树 满二叉树 完全二叉树 二叉树 树&#xff08;Tree&#xff09;是n(n≥0)个节点的有限集。在任意一棵…

Idea工具单工程使用卡顿设置

一、开启Idea内存占比指示 1.1、勾选Idea 内存指示器 1.2、右下角展示当前项目&#xff1a;内存/总内存&#xff08;726/1024MB&#xff09; ​​​​​​​ 二、修改Idea自定义内存 2.1、打开help - Edit Custom VM Options 2.2、自定义虚拟内存设置 -Xms1024m // 初始内存分…

【前端八股文】浏览器系列:浏览器渲染、前端路由、前端缓存(HTTP缓存)、缓存存储(HTTP缓存存储、本地存储)

文章目录渲染步骤DOM树与render树回流与重绘前端路由hash模式history模式两种模式对比前端缓存HTTP缓存强缓存协商缓存一般哪些文件对应哪些缓存HTTP缓存总结缓存存储HTTP缓存存储本地存储参考本系列目录&#xff1a;【前端八股文】目录总结 是以《代码随想录》八股文为主的笔记…

TTS | 语音合成论文概述

综述系列2021_A Survey on Neural Speech Synthesis论文&#xff1a;2106.15561.pdf (arxiv.org)论文从两个方面对神经语音合成领域的发展现状进行了梳理总结&#xff08;逻辑框架如图1所示&#xff09;&#xff1a;核心模块&#xff1a;分别从文本分析&#xff08;textanalysi…

【Linux内核一】在Linux系统下网口数据收发包的具体流向是什么?

在TCP/IP网络分层模型里&#xff0c;整个协议栈被分成了物理层、链路层、网络层&#xff0c;传输层和应用层。物理层对应的是网卡和网线&#xff0c;应用层对应的是我们常见的Nginx&#xff0c;FTP等等各种应用。Linux实现的是链路层、网络层和传输层这三层。 在Linux内核实现中…

Linux(Centos)安装Minio集群

目录1&#xff1a;简介2&#xff1a;功能与集成3&#xff1a;架构4&#xff1a;搭建集群4.1&#xff1a;挂载磁盘4.1.1&#xff1a;要求4.1.2&#xff1a;创建挂载目录4.1.3&#xff1a;注意&#xff1a;需要将新建的目录挂在到对应的磁盘下,磁盘不挂载好&#xff0c;集群启动会…

Tomcat 并发达太大导致系统崩溃解决方案

当 Tomcat 并发达太大导致系统崩溃时&#xff0c;可以通过以下几个步骤来解决这个问题&#xff1a; 1、分析原因 首先需要分析系统崩溃的原因&#xff0c;是因为Tomcat的性能瓶颈还是因为代码的Bug&#xff0c;或者是系统资源不足等等。 2、优化代码 如果是代码的问题&…

Unity Lighting -- 为场景烘焙lightmap

烘焙光照是什么&#xff1f; Unity中有两种不同的光照方式&#xff1a;实时光照和烘焙光照。 实时光照 Unity会在运行时实时计算光照。实时光源每一帧都会进行计算&#xff0c;这意味着它们对于场景中移动的角色和物体的响应性非常好&#xff0c;但它的开销也很大。 烘焙光照…

DJ1-3 操作系统引论

目录 操作系统的结构设计 1. 无结构操作系统 2. 模块化 OS 结构 3. 分层式 OS 结构 4. 微内核 OS 结构 操作系统的结构设计 操作系统是一个大型系统软件&#xff0c;其结构已经历了四代的变革&#xff1a; 第一代 OS 是无结构第二代 OS 采用模块式结构第三代是层次式结构…

韩国绿芯1~16通道触摸芯片型号推荐

随着技术的发展&#xff0c;触摸感应技术正日益受到更多关注和应用&#xff0c;目前实现触摸感应的方式主要有两种&#xff0c;一种是电阻式&#xff0c;另一种是电容式。电容式触摸具有感应灵敏、功耗低、寿命长等特点&#xff0c;因此逐步取代电阻式触摸&#xff0c;成为当前…

炫龙游戏本Win10系统总是蓝屏崩溃怎么办?

炫龙游戏本Win10系统总是蓝屏崩溃怎么办&#xff1f;有用户使用的炫龙游戏本最近总是在运行的过程中出现自动蓝屏的情况&#xff0c;有的时候自己还在操作电脑&#xff0c;而屏幕却蓝屏了&#xff0c;导致自己的工作被中断了。那么这个情况要怎么去进行修复呢&#xff1f;来看看…

关于centos虚拟机设置固定ip、设置后无法上网、宿主机无法ping通的解决方案

一.centos设置固定ip 1.虚拟机设置 编辑→虚拟网络编辑器&#xff0c;选择NAT模式。 2.NAT设置&#xff0c;设置网关&#xff08;前面三个需要与你设置的静态虚拟机ip一致&#xff08;我的是192.168.2.40&#xff09;&#xff09;。 3.虚拟机设置为NAT模式 4.设置虚拟…

数据结构2——线性表1:基本概念

1、线性表的概念 线性表是具有相同特性的数据元素的一个有限序列。就像用细线串珠子&#xff0c;一个接着一个串起来。 2、线性表的逻辑特征 ① 在非空的线性表中有且只有一个开始结点a1&#xff0c;他没有直接前驱&#xff0c;而仅有一个直接后继a2&#xff0c;相当于a1是…

关于JS中this对象指向问题总结

一、前言 关于JS中this对象指向问题&#xff0c;相信做过项目的小伙伴多多少少都会遇到过&#xff0c;明明感觉代码写的没问题&#xff0c;可是运行的时候&#xff0c;就会报错&#xff0c;比如报错 xxx is not a function。 我最近也遇到了&#xff0c;百度学习了不少前辈对于…