sizeof获取数组长度

2024-06-18 14:32
文章标签 数组 获取 sizeof 长度

本文主要是介绍sizeof获取数组长度,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

昨天开始看《算法导论》这本书,第一个算法就是(直接)插入排序,根据书里给出的伪代码写出了C语言代码,也根据自己的理解重新写了一个。虽然实现了算法的基本要求,但有些细节没有处理好,今天就来完善一下。

在昨天实现的代码中,用来测试的数组int array[] = {5,2,4,6,1,3};是固定的,如果要更换待排序数组话,原来程序中的相关变量就会因为数组长度的改变要做相应的改变。我想要的就是给定任意的数组,在不修改任意代码的条件下实现插入排序。所以,核心问题就是怎么获取给定数组中的元素个数。

自然而然想到的就是sizeof(C/C++中的一个操作符,其功能是返回一个对象或者类型所占的内存字节数)。

一、首先来看看怎么获取整数数组的长度

sizeof的语法形式如下:

(1)sizeof( object ); // sizeof( 对象 );
(2)sizeof( type_name ); // sizeof( 类型 );
(3)sizeof object; // sizeof 对象;

个人喜欢使用sizeof()的形式,因为形式统一,而且可以避免引起混乱导致出错。

假设有一给定数组 int array[] = {5,2,4,6,1,3};

length为数组array[]中的元素个数,那么

length = sizeof(array)/sizeof(*array);            //表达式1
//length = sizeof(array)/sizeof(array[0]);        //表达式2
//length = sizeof(array)/sizeof(int);             //表达式3

上述三个表达式都能得到正确的结果,虽然表达式略有不同,但原理是相同的,即通过sizeof(array)获取整个数组所占的内存字节数,再通过sizeof(*array)或者sizeof(array[0])或者sizeof(int)来获取每个元素所占的字节数,数组所占的字节数除以每个元素所占的字节数就是该数组的元素个数了。

二、字符串数组长度的获取

char str[] = {"This is a string!"};
length = sizeof(str)-1;//包含空格

之所结果要减1,是因为字符串结束符'\0‘在数组中占用了一个字节。

三、源代码

/**************************************
*获取字符串长度(元素个数)  By 羽墨
*print_length.c
***************************************/
#include <stdio.h>void main()
{int array[] = {5,2,4,6,1,3};char str[] = {"This is a string!"};int length = sizeof(array)/sizeof(*array);printf("The length of string '%s' is %d\n",str,sizeof(str)-1);printf("The length of array is %d\n",length);
}

运行结果

The length of string 'This is a string!' is 17
The length of array is 6

四、注意

1、如果在定义数组时就给定了数组的大小,如int array[len];,则不管数组中初始化了多少个(显然应不大于len)元素,最后的数组中的元素个数都是len。所以要想获得数组中真实元素的个数,在初始化数组时应注意这一点。

2、向子函数传递数组后,然后在子函数内部获取数组长度。先来看一个错误示例程序:

int getLength(int array[])
{int length;length=sizeof(array)/sizeof(array[0]); return length;
}len = getLength(array);
printf("The length of array is %d\n",len);

这样得到的结果始终都是1。因为数组作为参数传给函数时传的是指针而不是数组,传递的是数组的首地址。在本示例中,函数名array传递到子函数后就完全退化为一个指针,该指针指向的是数组array所在的地址,即数组array第一个元素array[0]所在的地址。也就是说系统只是告诉该函数这个存储空间存有数据,但并没有告诉函数这个数据存储空间有多大。sizeof(array)的结果是指针变量array所占内存的字节数,具体大小与系统有关,一般在32位机器上占4个字节,array[0]是int类型,同样占4个字节,所以结果为1。所以要获得数组的长度最好在数组定义所在的区域内。

对于上述情形,查阅资料后给出两种解决方案:

