如果没有看过第一部分的话,推荐先看第一部分,然后再来看第二部分~~
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int *p = &arr[0];
&arr[0]
的方式拿到了数组第一个元素的地址,但是其实数组名本来就是地址,而且 是数组首元素的地址,我们来做个测试。#include
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("&arr[0] = %p\n", &arr[0]);
printf("arr = %p\n", arr);
return 0;
}
4
,那是不是呢?#include
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("%d\n", sizeof(arr));
return 0;
}
sizeof(数组名),sizeof中单独放数组名,这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节
&数组名, 这里的数组名表示整个数组,取出的是整个数组的地址(整个数组的地址和数组首元素的地址是有区别的)
这时有好奇的同学,再试一下这个代码:
#include
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("&arr[0] = %p\n", &arr[0]);
printf("arr = %p\n", arr);
printf("&arr = %p\n", &arr);
return 0;
}
数组的地址和数组首元素的地址的值是一模一样的,那它们有什么区别呢,接下来继续看~~
我们再来看下面这段代码~~
#include
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("&arr[0] = %p\n", &arr[0]);
printf("&arr[0]+1 = %p\n", &arr[0] + 1);
printf("arr = %p\n", arr);
printf("arr+1 = %p\n", arr + 1);
printf("&arr = %p\n", &arr);
printf("&arr+1 = %p\n", &arr + 1);
return 0;
}
&arr
和 &arr+1
相差40个字节,这就是因为&arr
是数组的地址,+1
操作是跳过整个数组的。&arr[0]
它的类型是int*
,而&arr加一加了40个字节,它的类型是什么呢?我们这里留个悬念,后面都会将~~有了前面知识的支持,再结合数组的特点,我们就可以很方便的使用指针访问数组了。
#include
int main()
{
int arr[10] = { 0 };
//输入
int i = 0;
int sz = sizeof(arr) / sizeof(arr[0]);
//输入
int* p = arr;
for (i = 0; i < sz; i++)
{
scanf("%d", p + i);
//scanf("%d", arr+i);//也可以这样写
}
//输出
for (i = 0; i < sz; i++)
{
printf("%d ", *(p + i));
}
return 0;
}
我们定义了一个整型数组 arr 和一个指向该数组的指针 p,其中,表示数组元素的方法有两种,一种是 *(p + i),另一种是 arr[i]。
这个代码搞明白后,我们再试一下,如果我们再分析一下,数组名arr是数组首元素的地址,可以赋值给p,其实数组名arr和p在这里是等价的。那我们可以使用arr[i]可以访问数组的元素,那p[i]是否也可以访问数组呢?
#include
int main()
{
int arr[10] = { 0 };
//输入
int i = 0;
int sz = sizeof(arr) / sizeof(arr[0]);
//输入
int* p = arr;
for (i = 0; i < sz; i++)
{
scanf("%d", p + i);
//scanf("%d", arr+i);//也可以这样写
}
//输出
for (i = 0; i < sz; i++)
{
printf("%d ", p[i]);
}
return 0;
}
(p+i)
换成p[i]
也是能够正常打印的,所以本质上p[i]
是等价于 *(p+i)
。arr[i]
应该等价于*(arr+i)
,数组元素的访问在编译器处理的时候,也是转换成首元素的地址+偏移量求出元素的地址,然后解引用来访问的。arr[i]
== *(arr+i)
== *(i+arr)
== i[arr]
是不是也可以这样,照样也能访问~~int main() {
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
int* p = arr;
int i = 0;
int sz = sizeof(arr) / sizeof(arr[0]);
for (i = 0; i < sz; i++) {
printf("%p ======== %p\n", p + i, &arr[i]);
}
return 0;
}
#include
void test(int arr[])
{
int sz2 = sizeof(arr) / sizeof(arr[0]);
printf("sz2 = %d\n", sz2);
}
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
int sz1 = sizeof(arr) / sizeof(arr[0]);
printf("sz1 = %d\n", sz1);
test(arr);
return 0;
}
void test1(int arr[])//参数写成数组形式,本质上还是指针
{
printf("%d\n", sizeof(arr));
}
void test2(int* arr)//参数写成指针形式
{
printf("%d\n", sizeof(arr));//计算一个指针变量的大小
}
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
test1(arr);
test2(arr);
return 0;
}
总结: 一维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式。
接下来我们就学习一下这个冒泡排序,主要学习两个内容~~
- 学习冒泡排序
- 学习数组传参
int main() {
int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
//进行排序
return 0;
}
int main() {
int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
//进行排序
int sz = sizeof(arr) / sizeof(arr[0]);
sort(arr,sz);
return 0;
}
代码如下:
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;
}
}
}
}
void print(int arr[], int sz) {
int i = 0;
for (i = 0; i < sz; i++) {
printf("%d ", arr[i]);
}
}
int main() {
int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
//进行排序
int sz = sizeof(arr) / sizeof(arr[0]);
sort(arr,sz);
print(arr, sz);
return 0;
}
int arr[] = { 9,0,1,2,3,4,5,6,7,8 };
void sort(int arr[], int sz) {
//确定冒泡排序的趟数~~
int i = 0;
for (i = 0; i < sz - 1; i++) {
//一趟冒泡排序
int j = 0;
int flag = 1;//假设数组是有序的
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;
flag = 0;//不是有序
}
}
if (flag = 1) {
break;
}
}
}
#include
int main() {
int a = 10;
int* pa = &a;
int** ppa = &pa;
return 0;
}
int*** pppa = &ppa
对于二级指针的运算有:
*ppa
通过对ppa
中的地址进行解引用,这样找到的是pa
, *ppa
其实访问的就是
int b = 20;
*ppa = &b;//等价于 pa = &b;
**ppa
先通过*ppa
找到pa
,然后对pa
进行解引用操作: *pa
,那找到的是a
.
**ppa = 30;
//等价于*pa = 30;
//等价于a = 30;
我们类比一下:
整形数组和字符数组:
指针数组的每个元素都是用来存放地址(指针)的。
如下图:
指针数组的每个元素是地址,又可以指向一块区域。
比如:
int main() {
int a = 1;
int b = 2;
int c = 3;
int d = 4;
int e = 5;
int* parr[5] = { &a,&b,&c,&d,&e };
return 0;
}
int main() {
int a = 1;
int b = 2;
int c = 3;
int d = 4;
int e = 5;
int* parr[5] = { &a,&b,&c,&d,&e };
int i = 0;
for (i = 0; i < 5; i++) {
printf("%d ", *(parr[i]));
}
return 0;
}
parr[i]
找到了每个元素的地址,然后解引用,就找到了,便可以打印出来~~#include
int main()
{
int arr1[] = { 1,2,3,4,5 };
int arr2[] = { 2,3,4,5,6 };
int arr3[] = { 3,4,5,6,7 };
//数组名是数组首元素的地址,类型是int*的,就可以存放在parr数组中
int* parr[3] = { arr1, arr2, arr3 };
int i = 0;
int j = 0;
for (i = 0; i < 3; i++)
{
for (j = 0; j < 5; j++)
{
printf("%d ", parr[i][j]);
}
printf("\n");
}
return 0;
}
parr[i]
是访问parr数组的元素,parr[i]
找到的数组元素指向了整型一维数组,parr[i][j]
就是整型一维数组中的元素。好了,指针的第二部分就到这里就结束了~~
如果有什么问题可以私信我或者评论里交流~~
感谢大家的收看,希望我的文章可以帮助到正在阅读的你