整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:

javascript数组与对象

组和对象

什么是对象,其实就是一种类型,即引用类型。而对象的值就是引用类型的实例。在ECMAScript 中引用类型是一种数据结构,用于将数据和功能组织在一起。它也常被称做为类,但 ECMAScript 中却没有这种东西。虽然 ECMAScript 是一门面向对象的语言,却不具备传统面向对象语言所支持的类和接口等基本结构。


一.Object 类型

到目前为止,我们使用的引用类型最多的可能就是 Object 类型了。虽然 Object 的实例不具备多少功能,但对于在应用程序中的存储和传输数据而言,它确实是非常理想的选择。创建 Object 类型有两种。一种是使用 new 运算符,一种是字面量表示法。

1.使用 new 运算符创建 Object

var box = new Object();

box.name = '大圣网络';

box.age = 28;

2.new 关键字可以省略

var box = Object();

3.使用字面量方式创建 Object

var box = {

name : '大圣网络',

age : 28

};

4.属性字段也可以使用字符串形式

var box = {

'name' : '大圣网络',

'age' : 28

};

5.使用字面量及赋值方式

var box = {}; //字面量方式声明空的对象

box.name = '大圣网络'; //点符号给属性复制

box.age = 28;

6.两种属性输出方式

alert(box.age);

alert(box['age']);

7.给对象创建方法

var box = {

run : function () { //对象中的方法

return '运行';

}

}

alert(box.run()); //调用对象中的方法

8.使用 delete 删除对象属性

delete box.name; //删除属性

在实际开发过程中,一般我们更加喜欢字面量的声明方式。因为它清晰,语法代码少 ,而且还给人一种封装的感觉。字面量也是向函数传递大量可选参数的首选方式。

function box(obj) { //参数是一个对象

if (obj.name != undefined) alert(obj.name); //判断属性是否存在

if (obj.age != undefined) alert(obj.age);

}

box({ //调用函数传递一个对象

name : '大圣',

age : 28

});

二. Array 类型

除了 Object 类型之外,Array 类型是 ECMAScript 最常用的类型。而且 ECMAScript 中的 Array 类型和其他语言中的数组有着很大的区别。 虽然数组都是有序排列, 但 ECMAScript中的数组每个元素可以保存任何类型。ECMAScript 中数组的大小也是可以调整的。创建 Array 类型有两种方式:第一种是 new 运算符,第二种是字面量。

1.使用 new 关键字创建数组

var box = new Array(); //创建了一个数组

var box = new Array(10); //创建一个包含 10 个元素的数组

var box = new Array('李炎恢',28,'教师','盐城'); //创建一个数组并分配好了元素

2.以上三种方法,可以省略 new 关键字。

var box =Array();

3 使用字面量方式创建数组

var box = []; //创建一个空的数组

var box = ['大圣',28,'金箍棒']; //创建包含元素的数组

4.使用索引下标来读取数组的值

alert(box[2]); //获取第三个元素

box[2] = '学生'; //修改第三个元素

box[4] = '计算机编程'; //增加第五个元素

5.使用 length 属性获取数组元素量

alert(box.length) //获取元素个数

box.length = 10; //强制元素个数

box[box.length] = 'JS 技术'; //通过 length 给数组增加一个元素

6.创建一个稍微复杂一点的数组

var box = [

{

name : '李炎恢',

age : 28,

run : function () {

return 'run 了';

}

},

['马云','马化腾',new Object()],

'江苏',

25+25,

new Array(1,2,3)

];

alert(box);

三. 对象中的方法

转换方法

对象或数组都具有toLocaleString()、 toString()和valueOf()方法。 其中toString()和valueOf()无论重写了谁,都会返回相同的值。数组会讲每个值进行字符串形式的拼接,以逗号隔开。

var box = ['小学生',8,'计算机编程']; //字面量数组

alert(box); //隐式调用了 toString()

alert(box.toString()); //和 valueOf()返回一致

alert(box.toLocaleString()); //返回值和上面两种一致

默认情况下,数组字符串都会以逗号隔开。如果使用 join()方法(相当于php函数的implode),则可以使用不同的分隔符来构建这个字符串。

var box = ['小学生', 8, '计算机编程'];

alert(box.join('|')); //小学生|8|计算机编程

栈方法

