论C++如何优雅的使用数组

简介: C/C++中如果一个函数接受一个数组作为参数,那么数组将会被退化为指针,如果定义如下代码: //数组arr的大小未知。 int arrsize(int arr*) { cout

C/C++中如果一个函数接受一个数组作为参数,那么数组将会被退化为指针,如果定义如下代码:

//数组arr的大小未知。
int arrsize(int arr*) {
  cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << endl; //1 }
  • 1
  • 2
  • 3
  • 4

在上面那段代码中不仅得到的数组大小是不正确的,还会出现让调用则不明白是传递int变量的地址,还是传递一个指针(数组),为了解决第二个歧义现象,我们可以定义如下:

//数组arr的大小依旧未知。
int arrsize(int arr[]) {
  cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << endl; //1 }
  • 1
  • 2
  • 3
  • 4

即使我们按上面那种定义,但数组的的大小我们依旧不知道,但现在编译器还会提示类似如下警告:

 warning: ‘sizeof’ on array function parameter ‘arr’ will return size of ‘int*’ [-Wsizeof-array-argument]
  • 1

为了更好的解决上面的问题我们可以考虑使用一个引用形参,可以有如下代码:

//数组arr的大小必须是12,否则会报错。
int arrsize_const_size(int (&arr)[12]) {
  cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << endl;//12 } 
  • 1
  • 2
  • 3
  • 4
  • 5

即使我们使用引用形参解决了,在函数内部我们无法正确获取数组大小的问题,但更复杂的问题出现了,我们只能接受固定数量的大小的数组,解决这个问题,我们可以通过一种很常规的手法定义函数如下:

 //指定一个数组大小n
int arrsize_n(int arr[], int n) {

}
  • 1
  • 2
  • 3
  • 4
  • 5

上面虽然解决了,但我们多传递了一个参数,调用代码看起来没有前两个更加简洁了,虽然问题被很好的解决了,为了更好的解决这个问题我们可以把推断数组大小的事交个编译器,使用非类型模板参数。

template<int n>
int arrsize_template_size(int (&arr)[n]) {
  cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << endl;//12 cout << "n : " << n << endl;//12 for (int i = 0; i < n; i++) { cout << "arr[" << i << "] = " << arr[i] << endl; } return 0; } 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

下面给出完整的测试代码:

#include <iostream>

using namespace std;


//数组arr的大小未知。 int arrsize(int arr[]) { cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << endl;//1 return 0; } //数组arr的大小必须是12,否则会报错。 int arrsize_const_size(int (&arr)[12]) { cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << endl;//12 return 0; } //指定一个数组大小n int arrsize_n(int arr[], int n) { return 0; } template<int n> int arrsize_template_size(int (&arr)[n]) { cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << endl;//12 cout << "n : " << n << endl;//12 for (int i = 0; i < n; i++) { cout << "arr[" << i << "] = " << arr[i] << endl; } return 0; } int main() { int arr[12] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, }; int arr1[16] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, }; arrsize(arr); arrsize_const_size(arr); cout << "-------------------------------------" << endl; arrsize_template_size(arr); cout << "-------------------------------------" << endl; arrsize_template_size(arr1); cout << "-------------------------------------" << endl; return 0; }

如果array类型也是复用的话,那么就有最终版本:

template<typename T,int n>
        void print_arr_in_tpl(T (&arr)[n]){
            std::cout << "element num : " << sizeof(arr) / sizeof(arr[0]) << std::endl;//
            for(int i=0;i<n;++i){
                std::cout<<arr[i]<<std::endl;
            }
        }

 

哈哈, 大家拿去用吧




谋胆并重
目录
相关文章
|
5月前
|
搜索推荐 编译器 C语言
【C++核心】特殊的元素集合-数组与字符串详解
这篇文章详细讲解了C++中数组和字符串的基本概念、操作和应用,包括一维数组、二维数组的定义和使用,以及C风格字符串和C++字符串类的对比。
116 4
|
1月前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
40 5
|
9月前
|
存储 C++
C++指针数组
C++指针数组
67 1
|
5月前
|
C++
C++(十一)对象数组
本文介绍了C++中对象数组的使用方法及其注意事项。通过示例展示了如何定义和初始化对象数组,并解释了栈对象数组与堆对象数组在初始化时的区别。重点强调了构造器设计时应考虑无参构造器的重要性,以及在需要进一步初始化的情况下采用二段式初始化策略的应用场景。
|
6月前
|
算法 C++
c++学习笔记04 数组
这篇文章是C++学习笔记4,主题是数组。
56 4
|
6月前
|
C++ 索引
C++数组、vector求最大值最小值及其下标
C++数组、vector求最大值最小值及其下标
222 0
|
7月前
|
C++ 索引 运维
开发与运维数组问题之在C++中数组名和指针是等价如何解决
开发与运维数组问题之在C++中数组名和指针是等价如何解决
47 6
|
7月前
|
存储 安全 C++
开发与运维数组问题之声明一个数组如何解决
开发与运维数组问题之声明一个数组如何解决
52 6
|
7月前
|
存储 C++ 容器
开发与运维数组问题之C++标准库中提供数据容器作为数组的替代如何解决
开发与运维数组问题之C++标准库中提供数据容器作为数组的替代如何解决
83 5
|
6月前
|
安全 编译器 C语言
C++入门-数组
C++入门-数组