@Bios
2018-12-10T08:46:10.000000Z
字数 5328
阅读 975
js 数据结构 算法算法 + 数据结构 = 程序
数组是计算机编程世界里最常见的数据结构。
数组的标准定义:一个存储元素的线性集合(collection),元素可以通过索引来任意存取,索引通常是数字(也可以是其他字符)。
var arr = [];var arr = [1,2,3,4,5];var arr = new Array();var arr = new Array(10); // 10为数组的长度
indexOf()函数是最常用的存取函数之一,用来查找传进来的>参数再目标数组中是否存在。
var names = ['David', 'Cynthia', 'Raymond', 'Clayton', 'Jennifer'];putstr("Enter a name to search for:");var name = readline();var position = names.indexOf(name);if (position >= 0) {print("Found" + name + "at position" + position);} else {print(name + "not found in array.");}
有两个方法可以将数组转化为字符串:
join()和toString()。这两个方法都返回一个包含数组所有元素的字符串,各个元素之间用逗号分隔开。
var names = ['David', 'Cynthia', 'Raymond', 'Clayton', 'Jennifer'];var namestr = names.join();print(namestr); // David, Cynthia, Raymond, Clayton,Jennifer
事实上,当直接对一个数组使用
print()函数时,系统会>>自动调用那个数组的toString()方法
concat()和splice()方法允许通过已有数组创建新数组。concat()方法可以合并多个数组创建一个新数组,splice()方法截取一个数组的子集创建一个新数组。
var cisDept = ['Mike', 'Clayton', 'Terrill', 'Danny', 'Jennifer'];var dmpDept = ['Raymond', 'Cynthia', 'Bryan'];var itDiv = cisDept.concat(dmpDept);itDiv //Mike, Clayton, Terrill, Danny, Jennifer, Raymond, Cynthia, Bryan
concat()作为参数的数组,其中的元素都被连接到调用concat()方法的数组后面
var itDept = ['Mike', 'Clayton', 'Terrill', 'Danny', 'Jennifer', 'Raymond'];var dmpDept = itDept.splice(3, 3);var cisDept = itDept;print(dmpDept); // Danny, Jennifer, Raymondprint(cisDept); // Mike, Clayton
注释:
splice()该方法会改变原始数组。
push()和unshift()
push()将一个元素添加到数组末尾:
var nums = [1,2,3,4,5];print(nums); // 1,2,3,4,5nums.push(6);print(nums); // 1,2,3,4,5,6
unshift()方法可以将元素添加在数组的开头
如果不利用数组提供的可变函数,则新的元素添加进来后,需要把后面的每个元素都相应地向后移一个位置
ar nums = [2,3,4,5];var newnum = 1;var N = nums.length;for (var i = N; i >= 0; --i) {nums[i] = nums[i-1];}nums[0] = newnum;print(nums); // 1,2,3,4,5
unshift()
var nums = [2,3,4,5];print(nums); // 2,3,4,5var newnum = 1;nums.unshift(newnum);print(nums); // 1,2,3,4,5nums = [3,4,5];nums.unshift(newnum,1,2);print(nums); // 1,2,3,4,5
使用pop()方法可以删除数组末尾的元素:
var nums = [1,2,3,4,5,9];nums.pop();print(nums); // 1,2,3,4,5
shift()方法可以删除数组的第一个元素:
var nums = [9,1,2,3,4,5];nums.shift();print(nums); // 1,2,3,4,5
使用splice()方法为数组添加元素,需提供如下参数:
• 起始索引(也就是你希望开始添加元素的地方);
• 需要删除的元素个数(添加元素时该参数设为 0);
• 想要添加进数组的元素。
var nums = [1,2,3,7,8,9];var newElements = [4,5,6];nums.splice(3,0,newElements);print(nums); // 1,2,3,4,5,6,7,8,9
第一个方法是
reverse(),该方法将数组中元素的顺序进行翻转。
var nums = [1,2,3,4,5];nums.reverse();print(nums); // 5,4,3,2,1
如果元素是字符串类型,那么数组的可变方法
sort() 就非常好使:
var names = ["David","Mike","Cynthia","Clayton","Bryan","Raymond"];names.sort();print(names); // Bryan,Clayton,Cynthia,David,Mike,Raymond
但是如果数组元素是数字类型,
sort()方法的排序结果就不能让人满意了:
var nums = [3,1,2,100,4,200];nums.sort();print(nums); // 1,100,2,200,3,4
sort() 方法是按照字典顺序对元素进行排序的,因此它假定元素都是字符串类型,在上一
个例子中,即使元素是数字类型,也被认为是字符串类型。为了让sort()方法也能排序数
字类型的元素,可以在调用方法时传入一个大小比较函数,排序时,sort()方法将会根据
该函数比较数组中两个元素的大小,从而决定整个数组的顺序。
对于数字类型,该函数可以是一个简单的相减操作,从一个数字中减去另外一个数字。如
果结果为负,那么被减数小于减数;如果结果为 0,那么被减数与减数相等;如果结果为
正,那么被减数大于减数。
将这些搞清楚之后,传入一个大小比较函数,再来看看前面的例子:
function compare(num1, num2) {return num1 - num2;}var nums = [3,1,2,100,4,200];nums.sort(compare);print(nums); // 1,2,3,4,100,200
sort()函数使用了compare()函数对数组按照数字大小进行排序,而不是按照字典顺序。
我们要讨论的第一组迭代器方法不产生任何新数组,相反,它们要么对于数组中的每个元
素执行某种操作,要么返回一个值。
function square(num) {print(num, num * num);}var nums = [1,2,3,4,5,6,7,8,9,10];nums.forEach(square);
该程序的输出为:
1 12 43 94 165 256 367 498 649 8110 100
该方法接受一个返回值为布尔类型的函数,对数组中的每个元素使用该函数。如果对于所有的元素,该函数均返回 true,则该方法返回 true。
function isEven(num) {return num % 2 == 0;}var nums = [2,4,6,8,10];var even = nums.every(isEven);if (even) {print("all numbers are even");}else {print("not all numbers are even");}
// 输出为:all numbers are even// 将数组改为:var nums = [2,4,6,7,8,10];// 输出为:not all numbers are even
some() 方法也接受一个返回值为布尔类型的函数,只要有一个元素使得该函数返回true,该方法就返回 true。
function isEven(num) {return num % 2 == 0;}var nums = [1,2,3,4,5,6,7,8,9,10];var someEven = nums.some(isEven);if (someEven) {print("some numbers are even");}else {print("no numbers are even");}nums = [1,3,5,7,9];someEven = nums.some(isEven);if (someEven) {print("some numbers are even");}else {print("no numbers are even");}
// 该程序的输出为:some numbers are evenno numbers are even
reduce() 方法接受一个函数,返回一个值。该方法会从一个累加值开始,不断对累加值和数组中的后续元素调用该函数,直到数组中的最后一个元素,最后返回得到的累加值。
function add(runningTotal, currentValue) {return runningTotal + currentValue;}var nums = [1,2,3,4,5,6,7,8,9,10];var sum = nums.reduce(add);print(sum); // 显示 55
// reduce()方法和add()函数一起,从左到右,依次对数组中的元素求和,其执行过程如下所示:add(1,2) -> 3add(3,3) -> 6add(6,4) -> 10add(10,5) -> 15add(15,6) -> 21add(21,7) -> 28add(28,8) -> 36add(36,9) -> 45add(45,10) -> 55
有两个迭代器方法可以产生新数组:map()和 filter()。
map()和forEach()有点儿像,对数组中的每个元素使用某个函数。两者的区别是map()返回一个新的数组,该数组的元素是对原有元素应用某个函数得到的结果。
function curve(grade) {return grade += 5;}var grades = [77, 65, 81, 92, 83];var newgrades = grades.map(curve);print(newgrades); // 82, 70, 86, 97, 88
function first(word) {return word[0];}var words = ["for","your","information"];var acronym = words.map(first);print(acronym.join("")); // 显示 "fyi"
在上面这个例子中,数组 acronym保存了数组 words中每个元素的第一个字母。然而,如果想将数组显示为真正的缩略形式,必须想办法除掉连接每个数组元素的逗号,如果直接调用 toString() 方法,就会显示出这个逗号。使用 join() 方法,为其传入一个空字符串作为参数,则可以帮助我们解决这个问题。
filter() 和every()类似,传入一个返回值为布尔类型的函数。和 every()方法不同的是,
当对数组中的所有元素应用该函数,结果均为 true 时,该方法并不返回true,而是返回一个新数组,该数组包含应用该函数后结果为 true 的元素。
function isEven(num) {return num % 2 == 0;}function isOdd(num) {return num % 2 != 0;}var nums = [];for (var i = 0; i < 20; ++i) {nums[i] = i+1;}var evens = nums.filter(isEven);print("Even numbers: ");print(evens);var odds = nums.filter(isOdd);print("Odd numbers: ");print(odds);
该程序的执行结果如下:Even numbers:2,4,6,8,10,12,14,16,18,20Odd numbers:1,3,5,7,9,11,13,15,17,19