ECMAScript 数组提供了一种让数组的行为类似于其他数据结构的方法。也就是说,可以让数组像栈一样,可以限制插入和删除项的数据结构。栈是一种数据结构(后进先出),也就是说最新添加的元素最早被移除。而栈中元素的插入(或叫推入)和移除(或叫弹出),只发生在一个位置——栈的顶部。ECMAScript 为数组专门提供了 push()和 pop()方法。push()方法可以接收任意数量的参数,把它们逐个添加到数组的末尾,并返回修改后数组的长度。而 pop()方法则从数组末尾移除最后一个元素,减少数组的 length 值,然后返回移除的元素。

var box = ['大圣', 500, '打妖怪']; //字面量声明

alert(box.push('斗战胜佛')); //数组末尾添加一个元素,并且返回长度

alert(box); //查看数组

box.pop(); //移除数组末尾元素,并返回移除的元素

alert(box); //查看元素

队列方法

栈方法是后进先出,而列队方法就是先进先出。列队在数组的末端添加元素,从数组的前端移除元素。通过 push()向数组末端添加一个元素,然后通过 shift()方法从数组前端移除一个元素。ECMAScript 还为数组提供了一个 unshift()方法,它和 shift()方法的功能完全相反。

var box = ['大圣', 500, '打妖怪'];

alert(box.shift()); //移除数组开头元素,并返回移除的元素

alert(box); //查看数组

var box = ['大圣', 500, '打妖怪'];

alert(box.unshift('a','b')); //数组开头添加两个元素

alert(box); //查看数组

alert(box.pop()); //移除数组末尾元素,并返回移除的元素

alert(box); //查看数组

重排序方法

数组中已经存在两个可以直接用来排序的方法:reverse()和 sort()。

reverse() 逆向排序

var box = [1,2,3,4,5]; //数组

alert(box.reverse()); //逆向排序方法,返回排序后的数组

alert(box); //源数组也被逆向排序了,说明是引用

sort() 从小到大排序

var box = [4,1,7,3,9,2]; //数组

alert(box.sort()); //从小到大排序,返回排序后的数组

alert(box); //源数组也被从小到大排序了

sort 方法的默认排序在数字排序上有些问题,因为数字排序和数字字符串排序的算法是一样的。我们必须修改这一特征,修改的方式,就是给 sort(参数)方法传递一个函数参数。这点可以参考手册说明。

操作方法

ECMAScript 为操作已经包含在数组中的元素提供了很多方法。 concat()方法可以基于当前数组创建一个新数组。 slice()方法可以基于当前数组获取指定区域元素并创建一个新数组 。splice()主要用途是向数组的中部插入元素。

var box = ['大圣网络', 28, '中国']; //当前数组

var box2 = box.concat('计算机编程'); //创建新数组,并添加新元素

alert(box2); //输出新数组

var box = ['大圣网络', 28, '中国'];; //当前数组

var box2 = box.slice(1); //box.slice(1,3),2-4 之间的元素

alert(box2); //28,中国

splice 中的删除功能:

var box = ['大圣网络', 28, '中国']; //当前数组

var box2 = box.splice(0,2); //截取前两个元素

alert(box2); //返回截取的元素

alert(box); //当前数组被截取的元素被删除

splice 中的插入功能:

var box = ['大圣网络', 28, '中国']; //当前数组

var box2 = box.splice(1,0,'计算机编程','安徽'); //没有截取,但插入了两条

alert(box2); //在第 2 个位置插入两条

alert(box); //输出

splice 中的替换功能:

var box = ['大圣网络', 28, '中国']; //当前数组

var box2 = box.splice(1,1,100); //截取了第 2 条,替换成 100

alert(box2); //输出截取的 28

alert(box); //输出数组

Array 对象

Array 对象用于在变量中存储多个值:

var cars = ["Saab", "Volvo", "BMW"];

第一个数组元素的索引值为 0,第二个索引值为 1,以此类推。

更多有关JavaScript Array参考手册请参考 JavaScript Array 对象手册。

Array 对象属性

方法描述
concat()连接两个或更多的数组,并返回结果。
every()检测数值元素的每个元素是否都符合条件。
filter()检测数值元素,并返回符合条件所有元素的数组。
indexOf()搜索数组中的元素,并返回它所在的位置。
join()把数组的所有元素放入一个字符串。
lastIndexOf()返回一个指定的字符串值最后出现的位置,在一个字符串中的指定位置从后向前搜索。
map()通过指定函数处理数组的每个元素,并返回处理后的数组。
pop()删除数组的最后一个元素并返回删除的元素。
push()向数组的末尾添加一个或更多元素,并返回新的长度。
reverse()反转数组的元素顺序。
shift()删除并返回数组的第一个元素。
slice()选取数组的的一部分,并返回一个新数组。
some()检测数组元素中是否有元素符合指定条件。
sort()对数组的元素进行排序。
splice()从数组中添加或删除元素。
toString()把数组转换为字符串,并返回结果。
unshift()向数组的开头添加一个或更多元素,并返回新的长度。
valueOf()返回数组对象的原始值。

