C语言 数组做函数参数退化为指针的技术推演
//数组做函数参数退化为指针的技术推演
#include<stdio.h>
#include<stdlib.h>
#include<string.h> //一维数组做函数参数退化为指针的技术推演 void printfA(char * strarr[]);
//计算机中,数组都是线性存储,二维数组元素也是一个个的排列的
//例如: 1,2,3,4,5,6,7,8,9 像这组数据 我们可以认为是一维数组 int a[9]={1,2,3,4,5,6,7,8,9};
//也可以认为是二维数组 int b[3][3]={1,2,3,4,5,6,7,8,9};
//所以计算机并不清楚数组名的步长是多少 就是 a+1移动多少个字节 或者 b+1 移动多少个字节
//这就需要程序员去告诉计算机 数组名的步长是多少
//对于本题 void printfA(char * strarr[3]); 数组做函数参数 该数组是一个一维数组 数组元素类型是 char *
//那么数组strarr的步长应该是 sizeof(char *) 也就是4个字节
//这么来说 我们只需要告诉计算机 你跳4个字节 就OK了
//所以技术推演为printfB
void printfB(char * strarr[]);
//因为计算机根本不关心你有多少个元素 是3个 还是30个 与计算机没关系 是程序员需要关心的(这就是数组越界问题)
//函数参数 char * strarr[] 同样告诉计算机 我是一个一维数组 数组元素是 char * 类型
//那么数组strarr的步长还是 sizeof(char *) 也就是4个字节
//那么我们继续推演 既然计算机只需要确定 该数组每次移动的步长是 4个字节就好
//那么void printfC(char ** strarr);这么写也是可以的 strarr是个指针 strarr里的值指向一个类型是 char *的变量
//步长只与指针的值有关 ,因此strarr的 步长是 sizeof(char *) 也就是4个字节
//所以C语言的开发人员就做了优化 printfC (与我没关系 设计C语言的就是这么优化的)
void printfC(char ** strarr);
//char * strarr[3]做参数退化为char ** strarr) 有2个好处
//好处1:复制一个一维数组char * strarr[3] 比复制一个指针char ** strarr 会耗费更多的内存空间
//char * strarr[3] 需要耗费 sizeof(char *) * 3 =12 个字节的内存空间 ;
//而char ** strarr需要耗费 sizeof(char **) = 4 个字节的内存空间 ;
//节约了内存 和创建 数组时的资源消耗
//好处2:减少了无用解析 ;对于char * strarr[3] 元素个数3 没用,
//这是个一维数组 这个信息没用 ,因为遍历数组的时候从首地址开始遍历, 只要给计算机个首地址就行
//计算机从首地址向后遍历 无需知道他是什么 只需要知道每次的步长是多少就好了 //二维数组做函数参数退化为指针的技术推演
void printfD(int arr[][]);
//对于二维数组,C语言编译器同样需要知道 数组名arr的步长 就是在遍历的时候 每次计算机改移动多少个字节
//那么首先 我们应该确定数组名arr 到底是个什么类型
//数组名 是数组首元素的指针 (我自己的推论) 那么二维数组 可以想象成一维数组 只是这个一维数组的每个元素比较特殊,还是一个一维数组
//那么根据推论 数组名arr的类型是一个一维数组的指针 一维数组是这样定义的 typedef int Myarr[4];
//一维数组指针的类型定义是这样的 typedef int (* PMyarr)[4];
//一维数组指针的变量是这么定义的int (* pmyarr)[4];
//所以数组名arr的类型是 int (* PMyarr)[4]; 因为指针的步长与指针所指向的内存空间相关
//arr指向的是一个typedef int Myarr[4]类型的数组,这个数组有4个元素,每个元素都是int类型
//由此得出arr这个一维数组指针的步长是 sizeof(int)*4 = 16;
//由 一维数组的推演可知 数组元素的个数对C语言编译器并不重要 二位数组的元素可以看作一维数组
//推演出 void printfD(int arr[][4]);
void printfE(int arr[][]);
//又因为 C语言编译器 只需要知道 首地址 和步长 所以 可以用 int (*p)[4] 来代替 int arr[][4]
void printfF(int(*arr)[]); //综合以上分析,导出结论 数组做函数参数退化为指针,指针的类型就是数组名的类型 void main(){
char * strarr[] = { "", "", "" };
int arr[][] = { };
system("pause");
}
C语言 数组做函数参数退化为指针的技术推演的更多相关文章
- C语言 数组做函数参数不传数组个数的遍历方法
//数组做函数参数不传数组个数的遍历方法 #include<stdio.h> #include<stdlib.h> #include<string.h> void ...
- 3205: 数组做函数参数--数组元素求和1--C语言
3205: 数组做函数参数--数组元素求和1--C语言 时间限制: 1 Sec 内存限制: 128 MB提交: 178 解决: 139[提交][状态][讨论版][命题人:smallgyy] 题目描 ...
- 3204: 数组做函数参数--排序函数2--C语言
3204: 数组做函数参数--排序函数2--C语言 时间限制: 1 Sec 内存限制: 128 MB提交: 211 解决: 143[提交][状态][讨论版][命题人:smallgyy] 题目描述 ...
- 3203 数组做函数参数----排序函数--C语言版
3203: 数组做函数参数----排序函数--C语言版 时间限制: 1 Sec 内存限制: 128 MB提交: 253 解决: 151[提交][状态][讨论版][命题人:smallgyy] 题目描 ...
- C语言中数组做函数参数的问题
数组做函数参数,会退化成为一个指针变量.因此在进行数组参数传递的同时,需要传递一个数组长度的参数变量. 数组长度可以通过sizeof(arr)/siezof(arr[0])来得到.关于这个sizeof ...
- go语言基础之数组做函数参数是值拷贝
1.数组做函数参数是值拷贝 示例: package main //必须有个main包 import "fmt" //数组做函数参数,它是值传递 //实参数组的每个元素给形参数组拷贝 ...
- 【面试题003】c数组做为参数退化的问题,二维数组中的查找
[面试题003]c数组做为参数退化的问题,二维数组中的查找 一,c数组做为参数退化的问题 1.c/c++没有记录数组的大小,因此用指针访问数组中的元素的时候,我们要确保没有超过数组的边界, 通过下面 ...
- C语言数组作为函数参数
数组可以作为函数的参数使用,进行数据传送. 数组用作函数参数有两种形式,一种是把数组元素(下标变量)作为实参使用:另一种是把数组名作为函数的形参和实参使用. 数组元素作函数实参 数组元素就是下标变量, ...
- [GO]数组做函数参数
package main import "fmt" //数组为函数参数,实际上是值传递//实参数据里的每个元素,给形参数组拷贝一份//这里形参的数组其实就是实参的复制品 func ...
随机推荐
- RecyclerView解析--onViewDetachedFromWindow()/onViewAttachedToWindow()
先看这段源码介绍: /** * Called when a view created by this adapter has been detached from its window. * * &l ...
- 浅谈RSA加密算法
一.什么是非对称加密 1.加密的密钥与加密的密钥不相同,这样的加密算法称之为非对称加密 2.密钥分为:公钥,私钥 公钥:可以对外给任何人的加密和解密的密码,是公开的 私钥:通过私钥可以生成公钥,但从 ...
- 【读书笔记】iOS-NSPredicate
一,Cocoa提供了一个名为NSPredicate的类,它用于指定过滤器的条件.可以创建NSPredicate对象,通过该对象准确地描述所需的条件,对每个对象通过谓词进行筛选,判断它们是否与条件相匹配 ...
- Spring(七)持久层
一.Spring对DAO的支持 DAO:Data Access Object Spring提供了DAO框架,让开发人员无须耦合特定的数据库技术,就能进行应用程序的开发. Spring封闭了操作Orac ...
- 【mysql】关于innodb中MVCC的一些理解
一.MVCC简介 MVCC (Multiversion Concurrency Control),即多版本并发控制技术,它使得大部分支持行锁的事务引擎,不再单纯的使用行锁来进行数据库的并发控制,取而代 ...
- Oracle与SQL SERVER编程差异分析(入门)
网上有关Oracle与SQL SERVER性能差异的文章很多,结论往往是让你根据数据量与预算来选择数据库.但实际项目中,特别是使用 .Net 开发的系统,支持以上两种数据库或者更多已经成为Boss的普 ...
- 【转】JAVA CAS原理深度分析
java.util.concurrent包完全建立在CAS之上的,没有CAS就不会有此包.可见CAS的重要性. CAS CAS:Compare and Swap, 翻译成比较并交换. java.uti ...
- 将text 文件转为List
Integer 类型 ArrayList<Integer> Mlist = new ArrayList<Integer>(); Scanner scM = new Scanne ...
- [转]关于负margin在页面中布局的应用
本文转载自:http://www.cnblogs.com/jscode/archive/2012/08/28/2660078.html. 今天再写一个布局的时候用到一个margin-top是负值的情况 ...
- 【转】UVALive 5964 LCM Extreme --欧拉函数
题目大意:求lcm(1,2)+lcm(1,3)+lcm(2,3)+....+lcm(1,n)+....+lcm(n-2,n)+lcm(n-1,n)解法:设sum(n)为sum(lcm(i,j))(1& ...