【赛道动量图源码】【edux核心源码分析】【求源码的东西】list erase源码

时间:2025-01-28 01:06:47 编辑:飞鸟ssc源码 来源:unity游戏源码分享

1.从应用到源码理解STL反向迭代器
2.dat文件用什么软件打开
3.rpm命令手册和查看rpm安装包的安装路径的方法
4.x,y 都是int, x++=y++ 为什么错?

list erase源码

从应用到源码理解STL反向迭代器

       在实际应用中,我们可能需要从序列容器(如vector)的尾部移除不满足特定条件的部分元素。这通常涉及从尾部开始的迭代操作。然而,容器成员函数erase不接受反向迭代器作为参数。因此,赛道动量图源码我们需要将反向迭代器转换为普通迭代器。先来看看STL迭代器的分类和转换关系。

       STL迭代器主要分为用途迭代器,它们之间存在转换关系,但不是所有迭代器类型都可以相互转换。转换关系需通过迭代器的构造函数定义,有些可以直接转换,有些则需调用特定方法。

       特别地,反向迭代器到普通迭代器的转换可以通过调用反向迭代器的base()方法实现。但初版代码存在缺陷,未能按预期将元素正确删除。通过跟踪代码并参考cpp reference文档,我们发现base()方法返回的迭代器实际上比预期位置靠后一个元素。

       为了修正这个问题,我们需要将通过base()方法得到的迭代器向前移动一个位置,以正确指向第一个符合移除条件的元素。修改代码后,可以确保元素按约定进行删除。

       在一般场景下,迭代器的使用主要涉及遍历访问和遍历修改元素值。对于删除和插入操作,可能需要将反向迭代器转换为普通迭代器。STL容器的erase和insert成员函数仅接受普通迭代器作为参数。

       在执行插入操作时,直接使用base()将反向迭代器转换为普通迭代器,并传入insert函数,其语义是一致的。而在删除操作中,直接使用base()转换后的迭代器可能无法正确执行,因为反向迭代器和普通迭代器在终止位置上的处理存在差异。为了修正此问题,需要手动调整,确保迭代器的有效性。

       对于反向迭代器,通过正确的反向迭代操作得到的迭代器,在不等于rend()返回的迭代器时,都是edux核心源码分析指向有效值的。因此,除了rend().base()-1操作可能导致问题外,其他转换通常都是安全的。

       讨论end()迭代器的前移操作时,需要考虑是否能够安全地访问容器的尾端元素。对于随机访问迭代器,如vector容器,end()返回的迭代器可以进行前移操作,但需确保移动操作的合法性。对于双向访问迭代器如list,同样可以进行前移操作以访问尾端元素。

       结束讨论前,还需要确认iterator的-1操作是否对指向容器尾端元素的迭代器有效。在vector容器中,通过end成员函数返回的迭代器通过-1操作可以得到指向尾端元素的普通迭代器。对于list容器,其end成员函数返回的迭代器也支持前移操作。

       总结来说,支持向前移动操作的迭代器访问容器内元素的容器,其end成员函数通过前移操作可以得到一个指向容器尾端元素的迭代器。这符合双向迭代器的设定语义。通过反向迭代器的原理,我们也能验证end()函数返回的迭代器可以进行反向移动。

dat文件用什么软件打开

       可以直接使用系统自带的media play播放工具打开。以下是具体的操作方法:

       1、首先打开media play播放器,按下ALT键,弹出菜单快捷键,选择文件——打开,然后选择DAT文件。

       2、在弹出菜单中点击打开后,定位到音频或者视频文件的目录下面。

       3、系统默认显示的是视频文件的格式,DAT文件是不显示的,这个时候我们需要手动的选择文件类型为所有类型,这样才会显示DAT文件。

       4、如果DAT文件比较大或者比较复杂的话,播发器需要解码,所以打开时会需要一段时间,播发器会提示正在打开媒体,求源码的东西解码完后会直接打开媒体文件如下。

       4、正常打开DAT文件后,自动播放,如下所示,表名已经播放完媒体文件,证明这个文件可以正常打开了。