(1)进入子函数后用函数memcpy将数组拷贝出来,函数memcpy所需要的长度由另一个形参传递进来,像这样:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/**************************************
*摘自360百科(稍作修改)  By 羽墨
*将数组内容拷贝到函数内部
***************************************/
void fun(char *p, int len)
{char *pData;pData = (char*) malloc (len+1);      //申请内存空间memcpy(pData, p, len+1);             //拷贝数组printf("%s\n",pData);//计算数组大小(略)free(pData);                          //释放内存pData = NULL;
}
int main(int argc, char *argv[])
{char str[] = {"This is a string!"};//初始化数组fun(str, 17);return 0;
}

个人觉得没有必要怎么做。如果能够所需的长度作为形参传递到子函数中,显然就不需要在子函数当中另行计算,so,请看另一个方案。

(2)在子函数外部计算好元素个数后作为形参传递到子函数中进行其他操作。在这里仅给出(直接)插入排序算法的main函数,insert_sort(array,length)函数只需将子函数中用于记录数组元素个数的临时变量length放进子函数的参数列表即可。

#include <stdio.h>void print_array(int *arr, int len);int main(int argc, char *argv[])
{int array[] = {5,2,4,6,1,3};char str[] = {"This is a string!"};int length = sizeof(array)/sizeof(*array);    //计算数组中的元素个数printf("The original");print_array(array, length);insert_sort(array,length);                     //插入排序printf("The sorted");print_array(array, length);                    //输出排序结果return 0;
}void print_array(int *arr, int len)
{int i = 0;for(i=0;i<len;i++){printf("%d ", arr[i]);}
}

这样一来,对于任意的目标序列都可以直接进行排序而不需要修改任何程序语句。

这篇关于sizeof获取数组长度的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1072262

相关文章

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

JavaScript对象转数组的三种方法实现

《JavaScript对象转数组的三种方法实现》本文介绍了在JavaScript中将对象转换为数组的三种实用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录方法1:使用Object.keys()和Array.map()方法2:使用Object.entr

Java发送SNMP至交换机获取交换机状态实现方式

《Java发送SNMP至交换机获取交换机状态实现方式》文章介绍使用SNMP4J库(2.7.0)通过RCF1213-MIB协议获取交换机单/多路状态,需开启SNMP支持,重点对比SNMPv1、v2c、v... 目录交换机协议SNMP库获取交换机单路状态获取交换机多路状态总结交换机协议这里使用的交换机协议为常

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

C#使用iText获取PDF的trailer数据的代码示例

《C#使用iText获取PDF的trailer数据的代码示例》开发程序debug的时候,看到了PDF有个trailer数据,挺有意思,于是考虑用代码把它读出来,那么就用到我们常用的iText框架了,所... 目录引言iText 核心概念C# 代码示例步骤 1: 确保已安装 iText步骤 2: C# 代码程

Spring Boot中获取IOC容器的多种方式

《SpringBoot中获取IOC容器的多种方式》本文主要介绍了SpringBoot中获取IOC容器的多种方式,包括直接注入、实现ApplicationContextAware接口、通过Spring... 目录1. 直接注入ApplicationContext2. 实现ApplicationContextA

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

SpringBoot 获取请求参数的常用注解及用法

《SpringBoot获取请求参数的常用注解及用法》SpringBoot通过@RequestParam、@PathVariable等注解支持从HTTP请求中获取参数,涵盖查询、路径、请求体、头、C... 目录SpringBoot 提供了多种注解来方便地从 HTTP 请求中获取参数以下是主要的注解及其用法:1

Java中数组与栈和堆之间的关系说明

《Java中数组与栈和堆之间的关系说明》文章讲解了Java数组的初始化方式、内存存储机制、引用传递特性及遍历、排序、拷贝技巧,强调引用数据类型方法调用时形参可能修改实参,但需注意引用指向单一对象的特性... 目录Java中数组与栈和堆的关系遍历数组接下来是一些编程小技巧总结Java中数组与栈和堆的关系关于