在C++98中,标准允许使用花括号{}对数组元素或者结构体进行统一的列表初始值设定
struct Point
{
int _x;
int _y;
};
int main()
{
//使用大括号对数组元素进行初始化
int array1[] = { 1, 2, 3, 4, 5 };
int array2[5] = { 0 };
//使用大括号对结构体元素进行初始化
Point p = { 1, 2 };
//C++98时,map的初始化->使用insert逐个插入
std::map<int, std::string> myMap;
myMap.insert(make_pair(1, "One"));
myMap.insert(make_pair(2, "Two"));
myMap.insert(make_pair(3, "Three"));
// 使用 myMap
for (auto it = myMap.begin(); it != myMap.end(); ++it)
{
cout << it->first << " : " << it->second << std::endl;
}
return 0;
}
C++11扩大了用大括号括起来的列表{初始化列表}的使用范围,使其可用于所有的内置类型和用户自定义的类型,使用初始化列表时,可添加等号,也可不添加。比如:
// 内置类型变量
int x1 = {10};
int x2{10};//建议使用原来的
int x3 = 1+2;
int x4 = {1+2};
int x5{1+2};
// 数组
int arr1[5] {1,2,3,4,5};
int arr2[]{1,2,3,4,5};
// 动态数组,在C++98中不支持
int* arr3 = new int[5]{1,2,3,4,5};
// 标准容器
vector<int> v{1,2,3,4,5};//这种初始化就很友好,不用push_back一个一个插入
map<int, int> m{{1,1}, {2,2,},{3,3},{4,4}};
C++11中新增了initializer_list容器,该容器没有提供过多的成员函数
initializer_list 本质就是一个大括号括起来的列表,如果用auto关键字定义一个变量来接收一个大括号括起来的列表,然后以typeid(变量名).name()的方式查看该变量的类型,此时会发现该变量的类型就是initializer_list
initializer_list容器没有提供对应的增删查改等接口,因为initializer_list并不是专门用于存储数据的,而是为了让其他容器支持列表初始化的。比如:
class Date
{
public:
Date(int year, int month, int day)
:_year(year)
, _month(month)
, _day(day)
{
cout << "Date(int year, int month, int day)" << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
//用大括号括起来的列表对容器进行初始化
vector<int> v = { 1, 2, 3, 4, 5 };
//C++11将initializer_list加入到STL容器的构造函数当中,实现列表初始化
list<int> l = { 10, 20, 30, 40, 50 };
vector<Date> vd = { Date(2022, 8, 29), Date{ 2022, 8, 30 }, { 2022, 8, 31 } };
map<string, string> m{ make_pair("sort", "排序"), { "insert", "插入" } };
//用大括号括起来的列表对容器赋值
v = { 5, 4, 3, 2, 1 };
return 0;
}
在C++98中auto是一个存储类型的说明符,表明变量是局部自动存储类型,但是局部域中定义局部的变量默认就是自动存储类型,所以auto就没什么价值了。
C++11中废弃auto原来的用法,将其用于实现自动类型推断。这样要求必须进行显示初始化,让编译器将定义对象的类型设置为初始化值的类型。比如:
提示:这里描述项目中遇到的问题:
int main()
{
int i = 10;
auto p = &i;
auto pf = strcpy;
cout << typeid(p).name() << endl; //int *
cout << typeid(pf).name() << endl; //char * (__cdecl*)(char *,char const *)
map<string, string> dict = { { "sort", "排序" }, { "insert", "插入" } };
//map<string, string>::iterator it = dict.begin();
auto it = dict.begin(); //简化代码
return 0;
}
若是在C++98中我们要遍历一个数组,可以按照以下方式:
int main()
{
int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
//将数组元素值全部乘以2
for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
{
arr[i] *= 2;
}
//打印数组中的所有元素
for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
{
cout << arr[i] << " ";
}
cout << endl;
return 0;
}
以上方式也是C语言中所用的遍历数组的方式,但对于一个有范围的集合而言,循环是多余的,有时还容易犯错。
C++11中引入了基于范围的for循环,for循环后的括号由冒号分为两部分,第一部分是范围内用于迭代的变量,第二部分则表示被迭代的范围。比如
int main()
{
int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
//将数组元素值全部乘以2
for (auto& e : arr)
{
e *= 2;
}
//打印数组中的所有元素
for (auto e : arr)
{
cout << e << " ";
}
cout << endl;
return 0;
}