背景知识
int arr[10]; //arr是一个含有10个整数的数组
int *p1[10]; //p1是一个含有10个指针的数组
int (*p2)[10] = &arr; //p2是一个指针,指向含有10个整数的数组。
数组不能被拷贝,所以函数不能返回数组,以下是几种返回数组指针的方法
一、接数组维度
Type (*function(parameter_list))[dimension]
(*function(parameter_list))两端的括号必须存在,就像我们定义p2时两端必须有括号一样,如果没有这对括号,函数的返回类型将是指针的数组。
#include<iostream>
int a[5]={1,2,3,4,5};
using namespace std;
int(*make_array())[5]//如果是引用的形式也可以
{
return &a;
}
int main(int argc,char *argv[])
{
int (*p)[5]=make_array();
cout<<*p<<endl;
for(int i=0;i!=5;++i)
{
cout<<(**p)++<<" ";//*p只能得到数组的地址
}
}
二、类型别名
#include<iostream>
using namespace std;
using arrT = int[5];//或者typedef int arrT[5];
int arry[5] = {1,2,3,4,5};
arrT* func()
{
return &arry;
}
int main()
{
int (*p)[5] =func();
for(int i=0;i<5;i++)
{
cout<<*(*p + i)<<" "<<endl;
}
return 0;
}
三、尾置返回类型
在C++11中新增了一种方法,任何函数的定义都能使用尾置返回,但是这种形式对于返回类型教复杂的类型最有效,比如返回类型是数组的指针或者是数组的引用时。尾置返回类型跟在形参列表后面并以一个->符号开头。为了表示函数真正的返回类型跟在形参列表之后,在本应该出现返回类型的地方仿置一个auto。
#include<iostream>
using namespace std;
int arry[10] = {1,2,3,4,5,6,7,8,9,10};
auto func() -> int(*)[10]
{
return &arry;
}
int main()
{
int (*p)[10] =func();
for(int i=0;i<10;i++)
{
cout<<(**p)++<<" ";
}
cout<<endl;
return 0;
}
四、使用decltype
如果我们知道函数返回的指针将指向哪个数组,就可以使用decltype关键字声明返回类型。
注:decltype 并不负责把数组类型转换成对应的指针,所以decltype的结果是一个数组,要想func返回指针还必须在函数声明时加一个*符号。
#include <cstddef>
using std::size_t;
#include <iostream>
using namespace std;
int odd[] = {1,3,5,7,9};
int even[] = {0,2,4,6,8};
decltype(odd) *arrPtr(int i)
{
return (i % 2) ? &odd : &even; // returns a pointer to the array
}
int main()
{
int (*arrP)[5] = arrPtr(5); // arrP points to an array of five ints
for (size_t i = 0; i < 5; ++i)
cout << (*arrP)[i] << endl;
return 0;
}
改成引用形式
int odd[] = {1,3,5,7,9};
int even[] = {0,2,4,6,8};
decltype(odd) &arrPtr(int i)
{
return (i % 2) ? odd : even; // returns a pointer to the array
}
int main()
{
int (&arrP)[5] = arrPtr(5); // arrP points to an array of five ints
for (size_t i = 0; i < 5; ++i)
cout << (arrP)[i] << endl;
return 0;
}