小心使用STL中map的[]操作符

本文深入探讨了C++标准库中的map容器,特别是其下标运算符[]的使用方法及潜在陷阱。通过具体示例展示了不当使用可能导致的问题,并解析了map内部实现机制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一个map就是一个(关键码(key),值(value))对偶的序列,它提供基于关键码的快速提取操作。也就是说,可以用下标运算符[]将关键码作为下标去执行查找,并返回对应的值。因此可以把map的使用方法想象成有特殊下标的数组。在很多时候用下标运算符[]来对map中的元素进行存取是非常方便和简单的;但是,如果map下标运算符[]运用不得当,也会造成意想不到的问题。
我们知道,C++是不检查下标越界的。用超出数组下标范围的下标去访问数组元素的错误会在运行期出现,很有可能将程序搞崩溃。对于map而言,也没有类似的下标越界概念,但是却有作为下标的关键码(key)在map中不存在的现象。在这种情况下,如果运用不得当,就造成意想不到的问题,而且这种问题是属于比较隐蔽的问题。如果对map的下标运算符[]不是非常了解,将很难发现问题所在。
比如说有下面的代码片断:

// 定义一个map
map < string , string > mapTelDir;

// 用[]操作符可以方便快速的给map添加内容
mapTelDir[ " StarLee " ] = " 13813131313 " ;

// ...

// 修改关键码为LiXing的值
// 注意,这里的关键码在map里面并不存在
if (mapTelDir[ " LiXing " ] == " 13913131313 " )
{
mapTelDir[
" LiXing " ] = " 13513131313 " ;
}

// ...

// 输出
cout << " ThecellhonenumberofLiXingis: " << mapTelDir[ " LiXing " ] << endl;

上面的代码看起来似乎没有任何问题,但是,最后一行的输出的手机号码却永远为空!原因就是在判断语句里面对map的下标运算符[]的错误使用。
map的下标运算符[]的作用是:将关键码作为下标去执行查找,并返回对应的值;如果不存在这个关键码,就将一个具有该关键码和值类型的默认值的项插入这个map。
对于上面的代码来说,if (mapTelDir["LiXing"] == "13913131313")这行代码首先去查询map中是否有关键码为LiXing的项,结果发现没有该项,于是就向map中插入了一项<"LiXing", "">(sting的默认值为空字符串)。这就造成if语句的判断结果永远为false,那样就不可能修改修改关键码为LiXing的值。
为了更好的说明这个问题,可以看一下map下标运算符[]的实现代码:

// 下面的代码摘自VS2005中的..vcincludemap
mapped_type & operator []( const key_type & _Keyval)
{
// findelementmatching_Keyvalorinsertwithdefaultmapped
iterator_Where = this -> lower_bound(_Keyval);
if (_Where == this -> end()
|| this -> comp(_Keyval, this -> _Key(_Where._Mynode())))
_Where
= this -> insert(_Where,
value_type(_Keyval,mapped_type()));
return (( * _Where).second);
}

map的下标操作符[]使用上面的方法定义,使向map里插入值和使用关键码来访问相应的值变得非常方便而快速。虽然在使用的时候会出现有人写出类似上面那样的错误代码,但是这是程序员的错误,因为写出正确的代码是程序员的责任。这样做很符合C++的风格:非常灵活,但是要小心使用。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值