C语言--指针进阶1

news2025/1/10 20:56:52

目录

  • 回顾
  • 字符指针
  • 指针数组
  • 数组指针
    • &数组名和数组名的区别
    • 数组指针的使用
      • 指针作为形参
    • 练习
  • 数组参数、指针参数
    • 一维数组传参
    • 二维数组传参
    • 一级指针传参
    • 二级指针传参

回顾

指针的内容,我们在初级阶段已经有所涉及了,我们先来复习一下

  1. 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
  2. 指针的大小是固定的4/8个字节(32位平台/64位平台)。
  3. 指针是有类型,指针的类型决定了指针的±整数的步长,指针解引用操作的时候的权限。
  4. 指针的运算。(指针+整数,指针减整数,指针减指针等)。

下面,我们来继续探讨指针的高级主题。

字符指针

在指针的类型中我们知道有一种指针类型为字符指针 char*
看下面这段代码:

int main()
{
char ch = 'w';
char *pc = &ch;

return 0;
}

这段代码我们很容易理解指针pc存放的就是字符w的地址
我们再来看一段代码:

int main()
{
char c = 'abcdef';
char *p = &c;

return 0;
}

这时候p里面存放的是什么呢?是字符串abcdef的地址吗?其实不是的。
这里的指针p存放的其实是字符串首元素的地址。我们可以把这个字符串类比成一个数组,同样都是存放的首元素地址,那么二者又有什么不同呢?

char c="abcdef"              char arr[]="abcdef"
char *p=&c                   char *p=arr

区别在于,第一种表示的是常量字符串,是不能被修改的,而第二种的p指向的是数组的首元素,arr数组是可以被修改的。(如果我们令*p=‘w’,第一种情况程序就会崩掉,而第二种字符串就会被修改为wbcdef。为了防止第一种危险情况,我们可以加上const修饰)

const char c="abcdef"

我们来看一道剑指offer的题:

#include <stdio.h>
int main()
{
char str1[] = "hello bit.";
char str2[] = "hello bit.";
const char *str3 = "hello bit.";
const char *str4 = "hello bit.";
if(str1 ==str2)
printf("str1 and str2 are same\n");
else
printf("str1 and str2 are not same\n");
if(str3 ==str4)
printf("str3 and str4 are same\n");
else
printf("str3 and str4 are not same\n");
return 0;
}

运行结果如图1:
图1
下面来解释原因:
str1和str2由于是两个完全不同的数组,所以所开辟的空间肯定不相同,首元素的地址肯定也不相同。而·str3和str4所指向的都是h的地址,所以两者存放的值肯定也相同

指针数组

在前面的章节中我们讲过什么是指针数组,类比整型数组,整型数组存放的都是整型,那么指针数组存放的都是指针
举几个例子:

int* arr1[5]--整形指针数组
char* arr2[3]--字符指针数组

注意写法和接下来讲的数组指针的区别。

数组指针

数组指针本质上是一种指针,是一种指向数组的指针
我们来看下面这两行代码

int *p1[10];
int (*p2)[10];

p1,p2分别表示什么呢?
p1是指针数组,而p2是数组指针,为什么 p2加了一个括号就变成指针了呢,这是因为[ ]的优先级要高于号的,所以必须加上()来保证p先和结合*

&数组名和数组名的区别

我们常常说数组名指的是数组的首元素地址,那么取地址数组名又表示什么呢,我们放在编译器里来探究一下。

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

	printf("%p\n", &arr);
	return 0;
}

打印出来的结果如图2
图2
我们发现,三种表示方式的结果是一样的。
我们先来总结一下数组名的含义:
在绝大部分情况下,数组名表示的是数组首元素的地址(如上述的代码所演示的情况)
但是也有两个例外
1,sizeof(arr) – sizeof内部单独放一个数组名的时候,数组名表示的是整个数组,计算得到的是数组总大小。
2,sizeof(arr) &arr – 这里的数字组表示的是整个数组,去除的是整个数组的地址,从地址的值来讲和数组首元素的地址是一样的,但是含义不一样。

