элементы перебираются в порядке, определенном ключом. Например, если мы перемещаемся по контейнеру, описанному в примере, то получим следующий порядок обхода:
имеет странное возвращаемое значение, которое в простых программах, как правило, мы игнорируем. Это пара, состоящая из итератора, установленного на пару (ключ, значение), и переменной типа
bool
, принимающей значение
true
, если данная пара (ключ, значение) была вставлена с помощью вызова функции
insert
. Если ключ уже был в контейнере, то вставка игнорируется и значение типа
bool
принимает значение
false
.
Мы можем определить порядок обхода ассоциативного массива с помощью третьего аргумента (предикат
Cmp
в объявлении класса
map
). Рассмотрим пример.
map<string, double, No_case> m;
Предикат
No_case
определяет сравнение символов без учета регистра (см. раздел 21.8). По умолчанию порядок обхода определяется предикатом
less<Key>
, т.е. отношением “меньше”.
21.6.3. Еще один пример ассоциативного массива
Для того чтобы оценить полезность контейнера
map
, вернемся к примеру с индексом Доу–Джонс из раздела 21.5.3. Описанный там код работает правильно, только если все веса записаны в объекте класса
vector
в тех же позициях, что и соответствующие имена. Это требование носит неявный характер и легко может стать источником малопонятных ошибок. Существует много способов решения этой проблемы, но наиболее привлекательным является хранение всех весов вместе с их тикером, например (“AA”,2.4808). Тикер — это аббревиатура названия компании. Аналогично тикер компании можно хранить вместе с ценой ее акции, например (“AA”,34.69). В заключение для людей, редко сталкивающихся с фондовым рынком США, мы можем записывать тикер вместе с названием компании, например (“AA”,“Alcoa Inc.”); иначе говоря, можем хранить три аассоциативных массива соответствующих значений.
Сначала создадим ассоциативный контейнер, содержащий пары (символ,цена).
map<string,double> dow_price;
// Индекс Доу - Джонса (символ, цена);
// текущие котировки
см. на веб-сайте www.djindexes.com
dow_price["MMM"] = 81.86;
dow_price ["AA"] = 34.69;
dow_price ["MO"] = 54.45;
// ...
Ассоциативный массив, содержащий пары (символ, вес), объявляется так:
map<string,double> dow_weight; // Индекс Доу-Джонса (символ, вес)
dow_weight.insert(make_pair("MMM", 5.8549));
dow_weight.insert(make_pair("AA",2.4808));
dow_weight.insert(make_pair("MO",3.8940));
// ...
Мы использовали функции
insert
и
make_pair
для того, чтобы показать, что элементами контейнера
map
действительно являются объекты класса
pair
. Этот пример также иллюстрирует значение обозначений; мы считаем, что индексирование понятнее и — что менее важно — легче записывается.
Ассоциативный контейнер, содержащий пары (символ, название).
// записывает цену акции для каждой компании, входящей в индекс
// Доу - Джонса
for (Dow_iterator p = dow_price.begin; p!=dow_price.end; ++p) {
const string& symbol = p–>first; // тикер
cout << symbol << '\t'
<< p–>second << '\t'
<< dow_name[symbol] << '\n';
}
Мы можем даже выполнить некоторые вычисления, непосредственно используя ассоциативный контейнер. В частности, можем вычислить индекс, как в разделе 21.5.3. Мы должны извлечь цены акций и веса из соответствующих ассоциативных массивов и перемножить их. Можно без труда написать функцию, выполняющую эти вычисления с любыми двумя ассоциативными массивами