现代C++教程:高速上手(四)-容器
1、线性容器
std::array与std::vector不同的是,array对象的大小是固定的,如果容器大小是固定的,那么可以优先考虑使用std::array容器。
由于std::vector是自动扩容的,当存入大量的数据后,并且对容器进行了删除操作,容器并不会自动归还被删除元素相应的内存,这时候需要手动运行shrink_to_fit()释放这部分内存。
std::array C风格接口传参:
void foo(int *p, int len){
return;
}
std::array<int, 4> arr = {1,2,3,4};
//foo(arr,arr.size()); //非法,无法隐式转换
foo(&arr[0], arr.size());
foo(arr.data(), arr.size());
//使用std::sort
std::sort(arr.begin(), arr.end());
//升序
std::sort(arr.begin(), arr.end(), [](int a, int b){
return b > a;
})
std::forward_list是一个列表容器,使用方法和std::list基本类似。和list的双向链表的实现不同,forward_list使用单向链表进行实现,提供了O(1)复杂度的元素插入,不支持快速随机访问,也是标准库容器中唯一一个不提供size()方法的容器。当不需要双向迭代时,具有比list更高的空间利用率。
2、无序容器
传统c++中的有序容器 std::map / std::set,这些元素内部通过红黑树进行实现,插入和搜索的平均复杂度均为O(log(size))。在插入元素时,会根据<操作符比较元素大小并判断元素是否相同,并选择合适的位置插入到容器中。当对这个容器中的元素进行遍历时,输出结果会按照<操作符的顺序来逐个遍历。
而无序容器中的元素是不进行排序的,内部通过Hash表实现,插入和搜索元素的平均复杂度为O(constant),在不关心容器内部元素顺序时,能够获得显著的性能提升。
c++11引入了两组无序容器:std::unordered_map / std::unordered_multimap和std::unordered_set / std::unordered_multiset。
它们的用法和原有的std::map / std::multimap / std::set / std::multiset基本类似。
#include <iostream>
#include <string>
#include <unordered_map>
#include <map>
using namespace std;
int main(){
unordered_map<int, string> u = {
{1, "1"},
{3, "3"},
{2, "2"}
};
map<int, string> v = {
{1, "1"},
{3, "3"},
{2, "2"}
};
cout << "std::unordered_map" << endl;
for(const auto &n : u){
cout << "Key:[" << n.first << "] Value:[" << n.second << "]\n";
}
cout <<endl;
cout << "std::map" << endl;
for (const auto & n : v){
cout << "Key:[" << n.first << "] Value:[" << n.second << "]\n";
}
}
3、元组
传统c++中的容器,除了std::pair外,似乎没有现成的结构能够用来存放不同类型的数据。但std::pair的缺陷是显而易见的,只能保存两个元素。
元组基本操作
三个核心函数:
1、std::make_tuple: 构造元组
2、std::get:获得元组某个位置的值
3、std::tie:元组拆包
#include <iostream>
#include <tuple>
using namespace std;
auto get_student(int id){
switch (id)
{
case 0:
return make_tuple(3.8, 'A', "张三");
break;
case 1:
return make_tuple(2.9, 'C', "李四");
break;
case 2:
return make_tuple(1.7, 'D', "王五");
break;
default:
return make_tuple(0.0, 'D', "null");
break;
}
}
int main(){
auto student = get_student(0);
std::cout << "ID: 0, "
<< "GPA: " << get<0>(student) << ", "
<< "成绩:" << get<1>(student) << ", "
<< "姓名:" << get<2>(student) << "\n";
double gpa;
char grade;
string name;
//元祖进行拆包
tie(gpa, grade, name) = get_student(1);
std::cout << "ID: 1, "
<< "GPA: " << gpa << ", "
<< "成绩:" << grade << ", "
<< "姓名:" << name << "\n";
return 0;
}
std::get除了使用常量获取元组对象外,c++14增加了使用类型来获取元组中的对象:
std::tuple<std::string, double, double, int> t("123", 4.5, 6.7, 8);
std::cout << std::get<std::string>(t) << std::endl;
std::cout << std::get<double>(t) << std::endl; //非法,引发编译期错误
std::cout << std::get<int>(t) << std::endl;
运行期索引
std::get<>依赖一个编译期的常量,所以下面的方式是不合法的:
int index = 1;
std::get<index>(t); //非法
c++17引入了std::variant<>,提供给variant<>的类型模版参数 可以让一个variant<>从而容纳提供的几种类型的变量(在其他语言,例如Python/JavaScrpit等,表现为动态类型):
#include <variant>
template <size_t n, typename... T>
constexpr std::variant<T...> _tuple_index(const std::tuple<T...>& tpl, size_t i){
if constexpr(n >= sizeof...(T))
throw std::out_of_range("越界.");
if(i == n)
return std::variant<T...>{
std::in_place_index<n>, std::get<n>(tpl)
};
return _tuple_index<(n < sizeof...(T)-1 ? n+1 : 0)>(tpl, i);
}
template <typename... T>
constexpr std::variant<T...> tuple_index(const std::tuple<T...>& tpl, size_t i){
return _tuple_index<0>(tpl, i);
}
template <typename T0, typename ... TS>
std::ostream & operator<< (std::ostream & s, std::variant<T0, TS...> const & v){
std::visit([&](auto && x){s<<x;}, v);
return s;
}
这样我们就能:
int i = 1;
std::cout << tuple_index(student, i) << endl;
元组合并与遍历
还有一个常见的需求就是合并两个元组,这可以通过std::tuple_cat来实现:
auto new_tuple = std::tuple_cat(get_student(1), std::move(t));
要遍历首先我们需要知道一个元组的长度,可以:
template <typename T>
auto tuple_len(T &tpl){
return std::tuple_size<T>::value;
}
这样就能够对元组进行迭代了:
//迭代
for(int i = 0; i != tuple_len(new_tuple); ++i){
//运行期索引
std::cout << tuple_index<i, new_tuple) << std::endl;
}
总结
std::tuple虽然有效,但是标准库提供的功能有限,没办法满足运行期索引和迭代的需求,好在我们还有其他办法可以自行实现。
现代C++教程:高速上手(四)-容器的更多相关文章
- 【OpenCV新手教程之十四】OpenCV霍夫变换:霍夫线变换,霍夫圆变换合辑
本系列文章由@浅墨_毛星云 出品,转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/26977557 作者:毛星云(浅墨) ...
- 【OpenCV入门教程之十四】OpenCV霍夫变换:霍夫线变换,霍夫圆变换合辑
http://blog.csdn.net/poem_qianmo/article/details/26977557 本系列文章由@浅墨_毛星云 出品,转载请注明出处. 文章链接:http://blog ...
- SpringBoot进阶教程(六十四)注解大全
在Spring1.x时代,还没出现注解,需要大量xml配置文件并在内部编写大量bean标签.Java5推出新特性annotation,为spring的更新奠定了基础.从Spring 2.X开始spri ...
- Senparc.Weixin.MP SDK 微信公众平台开发教程(十四):请求消息去重
为了确保信息请求消息的到达率,微信服务器在没有及时收到响应消息(ResponseMessage)的情况下,会多次发送同一条请求消息(RequestMessage),包括MsgId等在内的所有文本内容都 ...
- 《ArcGIS Engine+C#实例开发教程》第四讲 状态栏信息的添加与实现
原文:<ArcGIS Engine+C#实例开发教程>第四讲 状态栏信息的添加与实现 摘要:在上一讲中,我们完成了 MapControl 和PageLayoutControl两种视图的同步 ...
- iOS 11开发教程(十四)iOS11应用代码添加视图
iOS 11开发教程(十四)iOS11应用代码添加视图 如果开发者想要使用代码为主视图添加视图,该怎么办呢.以下将为开发者解决这一问题.要使用代码为主视图添加视图需要实现3个步骤. (1)实例化视图对 ...
- struts2官方 中文教程 系列十四:主题Theme
介绍 当您使用一个Struts 2标签时,例如 <s:select ..../> 在您的web页面中,Struts 2框架会生成HTML,它会显示外观并控制select控件的布局.样式和 ...
- Linux Shell系列教程之(四)Shell注释
本文是Linux Shell系列教程的第(四)篇,更多shell教程请看:Linux Shell系列教程 与许多的编程语言一样,Shell中也有注释符号,今天就为大家来介绍下Shell中的注释的语法及 ...
- 《Visual C++ 2010入门教程》系列四:VC2010中初学者常见错误、警告和问题
<Visual C++ 2010入门教程>系列四:VC2010中初学者常见错误.警告和问题 这一章将帮助大家解释一些常见的错误.警告和问题,帮助大家去理解和解决一些常见问题,并了解它的 ...
随机推荐
- 收集邮票 (概率dp)
收集邮票 (概率dp) 题目描述 有 \(n\) 种不同的邮票,皮皮想收集所有种类的邮票.唯一的收集方法是到同学凡凡那里购买,每次只能买一张,并且买到的邮票究竟是 \(n\) 种邮票中的哪一种是等概率 ...
- Chrome扩展应用Postman地址(直接搜是搜不到的)
https://chrome.google.com/webstore/detail/postman/fhbjgbiflinjbdggehcddcbncdddomop
- golang []byte 和 string相互转换
原文链接:golang []byte和string相互转换 测试例子 package main import ( "fmt" ) func main() { str2 := &qu ...
- cinder 卷迁移进度的代码分析
一.cinder-api服务的入口函数 D:\code-program\cinder-ocata_cinder\cinder\api\contrib\admin_actions.py from cin ...
- Java算法——回溯法
回溯法一种选优搜索法,又称试探法.利用试探性的方法,在包含问题所有解的解空间树中,将可能的结果搜索一遍,从而获得满足条件的解.搜索过程采用深度遍历策略,并随时判定结点是否满足条件要求,满足要求就继续向 ...
- Java多线程_生产者消费者模式1
生产者消费者模型 具体来讲,就是在一个系统中,存在生产者和消费者两种角色,他们通过内存缓冲区进行通信,生产者生产消费者需要的资料,消费者把资料做成产品.生产消费者模式如下图.(图片来自网络 ...
- 焦大:seo如何快速理解谷歌PR的计算奥秘
http://www.wocaoseo.com/thread-248-1-1.html 前不久看到一个搞笑的说法,有人问谷歌PR的计算是PR值=0.15+0.85*(A网页传递值+B网页传递值--), ...
- CSS布局中浮动问题的四种解决方案
一.起因: 子盒子设置浮动之后效果: 由此可见,蓝色的盒子设置浮动之后,因为脱离了标准文档流,它撑不起父盒子的高度,导致父盒子高度塌陷.如果网页中出现了这种问题,会导致我们整个网页的布局紊乱 二.解决 ...
- 关于提高服务器的带宽策略bonding
一:bonding的概念 所谓bonding就是将多块网卡绑定同一IP地址对外提供服务,可以实现网卡的带宽扩容.高可用或者负载均衡. 二:bonding的优势 1 网络负载均衡 2 提高带宽网络传输效 ...
- 社区布道师揭秘消息中间件技术内幕,撰写MQ架构设计与实现原理
RocketMQ是什么 RocketMQ是由阿里捐赠给Apache的一款分布式.队列模型的开源消息中间件,经历了淘宝双十一的洗礼. RocketMQ的特性 RocketMQ基本概念 Client端 P ...