那么这个含义不一样具体体现在哪里呢?
我们来对指针进行一些运算

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;
}

运行结果如图3
图3

我们来对运行结果分析一下,如图4
图4

数组指针的使用

我们可以通过指针来访问数组

int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int i = 0;
	int sz = sizeof(arr) / sizeof(arr[0]);
	int* p = arr;
	for(i = 0; i < sz; i++)
	{
		//printf("%d", arr[i]);
		printf("%d", (*p+i));
	}
	return 0;
}

指针作为形参

我们先来看常规方法

void print(int arr[3][5], int r, int c)
{
	int i = 0;
	for (i = 0; i < r; i++)
	{
		int j = 0;
		for (j = 0;j < c; j++)
		{
			printf("%d", arr[i][j]);
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { 1,2,3,4,5, 2,3,4,5,6, 3,4,5,6,7 };
	print(arr, 3, 5);
	return 0;
}

图5
打印结果如图5.
那么我们把形参的部分写成指针又该怎么写呢?
首先我们要知道二维数组的数组名也表示首元素地址,即第一行的地址,我们可以把二维数组的每一行看作一个整体,作为一个元素,相当于一维数组是二维数组的数组,所以首元素就是第一行,所以在函数传参的时候,传的是首元素地址,代码如下

void print(int (* arr)[5], int r, int c)
{
	int i = 0;
	for (i = 0; i < r; i++)
	{
		int j = 0;
		for (j = 0;j < c; j++)
		{
			printf("%d", *(*(arr + i) + j));
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { 1,2,3,4,5, 2,3,4,5,6, 3,4,5,6,7 };
	print(arr, 3, 5);
	return 0;
}

我们对 ((arr+i)+j) 分析一下,如图6
图6

练习

我们来看下面这种写法表示什么

int (*parr3[10])[5];

分析如下:
parr3是数组
而这个parr3数组中存放的是指针,该指针指向的又是数组
。用文字可能不太好理解,我们用=来看图分析,如图7。

图7

数组参数、指针参数

在写代码的时候难免要把【数组】或者【指针】传给函数,那函数的参数该如何设计呢?

一维数组传参

void test(int arr[])
{}
void test(int arr[10])
{}
void test(int *arr)
{}
void test2(int *arr[20])
{}
void test2(int **arr)
{}
int main()
{
int arr[10] = {0};
int *arr2[20] = {0};
test(arr);
test2(arr2);
}

以上所有的形参写法都是可行的,我们来总结一下:
一维数组传参,形参可以是数组,也可以是指针的,当形参是指针的时候,要注意类型

二维数组传参

//可行                                                //不可行
void test(int arr[3][5])                             void test(int arr[3][5])                             
{}                                                   {}
void test(int arr[][5])                              void test(int* arr[5])
{}                                                   {}
void test(int (*arr)[5])                             void test(int **arr)
{}                                                   {}
int main()
{
int arr[3][5] = {0};
test(arr);
}

总结:
二维数组传参,参数可以是指针,也可以是数组。如果是数组,行可以省略,但列不能省略,如果是指针,传过去的是第一行的地址,形参就应该是数组指针

一级指针传参

看下面这段代码

#include <stdio.h>
void print(int *p, int sz)
{
int i = 0;
for(i=0; i<sz; i++)
{
printf("%d\n", *(p+i));
}
}
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9};
int *p = arr;
int sz = sizeof(arr)/sizeof(arr[0]);
//一级指针p,传给函数
print(p, sz);
return 0;
}

只要形参和实参类型相同即可,
那么当一个函数的参数部分为一级指针的时候,函数能接收什么参数?

int a;
print(&a);

int* pa=&a;
print(&a);

int arr[10};
print(arr);

由此可见,函数能接受变量地址,指针,整型数组的数组名

二级指针传参

#include <stdio.h>
void test(int** ptr)
{
printf("num = %d\n", **ptr);
}
int main()
{
int n = 10;
int*p = &n;
int **pp = &p;
test(pp);
test(&p);
return 0;
}

二级指针变量接受的就是一级指针变量的地址

那么当函数的参数为二级指针的时候,可以接收什么参数?

test(二级指针变量);

test(一级指针变量的地址);

int *arr[10];
test(arr);

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

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

相关文章

关于举办2023年中国可持续塑料峰会的通知

根据麦肯锡报道&#xff0c;如果塑料需求按照目前的趋势发展&#xff0c;到2030年&#xff0c;全球塑料废物量将从2016年的每年2.6亿吨增加到每年4.6亿吨&#xff0c;使已经严重的环境问题上升到一个全新的水平。目前世界塑料年产量超过3亿吨&#xff0c;为社会生产生活带来巨大…

数据结构与算法之最短路路径与最短路径和动态规划

If every unfolding we experience takes us further along in life, then, we are truly experiencing what life is offering.如果我们在人生中体验的每一次转变都让我们在生活中走得更远&#xff0c;那么&#xff0c;我们就真正的体验到了生活想让我们体验的东西。Do not tr…

【Java|golang】 1238. 循环码排列---格雷编码

给你两个整数 n 和 start。你的任务是返回任意 (0,1,2,…,2^n-1) 的排列 p&#xff0c;并且满足&#xff1a; p[0] start p[i] 和 p[i1] 的二进制表示形式只有一位不同 p[0] 和 p[2^n -1] 的二进制表示形式也只有一位不同 示例 1&#xff1a; 输入&#xff1a;n 2, start …

Linux基础命令-netstat显示网络状态

文章目录 netstat 命令介绍 语法格式 基本参数 显示各列内容分析 1&#xff09;netstat -a显示各列内容分析 2&#xff09;netstat -r显示各列内容分析 3&#xff09;netstat -i 显示各列内容分析 参考实例 1&#xff09;显示系统网络状态的所有连接 2&#xff09;…

Kubernetes之探针probe

deployment只保证pod的状态为running。如果pod状态是running&#xff0c;但是里面丢失了文件&#xff0c;导致用户不能访问数据&#xff0c;则deployment是不管用的&#xff0c;此时就需要probe来检测pod是否正常工作。 probe是定义在容器里的&#xff0c;可以理解为容器里加的…

python基于vue品牌首饰售卖平台

可定制框架:ssm/Springboot/vue/python/PHP/小程序/安卓均可开发目录 开发语言&#xff1a;Python python框架&#xff1a;django/flask 软件版本&#xff1a;python 数据库&#xff1a;mysql 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;PyCharmscode 项目介…

每日学术速递2.22

CV - 计算机视觉 | ML - 机器学习 | RL - 强化学习 | NLP 自然语言处理 Subjects: cs.CV 1.PriSTI: A Conditional Diffusion Framework for Spatiotemporal Imputation 标题&#xff1a;PriSTI&#xff1a;时空插补的条件扩散框架 作者&#xff1a;Mingzhe Liu, Han Huan…

mysql学习(复习)总结 - 索引机制及分类

mysql索引机制及分类1、索引简介2、索引机制2.1 红黑树(平衡二叉树)2.2 哈希2.3 平衡多路查找树&#xff08;B-Tree&#xff09;2.4 Btree2.5 不同数据结构的索引对比3、索引分类3.2 存储分类-聚簇、二级(辅助)索引3.2.1 聚簇(集)索引3.2.1.1 定义3.2.1.2 聚集索引的适用情形3.…

计算机网络(五):三次握手和四次挥手,TCP,UDP,TIME-WAIT,CLOSE-WAIT,拥塞避免,

文章目录零. TCP和UDP的区别以及TCP详解TCP是如何保证可靠性的TCP超时重传的原理TCP最大连接数限制TCP流量控制和拥塞控制流量控制拥塞控制TCP粘包问题一、三次握手和四次挥手二、为什么要进行三次握手&#xff1f;两次握手可以吗&#xff1f;三、为什么要进行四次挥手&#xf…

23.2.23 22湖北省赛 B

好久没打卡了, 随便找的个水题写 这题是简单难度的 ab1 所以可以找到固定规律, 通过手动模拟可以发现 假设两种水叫做a水和b水 先倒入a水 1:0 倒入b水 1:1 此时水杯为 倒出一半的混合物, 因为ab水互溶, 比例不变 再加入a水或者b水将容器填满 比例现在变为 3:1 混合之后再…

一文让你了解Docker的前世今生

Docker概述 Docker为什么会出现&#xff1f; 我们在开发一款产品的过程中一定是需要至少两套环境的&#xff1a;开发和生产。现在更多的公司都是在使用开发、测试、生产三套环境&#xff0c;我们对不同环境的配置和打包都是非常繁琐和复杂的&#xff0c;而且重复度很高&#…

代码随想录 || 回溯算法93 78 90

Day2493.复原IP地址力扣题目链接给定一个只包含数字的字符串&#xff0c;复原它并返回所有可能的 IP 地址格式。有效的 IP 地址 正好由四个整数&#xff08;每个整数位于 0 到 255 之间组成&#xff0c;且不能含有前导 0&#xff09;&#xff0c;整数之间用 . 分隔。例如&#…

高级信息系统项目管理(高项 软考)原创论文——成本管理(2)

1、如果您想了解如何高分通过高级信息系统项目管理师(高项)你可以点击链接: 高级信息系统项目管理师(高项)高分通过经验分享_高项经验 2、如果您想了解更多的高级信息系统项目管理(高项 软考)原创论文,您可以点击链接:

Apache Hadoop生态部署-zookeeper分布式安装

目录 查看服务架构图-服务分布、版本信息 一&#xff1a;安装前准备 1&#xff1a;zookeeper安装包选择--官网下载 2&#xff1a;zookeeper3.5.7安装包--百度网盘 二&#xff1a;安装与常用配置 2.1&#xff1a;下载解压zk安装包 2.2&#xff1a;配置环境变量 2.3&#x…

Firebase常用功能和官方Demo简介

一、Firebase简介Firebase刚开始是一家实时后端数据库创业公司&#xff0c;它能帮助开发者很快的写出Web端和移动端的应用。自2014年10月Google收购Firebase以来&#xff0c;用户可以在更方便地使用Firebase的同时&#xff0c;结合Google的云服务。现在的Firebase算是谷歌旗下的…

17.1 Display system tasks

系统任务的显示组分为三类&#xff1a;显示和写入任务、选通监视任务和连续监视任务。17.1.1 The display and write tasks $display和$write系统任务的语法如语法17-1所示。 display_tasks ::display_task_name [ ( list_of_arguments ) ] ; display_task_name ::$display | …

AI技术的发展,人工智能对我们的生活有那些影响?

人工智能&#xff08;Artificial Intelligence&#xff09;&#xff0c;英文缩写为AI。它是研究、开发用于模拟、延伸和扩展人的智能的理论、方法、技术及应用系统的一门新的技术科学。 人工智能是计算机科学的一个分支&#xff0c;它企图了解智能的实质&#xff0c;并生产出一…

使用spring boot自带log

spring boot starter 版本2.7.8 这个版本自带了slf4j-api 里面包含了logback-classic logback其他包 可以直接使用slf4j 但是配置的日志输出到文件是不能正常生效的 官网说明&#xff1a; Spring Boot Reference Documentation springboot 所有内部日志都使用 common loggin…

替代启攀微8按键触控八通道触摸芯片-GTC08L

能完美替代启攀微8按键触控八通道电触摸芯片-GTC08L芯片是一款非常适用于音响上超稳定超抗干扰低功耗八通道电容式触摸IC&#xff1b;可通过触摸实现各种逻辑功能控制&#xff1b;操作简单、方便实用&#xff1b;电压范围宽&#xff0c;可在2.7V&#xff5e;5.5V&#xff08;单…

JavaWeb--Mybatis练习

Mybatis练习Mybatis练习1 配置文件实现CRUD1.1 环境准备1.2 查询所有数据1.2.1 编写接口方法1.2.2 编写SQL语句1.2.3 编写测试方法1.2.4 起别名解决上述问题1.2.5 使用resultMap解决上述问题1.2.6 小结1.3 查询详情1.3.1 编写接口方法1.3.2 编写SQL语句1.3.3 编写测试方法1.3.4…