7

这无法在 gcc 4.1.2 / RedHat 5 中编译:

#include <string>
#include <vector>
#include <map>

class Toto {
public:
    typedef std::string SegmentName;
};

class Titi {
public:
    typedef Toto::SegmentName SegmentName; // import this type in our name space
    typedef std::vector<SegmentName> SegmentNameList;
    SegmentNameList segmentNames_;
    typedef std::map<SegmentName, int> SegmentTypeContainer;
    SegmentTypeContainer segmentTypes_;

    int getNthSegmentType(unsigned int i) const {
        int result = -1;

       if(i < segmentNames_.size())
       {
           SegmentName name = segmentNames_[i];
           result = segmentTypes_[ name ];
       }
       return result;
    }
};

错误是:

error: no match for 'operator[]' in '(...)segmentTypes_[name]'
/usr/lib/gcc/x86_64-redhat-linux/4.1.2/../../../../include/c++/4.1.2/bits/stl_map.h:340:
note: candidates are: _Tp& std::map<_Key, _Tp, _Compare, _Alloc>::operator[](const _Key&)
[with _Key = std::basic_string<char, std::char_traits<char>, std::allocator<char> >, _Tp = int, _Compare = std::less<std::basic_string<char, std::char_traits<char>, std::allocator<char> > >, _Alloc = std::allocator<std::pair<const std::basic_string<char, std::char_traits<char>, std::allocator<char> >, int> >]

为什么 ?地图相当简单。我想这与 typedefs 有关,但有什么问题?

[编辑]即使我删除所有typedefsstd::string在任何地方使用,问题仍然存在......我是否滥用地图?

4

2 回答 2

17

std::map::operator[]是非常量的,你正试图从一个const方法中使用它。

您可以使用 来实现此目的std::map::find,它会返回const_iterator

SegmentTypeContainer::const_iterator iter = segmentTypes_.find(name);

如果您使用的是 C++11,您也可以使用std::map::at,如果在地图中找不到键,则会引发异常:

result = segmentTypes_.at(name);
于 2013-03-27T14:07:02.857 回答
12

std::map::operator[]不是const方法,但您是从const类的方法中调用它。这样做的原因是,如果键不存在,它会添加一个元素。

您可以使用 C++11 at()

result = segmentTypes_.at(name); // throws exception if key not present.

或使用std::map::find.

SegmentTypeContainer::const_iterator it = segmentTypes_.find(name);
if (it != segmentTypes_.end())
{
  // OK, element with key name is present
  result = it->second;
}
于 2013-03-27T14:08:00.317 回答