rpm命令手册和查看rpm安装包的安装路径的方法

       rpm -qpl xxxxxx.rpm

       1.如何安装rpm软件包

           rmp软件包的安装可以使用程序rpm来完成。执行下面的命令

           rpm -i your-package.rpm

           其中your-package.rpm是你要安装的rpm包的文件名,一般置于当前目录下。

           安装过程中可能出现下面的警告或者提示:

           ... conflict with ... 可能是要安装的包里有一些文件可能会覆盖现有

           的文件,缺省时这样的情况下是无法正确安装的可以用

           rpm --force -i 强制安装即可

           ... is needed by ...

           ... is not installed ... 此包需要的一些软件你没有安装可以用

           rpm --nodeps -i 来忽略此信息

           也就是说,rpm -i --force --nodeps 可以忽略所有依赖关系和文件问题,什么包

           都能安装上,但这种强制安装的软件包不能保证完全发挥功能

           2.如何安装.src.rpm软件包

           有些软件包是以.src.rpm结尾的,这类软件包是包含了源代码的rpm包,在安装时

           需要进行编译。这类软件包有两种安装方法,

           方法一:

           1.执行rpm -i your-package.src.rpm

           2. cd /usr/src/redhat/SPECS

           3. rpmbuild -bp your-package.specs 一个和你的软件包同名的specs文件

           4. cd /usr/src/redhat/BUILD/your-package/ 一个和你的软件包同名的目录

           5. ./configure 这一步和编译普通的源码软件一样,可以加上参数

           6. make

           7. make install

       方法二:

       1.执行rpm -i you-package.src.rpm

       2. cd /usr/src/redhat/SPECS

           前两步和方法一相同

       3. rpmbuild -bb your-package.specs 一个和你的软件包同名的specs文件

           这时,在/usr/src/redhat/RPM/i/ (根据具体包的不同,也可能是i,noarch等等)

           在这个目录下,有一个新的rpm包,这个是编译好的二进制文件。

           执行rpm -i new-package.rpm即可安装完成。

       3.如何卸载rpm软件包

           使用命令 rpm -e 包名,包名可以包含版本号等信息,但是不可以有后缀.rpm

           比如卸载软件包proftpd-1.2.8-1,可以使用下列格式:

           rpm -e proftpd-1.2.8-1

           rpm -e proftpd-1.2.8

           rpm -e proftpd-

           rpm -e proftpd

           不可以是下列格式:

           rpm -e proftpd-1.2.8-1.i.rpm

           rpm -e proftpd-1.2.8-1.i

           rpm -e proftpd-1.2

           rpm -e proftpd-1

           有时会出现一些错误或者警告:

           ... is needed by ... 这说明这个软件被其他软件需要,不能随便卸载

           可以用rpm -e --nodeps强制卸载

           4.如何不安装但是获取rpm包中的文件

           使用工具rpm2cpio和cpio

           rpm2cpio xxx.rpm | cpio -vi

           rpm2cpio xxx.rpm | cpio -idmv

           rpm2cpio xxx.rpm | cpio --extract --make-directories

           参数i和extract相同,表示提取文件。v表示指示执行进程

           d和make-directory相同,表示根据包中文件原来的路径建立目录

           m表示保持文件的更新时间。

       5.如何查看与rpm包相关的文件和其他信息

       下面所有的例子都假设使用软件包mysql-3..a-

           1.我的系统中安装了那些rpm软件包

           rpm -qa 讲列出所有安装过的包

           如果要查找所有安装过的包含某个字符串sql的软件包

           rpm -qa |grep sql

           2.如何获得某个软件包的文件全名

           rpm -q mysql 可以获得系统中安装的mysql软件包全名,从中可以获得

           当前软件包的版本等信息。这个例子中可以得到信息mysql-3..a-

           3.一个rpm包中的文件安装到那里去了?

           rpm -ql 包名

           注意这里的是不包括.rpm后缀的软件包的名称

           也就是说只能用mysql或者mysql-3..a-而不是mysql-3..a-.rpm。

           如果只是想知道可执行程序放到那里去了,也可以用which,比如

           which mysql

           4.一个rpm包中包含那些文件

           一个没有安装过的软件包,使用rpm -qlp ****.rpm

           一个已经安装过的软件包,还可以使用rpm -ql ****.rpm

           5.如何获取关于一个软件包的版本,用途等相关信息?

           一个没有安装过的软件包,使用rpm -qip ****.rpm

           一个已经安装过的免税商品溯源码软件包,还可以使用rpm -qi ****.rpm

       6.某个程序是哪个软件包安装的,或者哪个软件包包含这个程序

           rpm -qf `which 程序名` 返回软件包的全名

           rpm -qif `which 程序名` 返回软件包的有关信息

           rpm -qlf `which 程序名` 返回软件包的文件列表

           注意,这里不是引号,而是`,就是键盘左上角的那个键。

           也可以使用rpm -qilf,同时输出软件包信息和文件列表

           7.某个文件是哪个软件包安装的,或者哪个软件包包含这个文件

           注意,前一个问题中的方法,只适用与可执行的程序,而下面的方法,不仅可以

           用于可执行程序,也可以用于普通的任何文件。前提是知道这个文件名。

           首先获得这个程序的完整路径,可以用whereis或者which,然后使用rpm -qf例如:

           # whereis ftptop

           ftptop: /usr/bin/ftptop /usr/share/man/man1/ftptop.1.gz

           # rpm -qf /usr/bin/ftptop

           proftpd-1.2.8-1

           # rpm -qf /usr/share/doc/proftpd-1.2.8/rfc/rfc.txt

           proftpd-1.2.8-1

           总结:

           获得软件包相关的信息用rpm -q,q表示查询query,后面可以跟其他选项,比如

           i 表示info,获得软件包的信息;

           l 表示list,获得文件列表;

           a 表示all,在所有包中执行查询;

           f 表示file,根据文件进行相关的查询;

           p 表示package,根据软件包进行查询

           需要的查询条件可以使用grep产生,或者从"` `"中的命令行产生

       6.关于rpm软件包的一些相关知识

1.什么是rpm

        rpm 即RedHat Package Management,是RedHat的发明之一

       2.为什么需要rpm

           在一个操作系统下,需要安装实现各种功能的软件包。这些软件包一般都有各自的

           程序,但是同时也有错综复杂的依赖关系。同时还需要解决软件包的版本,以及安装,

           配置,卸载的自动化问题。为了解决这些问题,RedHat针对自己的系统提出了一个

           较好的办法来管理成千上百的软件。这就是RPM管理系统。在系统中安装了rpm管理系统

           以后,只要是符合rpm文件标准的打包的程序都可以方便的安装,升级,卸载

       3.是不是所有的linux都使用rpm

           任何系统都需要包管理系统,因此很多linux都使用rpm系统。但rpm系统是河南到杭州源码为RH专门

           但是TL,Mandrake等系统也都使用rpm。由于rpm的源程序可以在别的系统上进行编译,

           所以有可能在别的系统上也使用rpm

           除了rpm,其他一些系统也有自己的软件包管理程序,例如debian的deb包,

           slakware也都有自己的包管理系统

       4.rpm包的文件名为什么那么长

           rpm包的文件名中包含了这个软件包的版本信息,操作系统信息,硬件要求等等。

           比如mypackage-1.1-2TL.i.rpm,其中mypackage是在系统中登记的软件包的名字

           1.1是软件的版本号,2是发行号,TL表示用于TL操作系统,还可能是RH等。i表示

           用于intel x平台,还可能是sparc等。

       5.软件包文件名中的i,i是什么意思

           rpm软件包的文件名中,不仅包含了软件名称,版本信息,还包括了适用的硬件架构

           的信息。

           i指这个软件包适用于intel 以上的x架构的计算机(AI)

           i指这个软件包适用于intel 以上(奔腾pro以上)的x架构的计算机(IA)

           noarch指这个软件包于硬件架构无关,可以通用。

           i软件包的程序通常针对CPU进行了优化,所以,向后兼容比较用以,i的包在

           x机器上都可以用。向前一般不兼容。不过现在的计算机,奔腾pro以下的CPU已经很少

           用,通常配置的机器都可以使用i软件包

       6.不同操作系统发行的rpm包可否混用?

           对于已经编译成二进制的rpm包,由于操作系统环境不同,一般不能混用。

           对于以src.rpm发行的软件包,由于需要安装时进行本地编译,所以通常可以在不同

           系统下安装。

       7.使用rpm时遇到的一些特殊问题

           Q 我用rpm -e **.rpm无法删除rpm包

           A 包名不要包括rpm,

           rpm -e 包名,可以包含版本号等信息,但是不可以有后缀.rpm

           Q 在MS的系统下有没有读RPM文件的工具?

           A wincmd with rpm plugins.....

           Q 是否可以通过ftp安装安装升级rpm包?

           A 可以。rpm -ivh

           ftp://xxxxxxxx/PATH2SomeRPM

       Q rpm安装时已有的包版本过高怎么办?

           A 有时由于安装的软件包太老,而系统中相关的软件包版本比较新,所以可能需要

           安装的包依赖的一些文件会找不到。这时有两种解决办法,

           第一是在系统文件中找到和需要的文件功能相同或相似的文件,做一个符号链接到

           需要的目录下。

       以下是我知道的在Linux下的rpm软件包安装时如何指定安装路径的方法,不少是从网上学来的。请大家多多指教啊。

       以下的options是可完成你想方式.

       --prefix path

             This sets the installation prefix to path for

             relocatable packages.

       --relocate oldpath=newpath

             For relocatable packages, translates the files that

             would be put in oldpath to newpath.

       --badreloc

             To be used in conjunction with --relocate, this

             forces the relocation even if the package isn't

             relocatable.

       --noorder

             Don't reorder the packages for an install. The list

             of packages would normally be reordered to satisfy

             dependancies.

RPM命令手册

       一、安装

       命令格式:

       rpm -i ( or --install) options file1.rpm ... fileN.rpm

       参数:

       file1.rpm ... fileN.rpm 将要安装的RPM包的文件名

       详细选项:

       -h (or --hash) 安装时输出hash记号 (``#'')

           --test 只对安装进行测试,并不实际安装。

           --percent 以百分比的形式输出安装的进度。

           --excludedocs 不安装软件包中的文档文件

           --includedocs 安装文档

           --replacepkgs 强制重新安装已经安装的软件包

           --replacefiles 替换属于其它软件包的文件

           --force 忽略软件包及文件的冲突

           --noscripts 不运行预安装和后安装脚本

           --prefix 将软件包安装到由 指定的路径下

           --ignorearch 不校验软件包的结构

           --ignoreos 不检查软件包运行的操作系统

           --nodeps 不检查依赖性关系

           --ftpproxy 用 作为 FTP代理

           --ftpport 指定FTP的端口号为

       通用选项

       -v 显示附加信息

           -vv 显示调试信息

           --root 让RPM将指定的路径做为"根目录",这样预安装程序和后安装程序都会安装到这个目录下

           --rcfile 设置rpmrc文件为

           --dbpath 设置RPM 资料库存所在的路径为

       二、删除

       命令格式:

       rpm -e ( or --erase) options pkg1 ... pkgN

       参数

       pkg1 ... pkgN :要删除的软件包

       详细选项

       --test 只执行删除的测试

           --noscripts 不运行预安装和后安装脚本程序

           --nodeps 不检查依赖性

       通用选项

       -vv 显示调试信息

           --root 让RPM将指定的路径做为"根目录",这样预安装程序和后安装程序都会安装到这个目录下

           --rcfile 设置rpmrc文件为

           --dbpath 设置RPM 资料库存所在的路径为

       三、升级

       命令格式

       复制代码

           

       代码如下:

       rpm -U ( or --upgrade) options file1.rpm ... fileN.rpm

       参数

       file1.rpm ... fileN.rpm 软件包的名字

       详细选项

       -h (or --hash) 安装时输出hash记号 (``#'')

           --oldpackage 允许"升级"到一个老版本

           --test 只进行升级测试

           --excludedocs 不安装软件包中的文档文件

           --includedocs 安装文档

           --replacepkgs 强制重新安装已经安装的软件包

           --replacefiles 替换属于其它软件包的文件

           --force 忽略软件包及文件的冲突

           --percent 以百分比的形式输出安装的进度。

           --noscripts 不运行预安装和后安装脚本

           --prefix 将软件包安装到由 指定的路径下

           --ignorearch 不校验软件包的结构

           --ignoreos 不检查软件包运行的操作系统

           --nodeps 不检查依赖性关系

           --ftpproxy 用 作为 FTP代理

           --ftpport 指定FTP的端口号为

       通用选项

       -v 显示附加信息

           -vv 显示调试信息

           --root 让RPM将指定的路径做为"根目录",这样预安装程序和后安装程序都会安装到这个目录下

           --rcfile 设置rpmrc文件为

           --dbpath 设置RPM 资料库存所在的路径为

       四、查询

       命令格式:

       复制代码

           

       代码如下:

       rpm -q ( or --query) options

       参数:

       pkg1 ... pkgN :查询已安装的软件包

       详细选项

       -p (or ``-'') 查询软件包的文件

           -f 查询属于哪个软件包

           -a 查询所有安装的软件包

           --whatprovides 查询提供了 功能的软件包

           -g 查询属于 组的软件包

           --whatrequires 查询所有需要 功能的软件包

       信息选项

       显示软件包的全部标识

           -i 显示软件包的概要信息

           -l 显示软件包中的文件列表

           -c 显示配置文件列表

           -d 显示文档文件列表

           -s 显示软件包中文件列表并显示每个文件的状态

           --scripts 显示安装、卸载、校验脚本

           --queryformat (or --qf) 以用户指定的方式显示查询信息

           --dump 显示每个文件的所有已校验信息

           --provides 显示软件包提供的功能

           --requires (or -R) 显示软件包所需的功能

       通用选项

       -v 显示附加信息

           -vv 显示调试信息

           --root 让RPM将指定的路径做为"根目录",这样预安装程序和后安装程序都会安装到这个目录下

           --rcfile 设置rpmrc文件为

           --dbpath 设置RPM 资料库存所在的路径为

       五、校验已安装的软件包

       命令格式:

       复制代码

           

       代码如下:

       rpm -V ( or --verify, or -y) options

       参数

       pkg1 ... pkgN 将要校验的软件包名

       软件包选项

       -p Verify against package file

           -f 校验所属的软件包

           -a Verify 校验所有的软件包

           -g 校验所有属于组 的软件包

       详细选项

       --noscripts 不运行校验脚本

           --nodeps 不校验依赖性

           --nofiles 不校验文件属性

       通用选项

       -v 显示附加信息

           -vv 显示调试信息

           --root 让RPM将指定的路径做为"根目录",这样预安装程序和后安装程序都会安装到这个目录下

           --rcfile 设置rpmrc文件为

           --dbpath 设置RPM 资料库存所在的路径为

       六、校验软件包中的文件

       语法:

       复制代码

           

       代码如下:

       rpm -K ( or --checksig) options file1.rpm ... fileN.rpm

       参数:

       file1.rpm ... fileN.rpm 软件包的文件名

       Checksig--详细选项

       --nopgp 不校验PGP签名

       通用选项

       -v 显示附加信息

           -vv 显示调试信息

           --rcfile 设置rpmrc文件为

       七、其它RPM选项

       --rebuilddb 重建RPM资料库

           --initdb 创建一个新的RPM资料库

           --quiet 尽可能的减少输出

           --help 显示帮助文件

           --version 显示RPM的当前版本

x,y 都是int, x++=y++ 为什么错?

       赋值运算需要左值,而x++不是左值。x++的本意是使用x的值,在使用后将x自加1,那么如果把x++用在赋值运算的左边的话,赋值时是赋给原先的x还是自加后的x呢?显然是未定义的,所以这种操作是错误的。

       下面转一篇关于左值和右值的文章,相信会对你帮助很大的。

       左值(lvalue)和右值(rvalue)是编程中两个非常基本的概念,但是也非常容易让人误解,看了很多文章,自我感觉真正将这个问题讲的很透彻的文章还没有看见,所以自告奋勇来尝试一下。如果左值右值的概念不是非常清楚的话,它们迟早会像拦路虎一样跳出来,让你烦心不已,就像玩电脑游戏的时候每隔一段时间总有那么几个地雷考验你的耐性,如果一次把所有地雷扫尽就好了。:)

       左值(lvalue)和右值(rvalue)最先来源于编译理论(感谢南大小百合的programs)。在C语言中表示位于赋值运算符两侧的两个值,左边的就叫左值,右边的就叫右值。比如:

       int ii = 5;//ii是左值,5是右值

       int jj = ii;//jj是左值,ii是右值

       上面表明,左值肯定可以作为右值使用,但反之则不然。左值和右值的最早区别就在于能否改变。左值是可以变的,右值不能变。注1

       注1:这一点在C++中已经猪羊变色,不再成立。拱猪游戏还是挺好玩的,我还真抓过好几次全红心,不过真的好险。:)

       在很多文章中提到,在C++中,左值更多的指的是可以定位,即有地址的值,而右值没有地址。注2

       注2:这一点仍然不准确,我在程序中生成一个临时右值std::vector(),你能够说它没有地址吗?难道它是没有肉体的鬼魂或幽灵?它是有地址的,而且它也是绝对的右值。

       在现代C++中,现在左值和右值基本上已经失去它们原本所具有的意义,对于左值表达式,通过具体名字和引用(pointer or reference)来指定一个对象。非左值就是右值。我来下一个定义:

       左值表示程序中必须有一个特定的名字引用到这个值。

       右值表示程序中没有一个特定的名字引用到这个值。

       跟它们是否可以改变,是否在栈或堆(stack or heap)中有地址毫无关系。

       1.左值

       在下面的代码中:

       int ii = 5;

       int const jj = ii;

       int a[5];

       a[0] = ;

       *(a+3) = ;

       int const& max( int const& a, int const& b ) //call by reference

       {

        return a > b ? a : b;

       }

       int& fun(int& a) //call by reference

       {

        a += 5;

        return a;

       }

       ii,jj,a[0],*(a+3),还有函数max的返回值比如max(ii, jj),注3函数fun的返回值fun(ii)都是左值。,它们都是有特定的引用名字的值。ii,jj,a[0],*(a+3),max(ii, jj),fun(ii)分别就是它们的名字。

       注3:在这里有一个不太容易分清楚的盲点。那就是有人会问max(8, 9)到达是左值还是右值,C++标准规定常量引用(reference to const)可以引用到右值,所以max(8, 9)似乎应该是右值,不过不管它是左值,还是右值,我们都不能试图去改变它。为了与前面的概念一致,我认为它是左值,不可改变的常量左值。

       左值有不能改变的,即被const所修饰的左值,比如上面的jj,max(ii, jj)都是被常量(const)魔咒所困住的左值。

       没有被const困住的左值当然是可以改变的,比如下面的代码都是成立的:

       ii = ;

       a[0] = ;

       fun(ii) = ; //OK!

       我们的眼睛没有问题,fun(ii) = ;完全正确,因为它是可以改变的左值。所以我们看STL的源码,就会理解std::vector中的重载operator[]运算符的返回值为什么要写成引用,因为operator[]必须返回左值。

       2.右值

       没有特定名字的值是右值。先看下面的代码:

       std::list();

       std::string(“It is a rvalue!”);

       int fun1() //call by value

       {

        …

       }

       int* fun2() //call by reference

       {

        …

       }

       其中std::list(),std::string(“It is a rvalue!”),函数fun1的返回值fun1(),函数fun2的返回值fun2()都是右值,它们的值都没有特定的名字去引用。也许有人会奇怪,fun2()也是右值?最前面的max(a,b)不是左值吗?

       请看清楚,函数fun2的返回值是pointer,pointer也是call by value,而函数max的返回值是reference,reference是call by reference。所以说C++中引入reference不仅仅是为了方便,它也是一种必须。注4

       注4:Scott Meyer写的《More Effective C++》的条款1专门讲了pointer和reference的区别,写的很好,辨别的非常清楚。

       fun2()是右值,但 *fun2()却是左值,就跟经常看到的*p一样,所以看C++库代码的时候,会发现重载operator*的函数返回值是reference。

       当然我还遗漏了一种右值,那就是字面上的(literal)值,比如5,8.,’a’等等理所当然的都是右值。

       右值最初出现的时候,一个最大的特征就是不可改变。但就跟我们的道德标准一样,时代不同了,标准也变化了,以前的三纲五常早已经被扔到历史的垃圾堆里面了。

       C++中有可以改变的右值,而且这个特性还非常有用。那就是用户自定义的类(class)的构造函数生成的临时对象。比如:

       std::vector(9),std::deque(),……都是可以改变的右值。在Herb Sutter的《More Exceptional C++》中的条款7的page页有这样几行代码:

       // Example 7-2(b): The right way to shrink-to-fit a vector.

       vector<Customer> c( );

       // ...now c.capacity() >= ...

       // erase all but the first elements

       c.erase( c.begin()+, c.end() );

       // the following line does shrink c's

       // internal buffer to fit (or close)

       vector<Customer>( c ).swap( c );

       // ...now c.capacity() == c.size(), or

       // perhaps a little more than c.size()

       认真看几遍,你会发现但vector的大小增大到一定程度,你又用不着这么多空间的时候,你会想办法把它收缩到最合适的大小,但利用别的办法比如调用成员函数reserve()都无法办到,这个时候就必须利用右值可以改变这个性质了。

       vector<Customer>( c ).swap( c );这行代码就是点睛之处。

       首先使用复制构造函数生成临时右值vector<Customer>( c ),这个右值正好是合适大小,然后和c交换注5,c就变成合适大小了,最后在整个表达式结束的时候,这个临时右值析构归还内存空间。真是绅士一般的优雅!

       注5:这个时候这个临时右值就发生了改变。

       如果还不理解,可以看看书,或者直接看库的源代码。

       至于为什么会这样?我思考了一下,我想是这样的,我们看类(class)的数据布置结构,会发现它的每一个数据成员都是有名字的,我想编译器在编译的过程中,都会生成一个外部不所知的对这个临时对象右值的名字引用,但需要改变这个临时对象的时候,这个名字就用上了。比如:

       class Point

       {

       public: //纯粹为了方便,我把数据成员公开,现实中尽量不要这样用

        int x, y ,z;

        ……//其他各种成员函数

       };

       我们现在就可以改变右值,用到了匿名的引用名字。

       Point().x = 6;//改变了右值

       Point().y = 6;//同意改变了右值,不过注意,这个右值跟上面的不是同一个。

       总结

       左值和右值的真正区别我想就是这些了,左值表示有特定的名字引用,而右值没有特定的名字引用。当然我仍然会有疏忽,希望大家能够提醒我,指正我的不足。

       前两天看Herb Sutter从邮件中寄来的新文章(我订阅他的新文章邮件通知),一篇是讲Tuple数据结构的,没有什么新意,以前好像看过,还有一篇名字是:(Mostly)Private,地址为/documents/s=/cujcexpsutter/ 内容本身并不深,但看完文章,发现随处可见C++的波诡云谲,又会对什么叫袖里乾坤,滴水藏海多一份感性认识。

       在下一篇文章我想从不同于一般的角度,从自己的经历谈谈在校毕业生在IT行业怎样找工作,我想会让所有读者都有一些思考,不仅仅是求职者。题目我已经想好了,就叫《扮虎吃猪》,不过现在我有一些别的事情要忙,所以可能会让大家等几天。

       转载请注明来源,谢谢!