C++ std-11 常用方法
- 对多个值取最值
C++标准库提供了获取最大值和最小值的方法:
int mi = std::min(x1, x2);
int ma = std::max(x1, x2);
如果想获取超过两个数的最值呢?
可以使用嵌套的方法,比如这样:
int mi = std::min(x1, std::min(x2, std::min(x3, x4)));
这样写有点麻烦,下面是更简单的方法:
int mi = std::min({x1, x2, x3, x4})
- 万能头文件
使用 C++ 进行开发时,每用到一个库函数,就需要在代码中 include 该库函数的头文件,比如 string, vector, fstream, thread, list, algorithm......
有时候光头文件就占了十多行,看起来比较冗长,这里有个万能头文件 bits/stdc++.h,它基本上包括所有的 C++ 标准库,所以在代码中只需要包含这一个头文件即可。
#include <bits/stdc++.h>
3.auto 关键字
使用 auto 关键字可以让编译器在编译时自动推导出变量的数据类型,而不需要你手动指定。
还有使用容器的场景,比如 map<string, vector<pair<int, int>>> 之类的数据类型,使用 auto 就非常方便。
4.Lambda 表达式
C++11 引入了 Lambda 表达式,可以实现匿名函数,一种没有函数名的函数对象,并且它基于一些简洁的语法可以在各种其他作用域内捕获变量。
如果你需要在代码内部完成一些小的快速操作,没有必要为此编写一个单独的函数,就可以使用 Lambda,一个很常用的场景是将它用在排序函数中。
在使用 sort 函数时,想自定义排序规则,则需要编写一个比较器函数来实现该规则,再将该函数传入 sort 函数中。
比如根据数字的个位数进行排序:
#include <vector>
#include <algorithm>
bool cmp(int a, int b){
// 按个位数从大到小排序
return (a % 10) > (b % 10);
}
int main() {
std::vector<int> v;
v = {1, 53, 17, 25};
// sort
std::sort(v.begin(), v.end(), cmp);
}
// output: [17 25 53 1]
通过 Lambda 表达式,就不用显式编写 cmp 函数。
std::sort(v.begin(), v.end(), [](int a,int b){return (a % 10) > (b % 10);});
可能你还不太会使用 Lambda,基本形式如下:
[](argument1,argument2,.....){//code}
在 () 中传入参数,在 {} 中编写代码,[] 是一个捕获列表,可以指定外部作用域中,可以使用的局部变量:
- [] — 捕获列表为空,表示在 lambda 表达式中不能使用任何外部作用域的局部变量,只能使用传进去的参数。
- [=] — 表示按照值传递的方法捕获父作用域的所有变量。
- [&] — 表示按照引用传递的方法捕获父作用域的所有变量。
- [this] — 在成员函数中,按照值传递的方法捕获 this 指针。
- [a, &b] — 不建议直接使用 [=] 或 [&] 捕获所有变量,可以按需显式捕获,就是在 [] 指定变量名,[a] 表示值传递,[&b] 表示引用传递。
5.用 emplace_back 代替 push_back
在 C++11 中,emplace_back 的用法功能与 push_back 类似,都是在向量的末尾添加元素。
但 emplace_back 比 push_back 更有效率,因为 push_back 会先创建一个临时变量(构造函数),然后将其拷贝到 vector 的末尾(拷贝构造函数)。但 emplace_back 就直接在 vector 的末尾构造值(构造函数),少调用了一次拷贝构造函数。
两者的用法相同:
#include <vector>
std::vector<int> v;
v.push_back(0);
v.emplace_back(1);
6.使用 tuple
对于两个不同类型的元素可以用:
std::pair<int, char> pi = std::make_pair(1, 'a');
std::cout << pi.first << " " << pi.second << std::endl;
如果有超过两个元素,且数据类型不同,可以使用 pair 的嵌套,使用时需要不断地调用 first 和 second。
std::pair<int, std::pair<char, std::string>> pi = std::make_pair(1, std::make_pair('a', "he"));
std::cout << pi.first << " " << pi.second.first << std::endl;
这种用法比较繁琐,也可以使用 tuple 进行简化,tuple 是各种数据类型的集合,相当于一个形式更简单的结构体。
#include <tuple>
std::tuple<int, char, std::string> tp = std::make_tuple(1, 'a', "he");
std::cout << std::get<0>(tp) << "," << std::get<1>(tp) << "," << std::get<2>(tp) << std::endl;
7.for 循环的简便写法
当遍历 vector 中每个元素时,我之前的写法:
std::vector<int> s = {8, 2, 3, 1};
for (std::vector<int>::iterator it = s.begin(); it != s.end(); ++it) {
std::cout << *it << ' ';
}
C++11 的 for 循环支持下面这种写法:
std::vector<int> v = {8, 2, 3, 1};
for (int i: v) {
std::cout << i << ' ';
}
其中 i 就是 vector 中的每个元素,还可以配合 auto 关键字,食用更佳。
8.**不用 for 循环,计算一个数字的位数 **
#include <iostream>
#include <cmath>
int main() {
// cmath
int N=105050;
int n= std::floor(std::log10(N))+1;
std::cout << n;
return 0;
}
9.不用 for 循环,对数组进行深拷贝
使用 copy_n 函数可以从源容器复制指定个数的元素到目的容器中。
#include <algorithm>
int x[3]={1,2,3};
int y[3];
std::copy_n(x, 3, y);
copy_n 有三个参数:
- 第一个参数是指向第一个源元素的输入迭代器;
- 第二个参数是需要复制的元素的个数;
- 第三个参数是指向目的容器的第一个位置的迭代器。
10.all_of, any_of, none_of
在 Python 中,有 all() 和 any() 来判断列表中的元素是否都满足条件,或者至少一个元素满足条件。
C++ 也能实现类似的功能,就是使用 all_of,any_of,none_of。这里需要用到前面提到的 Lambda 表达式。
判断数组所有元素是否都为正(all_of):
#include <algorithm>
#include <iostream>
int main() {
int ar[6] = {1, 2, 3, 4, 5, -6};
std::all_of(ar, ar+6, [](int x){ return x>0; })?
std::cout << "All are positive elements" :
std::cout << "All are not positive elements";
return 0;
}
判断数组是否至少有一个元素为正(any_of):
std::any_of(ar, ar+6, [](int x) { return x>0; })?
std::cout << "At least one are positive elements" :
std::cout << "all are not positive elements";
判断是否没有元素为正(none_of):
std::none_of(ar, ar+6, [](int x) { return x>0; })?
std::cout << "all are not positive elements" :
std::cout << "At least one are positive elements";
11.获取元祖多个值
在 Python 中,可以使用下面方法同时获取多个值:
arr = (1, 2.0, "3")
a, b, c = arr
现在 C++11 也可以实现这个功能了,用到了 std::tie:
#include <string>
#include <tuple>
#include <iostream>
int main() {
std::tuple<int, double, std::string> t0 = {1, 2.0, "3"};
int i = 0; double d = 0.; std::string s = "";
std::tie(i, d, s) = t0;
std::cout << i << " " << d << " " << s << std::endl;
return 0;
}
代码中定义了元祖 t0,通过 std::tie(i, d, s)= t0; 同时获取了元祖中的三个值。
有时候想只获取元祖中的部分值,则需要用到 std::ignore, 比如只获取前两个值:
std::tie(i, d, td::ignore) = t0;
std::tie 不但可用于 tuple,还能用于 pair。
12.数组去重
在 python 中对列表去重,有一个简单方法,就是将列表转换为集合 set,其实 C++ 也有类似的操作。
#include <set>
#include <vector>
int main()
{
std::vector<int> vec;
vec = {1, 2, 4, 5, 1, 6, 1, 2};
std::set<int> s(vec.begin(), vec.end());
vec.assign(s.begin(), s.end());
return 0;
}
C++ std-11 常用方法的更多相关文章
- std::map常用方法
map<string, int> Employees; Employees["Mike C."] = 12306; Employees.insert(make_pair ...
- C++编程优化心得(持续更新)
1. 对齐原则.比如64位总线,每次寻址读取8B.编程时注意变量地址,尽量消耗总线最少的寻址次数.堆内存申请时,系统严格按照对齐原则分配,故而使用时候也尽量不要跨寻址边界. 2. 需要的时候,可为了效 ...
- UDP 单播、广播和多播
阅读目录(Content) 一.UDP广播 二.UDP多播 1.多播(组播)的概念 2.广域网的多播 三.UDP广播与单播 广播与单播的比较 使用UDP协议进行信息的传输之前不需要建议连接.换句话说就 ...
- Select the best path in a matrix
Amazon interview question: Given a 2-dimensional array with arbitrary sizes and contains random posi ...
- RFC 2616
Network Working Group R. Fielding Request for Comments: 2616 UC Irvine Obsoletes: 2068 J. Gettys Cat ...
- jsoncpp第二篇------API
更多API参考jsoncpp头文件 1 jsoncpp的api简要说明 1,解析(json字符串转为对象) std::string strDataJson; Json::Reader JReader ...
- 借网站日记分析~普及一下Pandas基础
对网站日记分析其实比较常见,今天模拟演示一下一些应用场景,也顺便说说Pandas,图示部分也简单分析了下 1.数据清洗¶ 一般数据都不可能直接拿来用的,或多或少都得清理一下,我这边就模拟一下清洗完 ...
- 【OpenCV】选择ROI区域 (转)
问题描述:在测试目标跟踪算法时,需要选择不同区域作为目标,进行目标跟踪,测试目标跟踪的效果. 解决思路: 1.OpenCV中提供了鼠标交互控制,利用setMouseCallback()给固定的窗口设置 ...
- octomap的简介
装载自高翔博士的博客:https://www.cnblogs.com/gaoxiang12/p/5041142.html 什么是octomap? RGBD SLAM的目的有两个:估计机器人的轨迹,并建 ...
- 最大流(EK)
最大流 — Edmond Karp算法 Edmond Karp算法的大概思想: 反复寻找源点s到汇点t之间的增广路径,若有,找出增广路径上每一段[容量-流量]的最小值delta,若无,则结束. 在寻找 ...
随机推荐
- E: Unable to fetch some archives, maybe run apt-get update or try with --fix-missing
解决办法:apt-get update或者apt-get cleanapt-get update 或者 apt-get update --fix-missing问题解析1 source本身的问题 根据 ...
- Output of C++ Program | Set 15
Predict the output of following C++ programs. Question 1 1 #include <iostream> 2 using namespa ...
- Servlet(2):通过servletContext对象实现数据共享
一,ServletContext介绍 web容器在启动时,它会为每一个web应用程序都创建一个ServletContext对象,它代表当前web应用 多个Servlet通过ServletContext ...
- 第7章 使用性能利器——Redis
在现今互联网应用中,NoSQL已经广为应用,在互联网中起到加速系统的作用.有两种NoSQL使用最为广泛,那就是Redis和MongoDB.本章将介绍Redis和Spring Boot的结合.Redis ...
- 【Java基础】JAVA中优先队列详解
总体介绍 优先队列的作用是能保证每次取出的元素都是队列中权值最小的(Java的优先队列每次取最小元素,C++的优先队列每次取最大元素).这里牵涉到了大小关系,元素大小的评判可以通过元素本身的自然顺序( ...
- 使用匿名内部类和lamda的方式创建线程
1.匿名内部类的方式 1 /** 2 *匿名内部类的方式启动线程 3 */ 4 public class T2 { 5 public static void main(String[] args) { ...
- 了解C#的Expression
我们书接上文,我们在了解LINQ下面有说到在本地查询IEnumerbale主要是用委托来作为传参,而解析型查询 IQueryable则用Expression来作为传参: public static I ...
- 三维引擎导入obj模型不可见总结
最近有客户试用我们的三维平台,在导入模型的时候,会出现模型全黑和不可见的情况.上一篇文章说了全黑的情况.此文说下不可见的情况. 经过测试,发现可能有如下两种情况. 导入的模型不在镜头视野内 导入的模型 ...
- tableau绘制饼图
一.将类别拖拽至列,将销售额拖拽至行 二.点击右上角智能显示选择饼图 三.拖拽销售额至标记卡,右键快速表计算-合计百分比-细节处理最终结果如下图所示
- 替DateDif哭诉一把(Excel函数集团)
Excel中有个工作表函数DateDif,专门用来计算两日期之间的日差.月差.年差,传说十分好用. 具体用法在此就省略了,好奇的童鞋请自行*度~ 可是,在Excel里,他却是个"没户口&qu ...