如您还有不明白的可以在下面与我留言或是与我探讨QQ群308855039,我们一起飞!

么把一个对象当做数组使用?

我们知道在JS中对象和数组的操作方式是不一样的,但是我们可以通过封装,给对象加一层包装器,让它可以和数组拥有同样的使用方式。我们主要借助Object.keys()Object.values()Object.entries()Proxy

Object.keys

看一下MDN上的解释:

Object.keys() 方法会返回一个由一个给定对象的自身可枚举属性组成的数组,数组中属性名的排列顺序和正常循环遍历该对象时返回的顺序一致。

也就是Object.keys可以获取对象的所有属性名,并生成一个数组。

var obj = { a: 0, b: 1, c: 2 };
console.log(Object.keys(obj));
// console: ['a', 'b', 'c']

Object.values

看一下MDN上的解释:

Object.values()方法返回一个给定对象自身的所有可枚举属性值的数组,值的顺序与使用for...in循环的顺序相同 ( 区别在于 for-in 循环枚举原型链中的属性 )。

Object.values()返回一个数组,元素是对象上找到的可枚举属性值。

var obj = { foo: 'bar', baz: 42 };
console.log(Object.values(obj));
// ['bar', 42]

Object.entries

看一下MDN上的解释:

Object.entries()方法返回一个给定对象自身可枚举属性的键值对数组,其排列与使用 for...in 循环遍历该对象时返回的顺序一致(区别在于 for-in 循环还会枚举原型链中的属性)。

Object.entries()返回一个数组,元素是由属性名和属性值组成的数组。

const obj = { foo: 'bar', baz: 42 };
console.log(Object.entries(obj));
// [ ['foo', 'bar'], ['baz', 42] ]

Proxy

Proxy是JS最新的对象代理方式,用于创建一个对象的代理,从而实现基本操作的拦截和自定义(如属性查找、赋值、枚举、函数调用等)。

使用Proxy可以封装对象的原始操作,在执行对象操作的时候,会经过Proxy的处理,这样我们就可以实现数组操作命令。

基础 get 示例

const handler = {
  get: function(obj, prop) {
    return prop in obj ? obj[prop] : 37;
  }
}
const p = new Proxy({}, handler);
p.a = 1;
p.b = undefined;
console.log(p.a, p.b)
console.log('c' in p, p.c)

以上示例的中,当对象中不存在属性名时,默认返回值为37

无操作转发代理

使用Proxy包装原生对象生成一个代理对象p,对代理对象的操作会转发到原生对象上。

let target = {};
let p = new Proxy(target, {});
p.a = 37;   // 操作转发到目标
console.log(target.a);    // 37. 操作已经被正确地转发

我们要实现以下几个函数:forEachmapfilterreduceslicefindfindKeyincludeskeyOflastKeyOf

实现数组函数

forEach

数组中的forEach函数定义:arr.forEach(callback(currentValue [, index [, array]])[, thisArg])

数组中的forEach需要传入一个函数,函数的第一个参数是当前操作的元素值,第二个参数是当前操作的元素索引,第三个参数是正在操作的对象。

对于对象,我们将参数定为:currentValue、key、target。我们可以使用Object.keys来遍历对象。

Object.keys(target).forEach(key => callback(target[key], key, target))

这里需要targetcallback参数,我们通过函数封装一下

function forEach(target, callback) {
  Object.keys(target).forEach(key => callback(target[key], key, target))
}

这样我们就可以使用以下方式调用:

const a = {a: 1, b: 2, c: 3}
forEach(a, (v, k) => console.log(`${k}-${v}`))
// a-1
// b-2
// c-3

通过Proxy封装:

const handler = {
  get: function(obj, prop) {
    return forEach(obj)
  }
}
const p = new Proxy(a, handler)
p.forEach((v, k) => console.log(`${k}-${v}`))

