关于两个的区别,首先size()==0为bool表达式,empty()为函数调用,这一点很明显。查看源代码,

    bool empty() const { return _M_node->_M_next == _M_node; }  

    size_type size() const {
size_type __result = ;
distance(begin(), end(), __result);
return __result;
}

可以看出empty直接检查标记节点,而size是通过求首尾迭代器的距离来获取元素个数的。

查看的源代码来自http://www.sgi.com/tech/stl/download.html

Effective STL中文版:50条有效使用STL的经验

对任一容器c ,下面的代码
if (c.size() == 0)…

本质上与
if (c.empty())…

是等价的。既然如此,你或许会疑惑为什么要偏向于某一种形式,尤其是考虑到empty 通常被
实现为内联函数(inline function ),并且它所做的仅仅是返回 size 是否为0。

你应该使用empty 形式,理由很简单:empty 对所有的标准容器都是常数时间操作,而对一些list 实现,size 耗费线性时间。

到底是什么使list 这么讨厌呢?为什么它不也提供常数时间的size 呢?答案在于list 所独有的链接(splice)操作。考虑如下代码:
list<int> list1;
list<int> list2;

list1.splice( //把list2中从第一个含5 的节点
    list1.end(), list2,  //到最后一个含10的所有节点
    find(list2.begin(), list2.end(), 5), //移动到list1的末尾
    find(list2.rbegin(), list2.rend(), 10).base()
 //关于base() 调用的信息,见第28条
);

这段代码只有当list2 中在含5
的节点之后有含10的节点时才工作。假定这不是一个问题,而把注意力集中在下面这个问题上:链接后的list1
中有多少个元素?很明显,链接后的list1 中的元素个数是它链接前的元素个数加上链接过来的元素个数。但有多少个元素被链接过来了
呢?应该与find(list2.begin(), list2.end(), 5)和find(list2.rbegin(),
list2.rend(), 10). base()
所定义的区间中的元素个数一样多。好,那究竟是多少个呢?如果不遍历该区间来数一数,你是没法知道的。问题就在这里。

假定由你来负责实现list。list 不仅是容器,而且是标准容器,它将会被广泛使用。你自然会希望自己的实现尽量高效。你发现用户通常希望知道
list 中有多少个元素,所以你想使 size 成为常数时间操作。因此,你希望设计list,使它总知道自己含有多少个元素。

同时,你知道在所有的标准容器中,只有list 具有把元素从一处链接到另一处而不需要拷贝任何数据的能力。你推断,许多 list
的客户之所以选择它,是因为它提供了高效的链接操作。他们知道把一个区间从一个list 链接到另一个list
可以通过常数时间来完成。你很清楚他们知道这一点,所以你当然想满足他们的期望,使splice成为常数时间的成员函数。

这将使你左右为难。如果 size 是常数时间操作,那么 list 的每个成员函数都必须更新它们所操作的链表的大小(size ),当然也包括
splice。可是 splice更新它所改变的链表的大小的唯一方式是计算所链接的元素的个数,而这会使
splice不具有你所希望的常数时间操作性能。如果你不要求splice更新它所改变的链表的大小,则splice可以成为常数时间操作,可是这时size
会成为线性时间操作。通常,它需要遍历自己的整个数据结构来看一看自己含有多少个元素。不管你怎么看,某些方面——list
或splice——必须做出让步。其中的一个可以成为常数时间操作,但不可能二者都是。

不同的链表实现通过不同的方式解决这一冲突,具体方式取决于作者选择把 size 还是splice实现得最为高效。如果你使用的list
实现恰好是把splice的常数时间操作放在第一位的,那么你使用empty 而不是size 会更好些,因为empty
操作总是花费常数时间。即使现在你使用的list
实现不是这种方式,将来你也可能会发现自己在使用这样的实现。比如,你可能把自己的代码移植到不同的平台上,该平台上的 STL
采用了不同的实现;又如,你可能决定切换到当前平台上的不同的STL 实现。

不管发生了什么,调用empty 而不是检查size==0 是否成立总是没错的。所以,如果你想知道容器中是否含有零个元素,请调用empty。

 
 
 

