为啥 ES2015 中的字符串是可迭代的?

Posted

技术标签:

【中文标题】为啥 ES2015 中的字符串是可迭代的?【英文标题】:Why are strings iterables in ES2015?为什么 ES2015 中的字符串是可迭代的? 【发布时间】:2016-08-22 12:38:03 【问题描述】:

有人可以帮助我更好地理解在 ES2015 中使 javascript 字符串可迭代的设计决策吗?大多数可迭代对象代表其他事物的集合:ArraySetMapGenerator(有点)。字符串只能是字符串的集合(好吧,Unicode 代码点)。因此,如果您希望函数接受字符串集合(例如 Set 字符串或生成字符串的生成器)单个字符串,则无法区分用户的意图。她是指一个字符串还是一个可迭代的代码点?与传递字符串集合相比,迭代字符似乎是一种极端情况。在这种情况下,我倾向于在(预期)类型为 stringIterable<string> 时使用特殊情况的字符串参数。

function deleteDocuments(ids /* string or Iterable<string> */) 
  if('string' === typeof ids)  
    deleteSingleDocById(ids);
   else 
    for(let id of ids)  
      deleteSingleDocById(id);
    
  

我错过了什么?

【问题讨论】:

如果你的函数需要一个参数,它要么是一个可迭代的字符串,要么是一个可迭代的数组。为什么不让它处理一个可迭代的?您的困境尚不清楚。 对于esdiscuss.org 来说似乎是一个更好的问题。 @guest271314 deleteDocuments() 函数采用单个字符串或字符串集合,并且对于每个值,使用单个 id 调用 deleteSingleDocById()。不过,我不想限制自己只接受数组。我希望调用者能够指定任何可迭代对象,例如 Set 或生成器。这显然是一种过于简单的提炼。 那又怎样?如果您期望一个可迭代对象,请对可迭代对象执行您所做的操作。如果您期待一种特殊的可迭代对象,那么这个设计问题可能就是核心问题。 @Pointy 是的,但我正在尝试将数组以外的内容推广到可迭代对象。 【参考方案1】:

这实际上是 ES2015/ES6 之前的一个原则的延续。自 Internet Explorer 8 发布以来,每个浏览器都允许您使用相同的语法循环遍历数组和字符串的内容。

var array = ['a', 'b', 'c'];

for (var i = 0; i < array.length; i++) 
  console.log(array[i]);

var string = 'abc';

for (var i = 0; i < string.length; i++) 
  console.log(string[i]);

字符串已经“类似数组”;它们实现了隐式数组迭代协议(具有.length 和索引属性),并且每个字符在查找时都已被视为自己的字符串。我们可以争论这是否是正确的选择,但在这一点上这是遥远的历史。如果在 ES2015 之后,您被期望使用不同的语法来迭代数组和字符串,那将是一种奇怪的行为分歧。

【讨论】:

谢谢,杰里米。我就是这么想的。似乎是一个不幸的痕迹提出来。如果我想遍历组成我的字符串的(数组)字符,我实际上更愿意使用'string'.split('') 明确。不过,我想这既不是这里也不是那里。

以上是关于为啥 ES2015 中的字符串是可迭代的?的主要内容,如果未能解决你的问题,请参考以下文章

for/of与Set和Map

迭代器

python知识点

为啥链式迭代如此复杂?简化此代码

为啥斜线在 JSON 中是可转义字符? [复制]

ES8中的字符串扩展