数组做形参退化为指针

如果数组作为函数参数,则数组形参会退化为指针,以下代码在编译器看来是等价的

void fun1(int a[]);
void fun2(int a[]);
void fun3(int a[]);
void fun4(int *a);
#include<iostream>
using namespace std;
void fun1(int a[])
{
cout << sizeof(a) / sizeof(a[]) << endl;
}
void fun2(int a[])
{
cout << sizeof(a) / sizeof(a[]) << endl;
}
void fun3(int a[])
{
cout << sizeof(a) / sizeof(a[]) << endl;
}
void fun4(int *a)
{
cout << sizeof(a) / sizeof(a[])<<endl;
}
void main(int argc, char* argv[])
{
int arr[] = {,,,,,,,,,};
fun1(arr);
fun2(arr);
fun3(arr);
fun4(arr);
cout << sizeof(arr) / sizeof(arr[]) << endl;
system("pause");
}

数组

#include<stdio.h>

void main(int argc, char* argv[])
{
int arr[] = {};
printf("arr=%d, &arr=%d\n", arr, &arr);
printf("arr+1=%d, &arr+1=%d\n",arr+, &arr+);
system("pause");
}

//arr,&arr的数组类型不一样
//arr,数组首元素地址,一个元素4字节,+1,+4
//&arr,整个数组的首地址,一个数组4*10=40字节,+1,+40

char* argv[]和char** argv

void main(int argc, char* argv[])
void main(int argc, char** argv)

main函数写成上面2种形式都行,于是有人自然会认为char* argv[]与char** 等价。char* argv[]与char**本质上是不同的,char* argv[]是素组,你可以这样

char* argv[] = {"hello","world","Linux","NB"}

但是char**就不行,下面代码是语法错误

char** argv = {"hello","world","Linux","NB"}

指针只能指向一块内存,{"hello","world","Linux","NB"}是4块内存。数组可以指向多块内存。

那为啥在main函数中,2者就等价了呢? 因为数组作为形参,会自动退化为指针。

为啥void a不行,void* a就行

对于出void以外的内置类型,自定义类型。我们可以

typename  var;
typename* p_var;

但是唯独void不行

void   a; //语法错误
void* p; //OK

C/C++中类型就像模具,其本身不占用内存,根据模具产生的对象占内存。void a,单纯这一句,编译器无法计算出a到底占用多少内存。void*  p就行,这是因为在32位架构下,指针都是4Byte。

二维数组首行地址,和首行首元素地址的值是一样

#include<stdio.h>

void main(int argc, char* argv[])
{
int i;
char* str1[] = {"","",""};
char str2[][] = { "","","" };
printf("str1=%x, *str1=%x, &(**str1)=%x\n", str1, *str1, &(**str1));
printf("str2=%x, *str2=%x\n", str2, *str2);
for (i = ; i < sizeof(str2) / sizeof(str2[]); i++)
{
//printf("%s\n",str2+i); 与下一行输出内容一样
printf("%s\n", *(str2 + i));//*(str2 + i)等价于str2[i]
}
} system("pause"); }

二级指针的内存模型

 #include <string.h>
#pragma warning(disable:4996) // 等价于_CRT_SECURE_NO_WARNINGS
int main(int argc, char* argv[])
{
int i = ;
char* p0 = NULL;
p0 = (char*)malloc(sizeof(char)*);
strcpy(p0, "Hello World"); //3个char* ,每个的值都是空
char* ch_arr[] = { };
for ( i = ; i < sizeof(ch_arr)/sizeof(ch_arr[]); i++)
{
ch_arr[i] = (char*)malloc(sizeof(char)*);
strcpy(ch_arr[i], "Hello World");
} int arr[];//静态分配
int *p_arr = (int*)malloc(sizeof(int) * );//等价于arr[10],动态分配 char* str_arr[] = { };
char** p_str_arr = (char**)malloc(sizeof(char*) * );
strcpy(p_str_arr[], "Hello World");
}

最后一句会报错

strcpy(p_str_arr[], "Hello World");

出错原因在于向未分配内存的地址拷贝数据,这和7,8行道理一样。

改为如下则不会有问题

char* str_arr[] = {  };
char** p_str_arr = (char**)malloc(sizeof(char*) * );
//strcpy(p_str_arr[0], "Hello World");
for ( i = ; i < ; i++)
{
p_str_arr[i] = (char*)malloc(sizeof(char) * );
strcpy(p_str_arr[i], "Hello World");
}

此时内存模型

从上图可见,heap实际上分配了2大块,于是释放内存步骤为:

先释放那3个100B,再释放3个4B

for ( i = ; i < ; i++)
{
free(p_str_arr[i]);
p_str_arr[i] = NULL;
}
if (NULL != p_str_arr)
{
free(p_str_arr);
}

如果是函数返回二级指针的情况呢,看如下代码

#include <string.h>
#pragma warning(disable:4996) // 等价于_CRT_SECURE_NO_WARNINGS char** getBuf(int n)
{
int i;
char** buf = (char**)malloc(sizeof(char*) * );
//strcpy(buf[0], "Hello World");错误写法
for (i = ; i < ; i++)
{
buf[i] = (char*)malloc(sizeof(char) * );
strcpy(buf[i], "Hello World");
}
return buf;
}
void Free(char** buf, int n)
{
int i;
for (i = ; i < n; i++)
{
free(buf[i]);
buf[i] = NULL;
}
if (NULL != buf)
{
free(buf);
}
}
int main(int argc, char* argv[])
{
int n = ;
int i;
char** buf = getBuf(n);
Free(buf, );
buf = NULL;
}

