jQuery中有一个 map方法,循环时候是 k-v:
$(':checkbox[checked]').map( (k, v) => $(v).val() ).get();
到了js这里的map,就变成了 v - k
['a', 'b', 'c'].map( (v, k) => v) ;
js这样设计其实是为了方便省略 k,因为很多时候只需要v就够了,比如:
['2', '3', '4'].map( v => parseInt(v) );
但过度的省略可能会带来歧义,比如 :
['2', '3', '4'].map(parseInt);
=> [2, NaN, NaN]
这是一个非常著名的js大坑。
原因是 parseInt 的原型其实是 parseInt(string, base), 第二个参数其实是转换的进制,等价成了这样:
['2', '3', '4'].map( (v, k) => parseInt(v, k) );
循环调用分别为:parseInt('2', 0); parseInt('3', 1); parseInt('4', 2);
所以最好写成:
['2', '3', '4'].map( v => parseInt(v) ) ;
因为经常会忘记第一个到底是k,还是v,最好写成这样:
['2', '3', '4'].map( (v, k) => parseInt(v) ) ;
鉴于jQuery和原生js的map函数都是非常常用的,k,v 还是 v,k,经常容易混淆,这样的不一样设计非常不好。
再看看其他语言的 GO的,go没有map,但range也差不多,go使用的是 k, v:
for k, v := range []byte{'a', 'b', 'c'} { fmt.Printf("%d %c\n", k, v) } => 0 a 1 b 2 c
考虑到go不允许出现未使用的参数,所以很多时候需要这样:
for _, v := range []byte{'a', 'b', 'c'} { fmt.Printf("%c\n", v) }
虽然是麻烦了,但依然遵循 k, v 这样熟悉的排列方式。
Ruby中的循环,可选带上Index,但是确是js这样的 v - k 方式:
2.2.2 :005 > ('a'..'c').each_with_index{|v, k| puts "#{v} #{k}" } a 0 b 1 c 2
ruby好记住,主要是因为each_with_index 和 each,既然是带了index,那index在后面就刚刚好。
到底应该k-v还是,v-k,不能统一一下嘛。一会儿k-v,一会儿v-k,真是够了。