C语言教程——数组(2)

目录

系列文章目录

前言

4、数组作为函数参数

4.1冒泡函数的错误设计

4.2数组名是什么?

总结


前言

我们知道一维数组是连续存放的,随着数组下标的增长,地址是由低到高依次存放的,二维数组,也是在内存里面是连续存放的,而且要检查数组是否越界。


4、数组作为函数参数

往往我们在写代码的时候,会将数组作为参数传入函数,比如:我要写一个冒泡函数(这里要讲算法思想)函数,将一个整形数组排序。

排序的思想和方法有很多,例如冒泡排序,选择排序,插入排序,快速排序等等。

冒泡排序的思想就是两辆元素进行比较,有可能的话需要交换。

比如我们要对一组整形数据进行排序,要排成升序:

9 8 7 6 5 4 3 2 1 0

如果我们有十个数字,那么会需要9趟则可以完全排序,如果有n个数字,那么就需要n-1趟。两个一比较,就可以通过下标来访问元素。

cpp 复制代码
#include <stdio.h>
//排序:
void Sort(int arr[],int sz)
{
	
	int i = 0;
	//趟数
	for (i = 0; i < sz - 1; i++)
	{
		//一趟冒泡排序,决定了一趟排序进行了多少对比较
		int j = 0;
		for (j = 0; j < sz-1-i; j++)
		{
			if (arr[j] > arr[j + 1])
			{
				int tmp = arr[j];
				arr[j] = arr[j + 1];
				arr[j + 1] = tmp;
			}
		}
	}
}

int main()
{
	//整形数据
	int arr[] = { 3,1,4,2,9,8,6,5,7,0 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	//写一个函数对数组进行排序
	Sort(arr,sz);
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

以上代码就是升序排序1-0。

4.1冒泡函数的错误设计

再看看下面的代码:

cpp 复制代码
#include <stdio.h>
//排序:
void Sort(int arr[])
{
	int sz = sizeof(arr) / sizeof(arr[0]);
	int i = 0;
	//趟数
	for (i = 0; i < sz - 1; i++)
	{
		//一趟冒泡排序,决定了一趟排序进行了多少对比较
		int j = 0;
		for (j = 0; j < sz-1-i; j++)
		{
			if (arr[j] > arr[j + 1])
			{
				int tmp = arr[j];
				arr[j] = arr[j + 1];
				arr[j + 1] = tmp;
			}
		}
	}
}

int main()
{
	//整形数据
	int arr[] = { 3,1,4,2,9,8,6,5,7,0 };
	
	//写一个函数对数组进行排序
	Sort(arr);
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

这是一个错误的冒泡函数编写格式,这里面打印出的数组并没有发生排序,而是原封不动的给你打印出来。我们可以先放着这个问题,来进行下面的介绍和学习,相信看完和理解后就会知道为什么数组没有排序了。

4.2数组名是什么?

我们先定义一个里面数字都为0的数组:

cpp 复制代码
int main( )
{
    int arr[10]={0};
    printf("%p\n",arr);
    printf("%p\n",&arr[0]);


    return 0;
}

这里输出的是一个地址,每次运行都不一样,因为分配的内存每次编译是不一样的。

第一行输出的是数组名的地址,第二行输出的是数组首元素的地址,运行出来就可以发现,它们两个都是一个地址,所以得出结论,数组名是数组首元素的地址。

但是有两个例外

1、sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节。

2、&数组名,这里的数组名是表示整个数组,取地址数组名取出的是数组的地址。

我们可以通过以下代码更能直观的看出来:

cpp 复制代码
int main( )
{
    int arr[10]={0};
    printf("%p\n",arr);
    printf("%p\n",arr+1);

    printf("%p\n",&arr[0]);
    printf("%p\n",&arr[0]+1);

    printf("%p\n",&arr);
    printf("%p\n",&arr+1);

    return 0;
}

首先定义一个里面有10个0的数组,通过打印数组名和&数组名来分析。

我们知道数组名的话,是代表数组的首元素,所以前两个打印出来的一个是首元素的地址,另一个是第二个元素的地址,+1与首元素差了4个字节;中间的&arr[0],用下标来访问,也是一样的首元素的地址,第二个与它也差4个字节;最后两行输出,&arr虽然说是代表整个数组,但是整个数组是从首元素开始的,当&arr+1的时候,它们之间就差了40个字节,因为这个数组元素为十个且为整形,4*10=40,由此可以说明&arr代表了一整个数组。

其实指针+1,到底跳过几个字节,是和指针类型有关系的,后面会进行讲解。

了解这些后就可以回到上一篇文章去分析冒泡排序的函数。

数组传参的时候形参写成数组还是指针都是可以的。

我们可以知道函数调用的时候,参数传入的时候是对实参的一份临时拷贝,所以数组本质上传参是数组首元素的地址而已。

下一篇讲三子棋的实现,数组的实例。


总结

数组名实际上就是一个地址,传入参数就相当于传入的是数组的首元素的地址,当用到取地址+数组名的时候才是整个数组。

相关推荐
fmdpenny31 分钟前
Vue3初学之商品的增,删,改功能
开发语言·javascript·vue.js
涛ing1 小时前
21. C语言 `typedef`:类型重命名
linux·c语言·开发语言·c++·vscode·算法·visual studio
等一场春雨1 小时前
Java设计模式 十四 行为型模式 (Behavioral Patterns)
java·开发语言·设计模式
黄金小码农1 小时前
C语言二级 2025/1/20 周一
c语言·开发语言·算法
萧若岚2 小时前
Elixir语言的Web开发
开发语言·后端·golang
wave_sky2 小时前
解决使用code命令时的bash: code: command not found问题
开发语言·bash
水银嘻嘻2 小时前
【Mac】Python相关知识经验
开发语言·python·macos
ac-er88882 小时前
Yii框架中的多语言支持:如何实现国际化
android·开发语言·php
我的运维人生2 小时前
Java并发编程深度解析:从理论到实践
java·开发语言·python·运维开发·技术共享
大乔乔布斯3 小时前
JRE、JVM 和 JDK 的区别
java·开发语言·jvm