再说释放内存,为啥Free后还要让buff = NULL

Free(buf, );
buf = NULL;

C++——数组形参退化为指针的更多相关文章

  1. C语言 数组名不是指针

    今天上计算机系统课的时候老师讲到了C中的聚合类型的数据结构.在解释数组名的时候说"数组名是一个指针,指向该数组的第一个元素",附上ppt(第二行): 我觉得这是不正确的,是一个常见 ...

  2. CC++语法::数组名退化(array decaying)

    参考: CSDN::C/C++中数组名退化为指针的情况 stackoverflow::What is array decaying? 起因 笔者在写memset的时候总想偷一点懒(因为我们一般都是为了 ...

  3. C语言中,数组名作为参数传递给函数时,退化为指针

    C语言中,数组名作为参数传递给函数时,退化为指针   C语言中,数组名作为参数传递给函数时,退化为指针:需要数组大小时, 需要一个参数传数组名,另一个传数组大小. 数组名做函数参数时,就相当于指针了. ...

  4. C语言 数组做函数参数退化为指针的技术推演

    //数组做函数参数退化为指针的技术推演 #include<stdio.h> #include<stdlib.h> #include<string.h> //一维数组 ...

  5. c 数组做为形參时 该參数退化为指针

    当数组做为函数的形參的时候,该參数退化为指针,而且是无法直接求得数组的大小. 传数组给一个函数.数组类型自己主动转换为指针类型,因而传的实际是地址. void func(int array[10]) ...

  6. sizeof(数组名)和sizeof(指针)

    在做这道题时: 32位环境下,int *p=new int[10];请问sizeof(p)的值为()A.4              B.10              C.40           ...

  7. Linux C\C++基础——数组形参的使用

    1.数组形参 ]) void fun(int a[]) void fun(int *a) ],int n) void fun(char*p[],int n) void fun(char**p,int ...

  8. c语言 数组名是常量指针

    //数组名是常量指针 #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<stdlib.h> #include ...

  9. C语言 指针基础篇 数组,函数与指针的运用 2 14

    下面看看如何在函数中运用指针吧 下面是往函数传入指针的简单操作,不是传入数组的.判断一个a是否大于b是的话给,是的话对其进行操作,不是的话就直接返回. #include <stdio.h> ...

随机推荐

  1. 【c# 学习笔记】构造函数

    构造函数 主要用于创建类的实例对象.当调用构造函数创建一个对象时,构造函数会为对象分配内存空间,并初始化类的成员.构造函数分为实例构造函数和静态构造函数两种. 1.实例构造函数 实例构造函数用于创建和 ...

  2. 写一个java常用的加密工具类

    1.叙述 java security包下有很多加密算法类,我们可以很简单的调用它们.他们虽然功能很全,但是使用起来步骤有些繁琐.我在这里封装来一些常用的加密算法及他们常用的一些方法,来简化代码. 工具 ...

  3. 查询 ip占用导致ip不通的 问题 查IP对应的mac地址

    IP冲突,  同一个IP配到了多余1台的机器上 ,导致IP 不通的情况,此时需要查询 都有哪台机器配置了这个IP,用 arping 命令, 具体命令 类似于 ping ,直接 arping + 目标地 ...

  4. 【miscellaneous】各种音视频编解码学习详解

    编解码学习笔记(一):基本概念 媒体业务是网络的主要业务之间.尤其移动互联网业务的兴起,在运营商和应用开发商中,媒体业务份量极重,其中媒体的编解码服务涉及需求分析.应用开发.释放license收费等等 ...

  5. C#API解决自定义请求头下的跨域问题

    解决方法一: public class CrosHandler : DelegatingHandler { private const string Origin = "Origin&quo ...

  6. win7下exe文件设置为开机启动

    如何将自己的exe程序设置为开机自启动 如何将自己的exe程序设置为开机自启动 将自己的exe程序设置为开机自启动话不多说,直接看 首先1:cmd—>regedit 其次找到下面的路径就可以:( ...

  7. 教你成为全栈工程师(Full Stack Developer) 四十五-一文读懂hadoop、hbase、hive、spark分布式系统架构

    转载自http://www.shareditor.com/blogshow?blogId=96 机器学习.数据挖掘等各种大数据处理都离不开各种开源分布式系统,hadoop用于分布式存储和map-red ...

  8. super的实例及实现原理

    super实例 class A(): def go(self): print ("go A go!") def stop(self): print ("stop A st ...

  9. EventBus使用的坑

    最近使用eventbus发送通知,在想该怎么携带List集合数据.于是尝试直接发送List. 使用一次,正常接收.使用两次,出现类转换异常.原来在接收List类型的消息时,并不会管List内的泛型,是 ...

  10. 学习笔记:oracle学习一:oracle11g体系结构之物理存储结构

    目录 1.物理存储结构 1.1 数据文件 1.2 控制文件 1.3 日志文件 1.3.1 重做日志文件 1.3.2 归档日志文件 1.4 服务器参数文件 1.4.1 查看服务器参数 1.4.2 修改服 ...