blob: b289d9af204330d87090169f14a161c81a6b4c3c [file] [log] [blame]
// Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
part of dart.core;
/**
* An unordered collection of key-value pairs, from which you retrieve a value
* by using its associated key.
*
* Each key can occur at most once in a map.
*
* It is generally not allowed to modify the map (add or remove keys) while
* an operation is being performed on the map, for example in functions called
* during a [forEach] or [putIfAbsent] call.
* Modifying the map while iterating the keys or values will also most likely
* break the iteration.
*/
abstract class Map<K, V> {
/**
* Creates a Map instance with the default implementation, [LinkedHashMap].
*
* A `LinkedHashMap` requires the keys to implement compatible
* `operator==` and `hashCode`, and it allows null as a key.
*/
factory Map() = LinkedHashMap<K, V>;
/**
* Creates a [LinkedHashMap] instance that contains all key-value pairs of
* [other].
*
* A `LinkedHashMap` requires the keys to implement compatible
* `operator==` and `hashCode`, and it allows null as a key.
*/
factory Map.from(Map<K, V> other) = LinkedHashMap<K, V>.from;
/**
* Creates an identity map with the default implementation, [LinkedHashMap].
*
* The returned map allows `null` as a key.
*/
factory Map.identity() = LinkedHashMap<K, V>.identity;
/**
* Creates a Map instance in which the keys and values are computed from the
* [iterable].
*
* The created map is a [LinkedHashMap].
* A `LinkedHashMap` requires the keys to implement compatible
* `operator==` and `hashCode`, and it allows null as a key.
*
* For each element of the [iterable] this constructor computes a key-value
* pair, by applying [key] and [value] respectively.
*
* The example below creates a new Map from a List. The keys of `map` are
* `list` values converted to strings, and the values of the `map` are the
* squares of the `list` values:
*
* List<int> list = [1, 2, 3];
* Map<String, int> map = new Map.fromIterable(list,
* key: (item) => item.toString(),
* value: (item) => item * item));
*
* map['1'] + map['2']; // 1 + 4
* map['3'] - map['2']; // 9 - 4
*
* If no values are specified for [key] and [value] the default is the
* identity function.
*
* In the following example, the keys and corresponding values of `map`
* are `list` values:
*
* map = new Map.fromIterable(list);
* map[1] + map[2]; // 1 + 2
* map[3] - map[2]; // 3 - 2
*
* The keys computed by the source [iterable] do not need to be unique. The
* last occurrence of a key will simply overwrite any previous value.
*/
factory Map.fromIterable(Iterable iterable,
{K key(element), V value(element)}) = LinkedHashMap<K, V>.fromIterable;
/**
* Creates a Map instance associating the given [keys] to [values].
*
* The created map is a [LinkedHashMap].
* A `LinkedHashMap` requires the keys to implement compatible
* `operator==` and `hashCode`, and it allows null as a key.
*
* This constructor iterates over [keys] and [values] and maps each element of
* [keys] to the corresponding element of [values].
*
* List<String> letters = ['b', 'c'];
* List<String> words = ['bad', 'cat'];
* Map<String, String> map = new Map.fromIterables(letters, words);
* map['b'] + map['c']; // badcat
*
* If [keys] contains the same object multiple times, the last occurrence
* overwrites the previous value.
*
* It is an error if the two [Iterable]s don't have the same length.
*/
factory Map.fromIterables(Iterable<K> keys, Iterable<V> values)
= LinkedHashMap<K, V>.fromIterables;
/**
* Returns true if this map contains the given value.
*/
bool containsValue(Object value);
/**
* Returns true if this map contains the given key.
*/
bool containsKey(Object key);
/**
* Returns the value for the given [key] or null if [key] is not
* in the map. Because null values are supported, one should either
* use [containsKey] to distinguish between an absent key and a null
* value, or use the [putIfAbsent] method.
*/
V operator [](Object key);
/**
* Associates the [key] with the given [value].
*/
void operator []=(K key, V value);
/**
* If [key] is not associated to a value, calls [ifAbsent] and
* updates the map by mapping [key] to the value returned by
* [ifAbsent]. Returns the value in the map.
*
* Map<String, int> scores = {'Bob': 36};
* for (var key in ['Bob', 'Rohan', 'Sophena']) {
* scores.putIfAbsent(key, () => key.length);
* }
* scores['Bob']; // 36
* scores['Rohan']; // 5
* scores['Sophena']; // 7
*
* The code that [ifAbsent] executes must not add or remove keys.
*/
V putIfAbsent(K key, V ifAbsent());
/**
* Adds all key-value pairs of [other] to this map.
*
* If a key of [other] is already in this map, its value is overwritten.
*
* The operation is equivalent to doing `this[key] = value` for each key
* and associated value in other. It iterates over [other], which must
* therefore not change during the iteration.
*/
void addAll(Map<K, V> other);
/**
* Removes the association for the given [key]. Returns the value for
* [key] in the map or null if [key] is not in the map. Note that values
* can be null and a returned null value does not always imply that the
* key is absent.
*/
V remove(Object key);
/**
* Removes all pairs from the map.
*/
void clear();
/**
* Applies [f] to each {key, value} pair of the map.
*
* It is an error to add or remove keys from the map during iteration.
*/
void forEach(void f(K key, V value));
/**
* The keys of [this].
*
* The returned iterable has efficient `length` and `contains` operations,
* based on [length] and [containsKey] of the map.
*/
Iterable<K> get keys;
/**
* The values of [this].
*
* The returned iterable has an efficient `length` method based on the
* [length] of the map.
*/
Iterable<V> get values;
/**
* The number of key-value pairs in the map.
*/
int get length;
/**
* Returns true if there is no key-value pair in the map.
*/
bool get isEmpty;
/**
* Returns true if there is at least one key-value pair in the map.
*/
bool get isNotEmpty;
}