枚举或映射Dart中具有索引和值的列表


99

在dart中,有任何等同于常见的东西:

enumerate(List) -> Iterator((index, value) => f)
or 
List.enumerate()  -> Iterator((index, value) => f)
or 
List.map() -> Iterator((index, value) => f)

看来这是最简单的方法,但仍然不存在此功能似乎很奇怪。

Iterable<int>.generate(list.length).forEach( (index) => {
  newList.add(list[index], index)
});

编辑:

感谢@ hemanth-raj,我得以找到所需的解决方案。我将把它放在这里,供需要执行类似操作的任何人使用。

List<Widget> _buildWidgets(List<Object> list) {
    return list
        .asMap()
        .map((index, value) =>
            MapEntry(index, _buildWidget(index, value)))
        .values
        .toList();
}

或者,您可以创建一个同步生成器函数以返回一个可迭代

Iterable<MapEntry<int, T>> enumerate<T>(Iterable<T> items) sync* {
  int index = 0;
  for (T item in items) {
    yield MapEntry(index, item);
    index = index + 1;
  }
}

//and use it like this.
var list = enumerate([0,1,3]).map((entry) => Text("index: ${entry.key}, value: ${entry.value}"));

Map#forEach?是你想要的吗?
pskink

它是通过列表而不是地图进行枚举的
David Rees

Map#forEach通过List?枚举?你什么意思?文档说:“将f应用于映射的每个键/值对。调用f不得在映射中添加或删除键。”
pskink

我也搞不懂你的意思与“历数或通过与指数和值列表图”
冈特Zöchbauer

1
@GünterZöchbauergithub.com / dart
C

Answers:


151

有一种asMap方法可以将列表转换为映射,其中键是索引,值是索引中的元素。请在这里看一下文档。

例:

List _sample = ['a','b','c'];
_sample.asMap().forEach((index, value) => f);

希望这可以帮助!


35

没有内置函数来获取迭代索引。

如果像我一样,您不喜欢Map仅为简单索引构建(数据结构)的想法,则可能需要的是map为您提供索引的(函数)。让我们称之为mapIndexed(就像在Kotlin中一样):

children: mapIndexed(
  list,
  (index, item) => Text("event_$index")
).toList();

的实现mapIndexed很简单:

Iterable<E> mapIndexed<E, T>(
    Iterable<T> items, E Function(int index, T item) f) sync* {
  var index = 0;

  for (final item in items) {
    yield f(index, item);
    index = index + 1;
  }
}

1
这是一个很好的答案,但写成同步发电机可能会更好
David Rees

2
@DavidRees thx的建议!我还重命名了该功能mapIndexed
Vivien

1
可惜的是,dart没有内置的简便方法来执行此操作。即使是30岁的python也可以轻松做到这一点!
osamu

事实证明,这.asMap().forEach()基本上是相同的-请参阅我的答案。
Timmmm

1
我认为@TimmmmasMap()会花费额外的循环来检索数据,因此mapIndexed()如上所述效率不高。
anticafe

18

建立在@Hemanth Raj答案上。

要将其转换回您可以

List<String> _sample = ['a', 'b', 'c'];
_sample.asMap().values.toList(); 
//returns ['a', 'b', 'c'];

或者,如果您需要映射函数的索引,则可以执行以下操作:

_sample
.asMap()
.map((index, str) => MapEntry(index, str + index.toString()))
.values
.toList();
// returns ['a0', 'b1', 'c2']

16

从Dart 2.7开始,您可以使用extension扩展功能,Iterable而不必编写辅助方法

extension ExtendedIterable<E> on Iterable<E> {
  /// Like Iterable<T>.map but callback have index as second argument
  Iterable<T> mapIndex<T>(T f(E e, int i)) {
    var i = 0;
    return this.map((e) => f(e, i++));
  }

  void forEachIndex(void f(E e, int i)) {
    var i = 0;
    this.forEach((e) => f(e, i++));
  }
}

用法:

final inputs = ['a', 'b', 'c', 'd', 'e', 'f'];
final results = inputs
  .mapIndex((e, i) => 'item: $e, index: $i')
  .toList()
  .join('\n');

print(results);

// item: a, index: 0
// item: b, index: 1
// item: c, index: 2
// item: d, index: 3
// item: e, index: 4
// item: f, index: 5
inputs.forEachIndex((e, i) => print('item: $e, index: $i'));

// item: a, index: 0
// item: b, index: 1
// item: c, index: 2
// item: d, index: 3
// item: e, index: 4
// item: f, index: 5

2
这是最好的答案,这甚至适用于Flutter项,您可以在其中返回Widget而不只是String。

8

Lukas Renggli的更多软件包包括许多有用的工具,其中包括“索引”,它可以完全满足您的需求。从文档:

indexed(['a', 'b'], offset: 1)
  .map((each) => '${each.index}: ${each.value}')
  .join(', ');

(除非您具有Smalltalk背景,否则您可以忽略offset参数)。


7

最初,我认为['one', 'two', 'three'].asMap().forEach((index, value) { ... });效率真的很低,因为它看起来像是将列表转换为地图。实际上不是-文档说它创建了列表的不变视图。我仔细检查了dart2js以下代码:

void main() {
  final foo = ['one', 'two', 'three'];
  foo.asMap().forEach((idx, val) {
    print('$idx: $val');
  });
}

它生成很多代码!但是要点是:

  main: function() {
    var foo = H.setRuntimeTypeInfo(["one", "two", "three"], ...);
    new H.ListMapView(foo, ...).forEach$1(0, new F.main_closure());
  },

  H.ListMapView.prototype = {
    forEach$1: function(_, f) {
      var t1, $length, t2, i;
      ...
      t1 = this._values;
      $length = t1.length;
      for (t2 = $length, i = 0; i < $length; ++i) {
        if (i >= t2)
          return H.ioore(t1, i);
        f.call$2(i, t1[i]);
        t2 = t1.length;
        if ($length !== t2)
          throw H.wrapException(P.ConcurrentModificationError$(t1));
      }
    },
    ...
  },

  F.main_closure.prototype = {
    call$2: function(idx, val) {
      ...
      H.printString("" + idx + ": " + H.S(val));
    },
    $signature: 1
  };

因此,做高效的事情足够聪明!很聪明

当然,您也可以只使用普通的for循环:

for (var index = 0; index < values.length; ++index) {
  final value = values[index];

4

为了方便起见,您可以使用此扩展方法。

extension CollectionUtil<T> on Iterable<T>  {

  Iterable<E> mapIndexed<E, T>(E Function(int index, T item) transform) sync* {
    var index = 0;

    for (final item in this) {
      yield transform(index, item as T);
      index++;
    }
  }
}

3

使用asMap将List首先转换为地图。元素的索引是关键。元素成为价值。使用条目将键和值映射到所需的任何内容。

List rawList = ["a", "b", "c"];
List<String> argList = rawList.asMap().entries.map((e) => '${e.key}:${e.value}').toList();
print(argList);

输出:

[0:a, 1:b, 2:c]

0

您可以使用Iterable.generate工厂。以下代码将映射一个Iterable使用索引和值。

extension IterableMapIndex<T> on Iterable<T> {
  Iterable<E> mapIndexed<E>(E f(int index, T t)) {
    return Iterable.generate(this.length, (index)=>f(index, elementAt(index)));
  }
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.