以上方式当然是不行的,我们主要看最后一句,其执行方式和数组的forEach完全相同,我们在调用Proxy封装的对象时,获取数据时,会调用get函数,第一个参数为原生对象,第二个参数为属性名-forEach,在这里就要修改我们的forEach函数了。首先p.forEach的参数是一个函数,因此我们代理对象的返回值需要接收一个函数作为参数,因此修改如下:

function forEach(target) {
  return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target))
}

因此完成代码为:

function forEach(target) {
  return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target))
}
const handler = {
  get: function(obj, prop) {
    return forEach(obj)
  }
}
const a = {a: 1, b: 2, c: 3}
const p = new Proxy(a, handler)
p.forEach((v, k) => console.log(`${k}-${v}`))
// a-1
// b-2
// c-3

我们应该把以上代码封装为模块,方便对外使用:

const toKeyedArray = (obj) => {
  const methods = {
    forEach(target) {
      return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target));
    }
  }
  const methodKeys = Object.keys(methods)
  const handler = {
    get(target, prop) {
      if (methodKeys.includes(prop)) {
        return methods[prop](target)
      }
      return Reflect.get(...arguments)
    }
  }
  return new Proxy(obj, handler)
}
const a = { a: 1, b: 2, c: 3}
const p = toKeyedArray(a)
p.forEach((v, k) => console.log(`${k}-${v}`))

以上是forEach的实现和封装,其他函数的实现方式类似。

全部源码如下:

const toKeyedArray = (obj) => {
  const methods = {
    forEach(target) {
      return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target));
    },
    map(target) {
      return (callback) =>
        Object.keys(target).map(key => callback(target[key], key, target));
    },
    reduce(target) {
      return (callback, accumulator) =>
        Object.keys(target).reduce(
          (acc, key) => callback(acc, target[key], key, target),
          accumulator
        );
    },
    forEach(target) {
      return callback =>
        Object.keys(target).forEach(key => callback(target[key], key, target));
    },
    filter(target) {
      return callback =>
        Object.keys(target).reduce((acc, key) => {
          if (callback(target[key], key, target)) acc[key] = target[key];
          return acc;
        }, {});
    },
    slice(target) {
      return (start, end) => Object.values(target).slice(start, end);
    },
    find(target) {
      return callback => {
        return (Object.entries(target).find(([key, value]) =>
          callback(value, key, target)
        ) || [])[0];
      };
    },
    findKey(target) {
      return callback =>
        Object.keys(target).find(key => callback(target[key], key, target));
    },
    includes(target) {
      return val => Object.values(target).includes(val);
    },
    keyOf(target) {
      return value =>
        Object.keys(target).find(key => target[key] === value) || null;
    },
    lastKeyOf(target) {
      return value =>
        Object.keys(target)
          .reverse()
          .find(key => target[key] === value) || null;
    }
  }
  const methodKeys = Object.keys(methods)
  const handler = {
    get(target, prop) {
      if (methodKeys.includes(prop)) {
        return methods[prop](target)
      }
      const [keys, values] = [Object.keys(target), Object.values(target)];
      if (prop === 'length') return keys.length;
      if (prop === 'keys') return keys;
      if (prop === 'values') return values;
      if (prop === Symbol.iterator)
        return function* () {
          for (value of values) yield value;
          return;
        };
      return Reflect.get(...arguments)
    }
  }
  return new Proxy(obj, handler)
}
const x = toKeyedArray({ a: 'A', b: 'B' });
x.a;          // 'A'
x.keys;       // ['a', 'b']
x.values;     // ['A', 'B']
[...x];       // ['A', 'B']
x.length;     // 2
// Inserting values
x.c = 'c';    // x = { a: 'A', b: 'B', c: 'c' }
x.length;     // 3
// Array methods
x.forEach((v, i) => console.log(`${i}: ${v}`)); // LOGS: 'a: A', 'b: B', 'c: c'
x.map((v, i) => i + v);                         // ['aA', 'bB, 'cc]
x.filter((v, i) => v !== 'B');                  // { a: 'A', c: 'c' }
x.reduce((a, v, i) => ({ ...a, [v]: i }), {});  // { A: 'a', B: 'b', c: 'c' }
x.slice(0, 2);                                  // ['A', 'B']
x.slice(-1);                                    // ['c']
x.find((v, i) => v === i);                      // 'c'
x.findKey((v, i) => v === 'B');                 // 'b'
x.includes('c');                                // true
x.includes('d');                                // false
x.keyOf('B');                                   // 'b'
x.keyOf('a');                                   // null
x.lastKeyOf('c');                               // 'c'

JS系列1-布尔陷阱以及如何避免