Day16-指针2
数组指针与指针数组
变量指针:指向变量的地址。
数组指针:指向数组的地址。
指针变量:存放其他变量地址的变量。
指针数组:存放数组元素指针的变量。
数组指针
- 概念:数组指针是指向数组的指针。
- 特点:
- 先有数组,后有指针。
- 它指向的是一个完整的数组。
- 示例:一维数组指针
- 语法:
数据类型(*指针变量)[容量];
- 案例:一维数组指针
/* 数组指针:指向数组的指针 */ #include <stdio.h>int main() {// 一维数组指针// 先有数组,再有指针int arr[3] = {100, 200, 300};// 获取数组元素个数int len = sizeof arr / sizeof arr[0];// 定义一个数组指针,指向arr数组// 数组指针的语法:数据类型(*指针变量名)[数据容量]int(*p)[3] = &arr; // 此时p不是指向数组中arr的第一个元素,而是指向arr这个数组本身printf("%p\n", p);// p++; 此时p++会跳出整个数组// printf("%p\n", p);printf("%d\n", (*p)[2]); // 300// 遍历for (int i = 0; i < len; i++){printf("%d\n", (*p)[i]);}printf("\n");// int *p = &arr[0] | arr; 这种写法,代表p指向的不是数组本身,是数组中的第一个元素 }
- 二维数组指针
- 语法:
数据类型(*指针变量名)[容量];
- 指针和数组中符号优先级: () > [ ] > *
- 案例:
- 语法:
- 语法:
/*二维数组指针案例:
*/
#include <stdio.h>int main()
{// 数组指针:先有数组,再有指针int arr[][3] = {{100, 200, 300}, {1000, 2000, 3000}, {10, 20, 30}};// 创建一个数组指针,指向二维数组// int(*p)[3] = &arr; // p[0] --> {100,200,300},p[1] -->{1000,2000,3000},也就是说:p[0] = 元素100的首地址,p[1] = 元素1000的首地址int(*p)[3] = arr; // arr等价于&arr[0],p[0] = 元素100的首地址// 获取元素100?printf("100-%d\n", (*p)[0]); // arr[0][0]// 获取元素2000?printf("2000-%d,%d,%d\n", *(*(p + 1) + 1), *(p[1] + 1), p[1][1]); // arr[i][j] ===> *(*(p+i)+j) ==> *(p[i]+j) ==>p[i][j]return 0;}
指针数组
- 概念:指针数组是一个数组,数组中的每一个元素都是一个指针。
- 特点:
- 先有“指针”,后有“数组”
- 指针数组的本质是一个数组,只是数组中的元素类型为指针。
- 语法:
数据类型 *数组名 [容量];
- 案例:
/*
指针数组
*/
#include <stdio.h>int main()
{// 定义三个变量int a = 10, b = 20, c = 30;// 定义指针数组,指针数组用来存放指针(变量或者常量的内存地址)int *arr[3] = {&a, &b, &c};// 获取数组大小int len = sizeof arr / sizeof arr[0];// 遍历数组for (int i = 0; i < len; i++){printf("%3d", *(arr[i])); // 输出每个指针所指向的值,需要解引用}printf("\n");return 0;
}
- 建议:我们一般使用指针数组处理字符串
char [][10] = {"hello","majie"
}
字符串指针
字符串实现
在C语言中,表示一个字符串有以下两种方式
1.用字符数组存放一个字符串
2.用字符指针指向一个字符串
/*字符串的两种实现方式
*/
#include <stdio.h>
// 使用字符数组实现
void fun()
{// 定义伪字符串char str[] = "I Love You!";printf("%s\n", str);
}// 使用字符指针实现
void fun1()
{char *str = "I Love You!";printf("%s\n", str);
}int main()
{fun();fun1();return 0;
}
注意:字符数组和字符指针变量都能实现字符串的存储与运算。
字符数组和字符指针的联系
- 字符数组由元素组成,每个元素中存放一个字符,而字符指针变量中存放的是地址,也能作为函数参数。
- 只能对字符数组中的各个元素赋值,而不能用赋值语句对整个字符数组赋值。
- 字符数组名虽然代表地址,但数组名的值不能改变。因为数组名是常量。
- 对于字符串中字符的存取,可以用下标法,也可以用指针法。
案例:
/*字符数组和字符指针的联系
*/
#include <stdio.h>int main()
{// 使用两种方式定义字符串char str1[] = "你好,马杰克!";char *str2 = "你好,马杰克!";// 测试赋值// str1 = "你好,龙1"; 不能对字符数组整体赋值,如果要赋值,请使用strcpy()str2 = "你好,龙2";// 打印输出printf("%s\n%s\n", str1, str2);char a[] = "I Love You!";char *b = "I love You!";// 下标法和指针法访问字符串printf("%c\n%c\n", a[3], *(b + 3)); // 0 0return 0;
}
字符串指针作为形式参数
- 实参与形参都可以是字符数组
- 实参用字符数组,形参用字符指针(在函数内部不能对字符串中的字符做修改)
test5(char *arr,int len)
{arr[2] = 'A'; // 错误,字符串常量一旦创建,就不能被更改
}main()
{char arr[] = "abc"; //字符串常量,常量是不可修改的test(arr,3);
}
- 实参和形参都是指针变量(在函数内部不能对字符串中的字符做修改)
test(char *arr,int len)
{arr[2] = 'A'; // 错误
}
main()
{char arr[] = "abc";char *p = arr; // &arr[0]test(p,3);
}
- 实参是指针类型,形参是字符数组
1
注意:
1.字符数组在创建的时候,会在内存中开辟内存空间,内存空间可以存放字符数据;字符指针在创建的时候,需要依赖字符数组,字符指针在内存中开辟的内存空间中,存放的是数组元素的内存地址。字符指针的创建依赖字符数组,字符数组可以独立存在,而字符指针不能独立存在。
2.字符数组可以初始化,但是不能赋值;字符指针可以初始化,也可以赋值。
// 字符数组
char str1[11] = "I love You!";str1 ="Home";//错误str1[]=“home”;// 错误// 字符指针
char *str2 ="I love You!";str2="home";// 正确
案例2:
/*给定一个字符串,截取start到end之间字符串,含头不含尾
*/
#include <stdio.h>
// 字符串截取函数
int str_split(char *str, int start, int end, char *temp)
{// 定义一个循环变量int i = 0, k = 0;// 定义一个字符指针,用来接收str中截取到的字符串// char *temp ;// 循环遍历每一个字符while (str[i] != '\0'){if (i >= start && i < end){temp[k] = str[i];k++;}i++;}temp[k] = '\0';// 更新str中的数据str = temp;return k;
}
int main()
{char *str = "abcdefg";char temp[100];int len = str_split(str, 2, 5, temp);printf("str=%s,len=%d\n", temp, len);return 0;
}
函数指针与指针函数
函数指针
- 定义:函数指针本质是指针,它是函数的指针(定义一个指针变量 ,变量中存储了函数的地址)
- 函数指针存在的意义:
1.让函数多了一种调用方式
2.函数指针作为形参,可以形式调用(回调函数)
- 定义格式:
返回值类型 (*变量名)(形式参数列表);
举例:
int (*p) (int a,int b);
- 函数指针的初始化
1.定义同时赋值
2.先定义后赋值//得先有函数,才能定义函数的指针 int fun(int a,int b){......}//定义函数指针并给它赋值 int (*p) (int a,int b) = fun; // fun不能跟()
//得先有函数,才能定义这个函数的指针 float int fun(int a,double b,char c){....}//定义函数指针 float (*p) (int a,double b,char c); //赋值 p = fun;
总结:
1.函数指针指向的函数要和函数指针定义的返回值类型,形参列表对应,否则编译报错。
2.函数指针是指针,但不能指针运算,如p++等,没有实际意义。
3.函数指针作为形参,可以形成回调(后面说)。
4.函数指针作为形参,函数调用时的实参只能是与之对应的函数名,不能带小括号。
5.函数指针的形参列表中的变量名可以省略。
案例:
/*函数指针:指向函数的指针变量就是函数指针需求:求a,b两个数的最大值
*/
#include <stdio.h>
int max(int a, int b)
{if (a > b){return a;}return b;
}int main()
{int a = 3, b = 2, c;// 普通函数调用c = max(3, 2);printf("%d,%d两个数中的最大值是: %d\n", a, b, c);// 通过指针变量访问它指向的函数// 创建指针并初始化int (*p)(int a, int b) = max;// 调用函数指针c = p(a, b);printf("%d,%d两个数中的最大值是: %d\n", a, b, c);// 调用函数指针c = (*p)(a, b);printf("%d,%d两个数中的最大值是: %d\n", a, b, c);return 0;
}
指针函数
- 定义:本质是函数,这个函数的返回值是指针,这个函数称为指针函数。
- 定义格式:
指针类型 函数名 (形参列表) {函数体;return 指针变量; }
举例:
int *get(int a) {int *b = &a;//return &a; // 编译报警告return b; }
- 注意:
在函数中不要直接返回一个局部变量的地址,因为函数调用完毕后,局部变量会被回收,使得返回的地址就不明确,此时返回的指针就是野指针。
解决方案:
如果非要访问,可以给这个局部变量添加 static ,可以延长它的生命周期,从而避免野指针(尽量少用,因为存在内存泄漏)。
- 注意:
案例:
/*指针函数:函数的返回值是指针类型需求:有若干个学生的成绩(每个学生有4门课程)要求在用户输入学生序号以后:能输出该学生的全部成绩。用指针函数来实现。
*/
#include <stdio.h>// 定义一个函数,传入学生的序号,返回这个学生的所有课程成绩
float *search(float (*p)[4], int n)
{// 定义一个指针,用来接收查询到的某个学生的所有课程float *pt;pt = *(p + n);return pt;
}int main()
{// 准备所有学生的成绩float score[][4] = {{60, 70, 80, 90},{56, 66, 76, 76},{35, 68, 90, 37}};int i, m;float *p;printf("请输入学生序号(0-2): \n");scanf("%d", &m);printf("第%d个学生的成绩: \n", m);p = search(score, m); // 函数返回值为行的首地址// 遍历for (i = 0; i < 4; i++){printf("%5.2f\t", p[i]);}printf("\n");return 0;
}
相关文章:
Day16-指针2
数组指针与指针数组 变量指针:指向变量的地址。 数组指针:指向数组的地址。 指针变量:存放其他变量地址的变量。 指针数组:存放数组元素指针的变量。 数组指针 概念:数组指针是指向数组的指针。特点: 先…...

数据结构(5.5_3)——并查集的进一步优化
Find操作的优化(压缩路径) 压缩路径——Find操作,先找到根节点,再将查找路径上所有结点都挂到根结点下 代码: //Find "查"操作优化,先找到根节点,再进行"路径压缩" int Find(int S[], int x) {…...
(回溯) LeetCode 131. 分割回文串
原题链接 一. 题目描述 给你一个字符串 s,请你将 s 分割成一些子串,使每个子串都是 回文串。返回 s 所有可能的分割方案。 示例 1: 输入:s "aab" 输出:[["a","a","b"],[…...

【Linux】阻塞信号|信号原理|深入理解捕获信号|内核态|用户态|sigaction|可重入函数|volatile|SIGCHILD|万字详解
目录 编辑 一,常见的信号术语 二,信号在内核中的表示 信号标志位 Pending表 Block表 handler表 POSIX.1标准 三,sigset_t 信号集操作函数 sigemptyset sigfillset sigaddset sigdelset sigismember sigprocmask sig…...

基于Linux对 【进程地址空间】的详细讲解
研究背景: ● kernel 2.6.32 ● 32位平台 –❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀-正文开始-❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀– 在学习操作系统中想必大家肯定都见过下面这…...
[python]使用Pandas处理多个Excel文件并汇总数据
在数据分析和处理过程中,经常需要处理多个Excel文件,并将其中的数据进行汇总和分析。本文介绍使用Python的Pandas库来读取多个Excel文件,并汇总不同类型的数据,例如员工工资、工件数量等。 代码示例 以下是一个完整的代码示例&a…...

提升体验:UI设计的可用性原则
在中国,每年都有数十万设计专业毕业生涌入市场,但只有少数能够进入顶尖企业。尽管如此,所有设计师都怀揣着成长和提升的愿望。在评价产品的用户体验时,我们可能会依赖直觉来决定设计方案,或者在寻找改善产品体验的切入…...

x264 编码器 SSIM 算法源码分析
SSIM SSIM(Structural Similarity Index)是一种用于衡量两幅图像之间视觉相似度的指标。它不仅考虑了图像的亮度、对比度和饱和度,还考虑了图像的结构信息。SSIM的值介于-1到1之间,值越接近1表示两幅图像越相似。 SSIM是基于以下三个方面来计算的: 亮度(Luminance):比…...

echarts使图表组件根据屏幕尺寸变更而重新渲染大小
效果图: 通过 window.addEventListener(resize, this.resizeChart); 实现 完整代码: <template><div class"dunBlock"><div class"char2" id"char2" ref"chart"></div></div…...

电脑图片损坏打不开怎么办?能修复吗?
照片和视频是记录和保存现实生活中的事件的最好方式。由于手机储存空间有限,一般我们会把有纪念意义的照片放到电脑上进行保存,但有时难免会遇到照片被损坏打不开的情况,一旦遇到这种情况,先不要急,也不要因为照片打不…...

vue-cli(二)
箭头函数 一般的函数: 这里window是用来调用函数的 function fun(){console.log(this) } window.fun(); 箭头函数: 1、如果只有一个参数,形参的小括号可以省略 2、如果只有一条语句,{}可以省略 完整的写法 let fun2 a>…...

今日头条的账号id在哪里看(网页版)
今日头条的账号id在哪里看(网页版) 1.https://mp.toutiao.com/profile_v4/index2.登录今日头条账号3.设置->头条号ID 1.https://mp.toutiao.com/profile_v4/index 2.登录今日头条账号 3.设置->头条号ID 打开下方链接: https://mp.to…...
单体应用提高性能和高并发处理-合理使用多核处理
合理使用多核处理能力是提升单体应用性能和处理高并发能力的重要手段。以下是关于如何合理利用多核处理器的详细讲解,包括多线程编程、线程池的使用、并行计算、以及如何避免常见的性能陷阱。 1. 多线程编程 多线程编程是利用多核处理器的直接方式。每个线程可以在…...

基于STM32/GD32的双CAN、一路485开发板
双CAN开发板 双CAN、一路485开发板的设计开发板配置器件选型CAN设计硬件设计软件设计 485设计硬件设计软件设计 其他设计LED硬件按键硬件 PCB板子和实物图开发板测试视频其他资料 双CAN、一路485开发板的设计 最近工作经常会出现一些小问题。就想设计一款带CAN的开发板用来测试…...

快排/堆排/归并/冒泡/
常见的内排序算法 插入排序 直接插入排序 原理:相当于扑克牌变成有序,先拿第一张,把他调节成有序,再拿第二张,与第一张相比找到第二张的位置,再继续拿第三张,以此类推。 void InsertSort(in…...
React基础教程(08):state体验
文章目录 7、state再体验7.1 异步更新状态7.2 同步更新状态方式17.3 同步更新状态方式27.4 betterScroll7.5 列表案例7、state再体验 7.1 异步更新状态 完整代码 import React from "react";export default class App extends React.Component{state = {count:1,}…...

Win10 创建新的桌面2,并实现桌面切换
1. Win10 创建新的桌面2 Win - Tab 2. Win10 桌面切换 Ctrl - Win - ←/→ 我们下期见,拜拜!...

MySQL数据库介绍及基础操作
目录: 一.数据库介绍 二.数据库分类 三. 数据库的操作 四. 常用数据类型 五. 表的操作 一.数据库介绍 1.文件保存数据有以下几个缺点: 1.1文件的安全性问题 1.2文件不利于数据查询和管理 1.3文件不利于存储海量数据 1.4文件在程序中控制不方便 为了解决上述问题&…...

【C语言篇】C语言常考及易错题整理DAY2
文章目录 C语言常考及易错题整理选择题编程题至少是其他数字两倍的最大数两个数组的交集图片整理寻找数组的中心下标多数元素除自身以外数组的乘积不使用加减乘除求两个数的加法 C语言常考及易错题整理 选择题 下列 for 循环的次数为( ) for(int i 0…...

javase入门
最近在学习大数据,学到flume拦截器的时候发现自定义拦截器需要使用java编写,现在开始学一些java入门的东西. 一. java相关组成 path环境变量: 环境变量用于记住程序路径,方便在命令行窗口任意目录启动程序. 二 java中的变量 变量要先定义在使用. int age 15 定义变量要定义其…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...
【磁盘】每天掌握一个Linux命令 - iostat
目录 【磁盘】每天掌握一个Linux命令 - iostat工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景 注意事项 【磁盘】每天掌握一个Linux命令 - iostat 工具概述 iostat(I/O Statistics)是Linux系统下用于监视系统输入输出设备和CPU使…...
生成 Git SSH 证书
🔑 1. 生成 SSH 密钥对 在终端(Windows 使用 Git Bash,Mac/Linux 使用 Terminal)执行命令: ssh-keygen -t rsa -b 4096 -C "your_emailexample.com" 参数说明: -t rsa&#x…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院查看报告小程序
一、开发环境准备 工具安装: 下载安装DevEco Studio 4.0(支持HarmonyOS 5)配置HarmonyOS SDK 5.0确保Node.js版本≥14 项目初始化: ohpm init harmony/hospital-report-app 二、核心功能模块实现 1. 报告列表…...
【RockeMQ】第2节|RocketMQ快速实战以及核⼼概念详解(二)
升级Dledger高可用集群 一、主从架构的不足与Dledger的定位 主从架构缺陷 数据备份依赖Slave节点,但无自动故障转移能力,Master宕机后需人工切换,期间消息可能无法读取。Slave仅存储数据,无法主动升级为Master响应请求ÿ…...
MySQL 8.0 事务全面讲解
以下是一个结合两次回答的 MySQL 8.0 事务全面讲解,涵盖了事务的核心概念、操作示例、失败回滚、隔离级别、事务性 DDL 和 XA 事务等内容,并修正了查看隔离级别的命令。 MySQL 8.0 事务全面讲解 一、事务的核心概念(ACID) 事务是…...

脑机新手指南(七):OpenBCI_GUI:从环境搭建到数据可视化(上)
一、OpenBCI_GUI 项目概述 (一)项目背景与目标 OpenBCI 是一个开源的脑电信号采集硬件平台,其配套的 OpenBCI_GUI 则是专为该硬件设计的图形化界面工具。对于研究人员、开发者和学生而言,首次接触 OpenBCI 设备时,往…...

给网站添加live2d看板娘
给网站添加live2d看板娘 参考文献: stevenjoezhang/live2d-widget: 把萌萌哒的看板娘抱回家 (ノ≧∇≦)ノ | Live2D widget for web platformEikanya/Live2d-model: Live2d model collectionzenghongtu/live2d-model-assets 前言 网站环境如下,文章也主…...

手机平板能效生态设计指令EU 2023/1670标准解读
手机平板能效生态设计指令EU 2023/1670标准解读 以下是针对欧盟《手机和平板电脑生态设计法规》(EU) 2023/1670 的核心解读,综合法规核心要求、最新修正及企业合规要点: 一、法规背景与目标 生效与强制时间 发布于2023年8月31日(OJ公报&…...

消息队列系统设计与实践全解析
文章目录 🚀 消息队列系统设计与实践全解析🔍 一、消息队列选型1.1 业务场景匹配矩阵1.2 吞吐量/延迟/可靠性权衡💡 权衡决策框架 1.3 运维复杂度评估🔧 运维成本降低策略 🏗️ 二、典型架构设计2.1 分布式事务最终一致…...