[转载] C++ STL中判断list为空,size()==0和empty()有什么区别的更多相关文章

  1. (转)Java 中关于String的空对象(null) ,空值(empty),空格

    原文出处:Java 中关于String的空对象(null) ,空值(empty),空格 定义 空对象: String s = null; 空对象是指定义一个对象s,但是没有给该对象分配空间,即没有实例 ...

  2. PHP中判断变量为空的几种方法

    判断变量为空,在许多场合都会用到,同时自己和许多新手一样也经常会犯一些错误, 所以自己整理了一下PHP中一些常用的.判断变量为空的方法. 1. isset功能:判断变量是否被初始化本函数用来测试变量是 ...

  3. c#中判断对象为空的几种方式(字符串等)

    (1)先了解几个与空类型相关的关键字和对象  Null : 关键字表示不引用任何对象的空引用,它是所有引用类型变量的默认值,在2.0版本之前也就只有引用变量类型可以为null,如(string a=n ...

  4. mysql中判断字段为空

    mysql中判断字段为null或者不为null   在mysql中,查询某字段为空时,切记不可用 = null, 而是 is null,不为空则是 is not null   select nulco ...

  5. (转载) STL中map用法详解

    Map是STL的一个关联容器,它提供一对一(其中第一个可以称为关键字,每个关键字只能在map中出现一次,第二个可能称为该关键字的值)的数据处理能力,由于这个特性,它完成有可能在我们处理一对一数据的时候 ...

  6. Java中判断是否为空的方法

    1.判断字符串或者对象是否为空 首先来看一下工具StringUtils的判断方法: 一种是org.apache.commons.lang3包下的: 另一种是org.springframework.ut ...

  7. PHP中判断变量为空的几种方法小结

    isset  主要用来判断变量是否被初始化过empty  可以将值为 "假"."空"."0"."NULL"." ...

  8. (转载)java中判断字符串是否为数字的方法的几种方法

    java中判断字符串是否为数字的方法: 1.用JAVA自带的函数 public static boolean isNumeric(String str){ for (int i = 0; i < ...

  9. shell中判断一个变量是否为0或者为某个具体的值

    需求说明: 在实际写脚本的过程中,需要判断某个变量的值是否为某个数字, 比如,判断某个进程的数量是否为0用来确定进程是否存在,这样的情况. 简单来说,算术比较. 测试过程: 通过以下的脚本来判断mys ...

随机推荐

  1. 1.1.1 vue-cli脚手架工具

    参考文档: windows下npm安装vue(以下教程大部分都是参考这篇博客的,按照着这篇博客自己实现了一遍) npm安装vue-cli脚手架 一.前言 npm:nodejs下的包管理器,安装好nod ...

  2. [原][osg][osgEarth]EarthManipulator关于oe漫游器的handle部分解读以及修改(仿照谷歌,修改oe漫游器中focal(视角切换)功能 续 二)

    bool EarthManipulator::handle(const osgGA::GUIEventAdapter& ea, osgGA::GUIActionAdapter& aa) ...

  3. C#6.0 语法

    属性表达式 属性值初始化 public string name {get;set;} = "张三"; 函数表达式 NULL检查运算符 var aa = Created?.Date; ...

  4. 力扣(LeetCode)728. 自除数

    自除数 是指可以被它包含的每一位数除尽的数. 例如,128 是一个自除数,因为 128 % 1 == 0,128 % 2 == 0,128 % 8 == 0. 还有,自除数不允许包含 0 . 给定上边 ...

  5. 力扣(LeetCode)832. 翻转图像

    给定一个二进制矩阵 A,我们想先水平翻转图像,然后反转图像并返回结果. 水平翻转图片就是将图片的每一行都进行翻转,即逆序.例如,水平翻转 [1, 1, 0] 的结果是 [0, 1, 1]. 反转图片的 ...

  6. centos7在upgrade的时候显示:Delta RPMs disabled because /usr/bin/applydeltarpm not installed

    前面安装信息,太多,省略 总计:113 MIs this ok [y/d/N]: yDownloading packages:Delta RPMs disabled because /usr/bin/ ...

  7. Dreamweaver 1 网页制作

    1.站点 1.1 创建站点 点击菜单栏中站点进行站点创建,输入站点名称,路径 1.2 设置图像文件夹 1.3 站点管理 站点的编辑.复制.删除 2.页面属性栏 2.1 外观 1.设置页面整体的字体.大 ...

  8. MarkerOpter marker操作类

    构造函数:MarkerOpter=function(p_params): p_params={} 参数描述: p_params.layer; // markerlayer p_params.imgUr ...

  9. Linux 安装SSH

    ●centOS/redhat安装SSH 查询openssh server服务状态:systemctl status sshd 安装sshd命令: yum install openssh-server ...

  10. 关于MySQL大量数据分页查询优化

    select * form user id in(select id from user limit 1000000,10);