视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
es6中代理的详细介绍(代码示例)
2020-11-27 19:29:00 责编:小采
文档


本篇文章给大家带来的内容是关于es6中代理的详细介绍(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

概述

代理嘛,就是请代理人代替自己做某件事,但是和自己不一样的是代理人可以有自己的行为,甚至可以做出和预期相违背的行为。

栗子

声明一个普通对象origin,他有一个属性name

let origin={
 name: 'origin'
}

声明一个代理对象

let proxy=new Proxy(origin, {
 get: (target, key)=>target[key]+" from proxy",
 set: (target, key, value)=>target[key]="set by proxy "+value
})

此时输出originproxy,可以发现,proxy拥有和origin一样的name属性

console.log(origin) // {name: 'origin'}
console.log(proxy) // Proxy {name: 'origin'}

origin添加age属性,再输出,可以发现,originproxy都拥有了age属性

origin.age=1 
console.log(origin) // {name: 'origin', age: '1'}
console.log(proxy) // Proxy {name: 'origin', age '1'}

那就是代理吗,当然不是,我们尝试为proxy添加属性

proxy.x=1
console.log(origin) // {name: 'origin', age: '1', x:'set by proxy 1'}
console.log(proxy) // Proxy {name: 'origin', age '1'}

可以发现,虽然originproxy都拥有了x属性,但是并不是我们赋值的1,而是多了set by proxy 几个字符串,很明显,这里是执行了初始化proxy时传入的第二个对象的set方法
那如果我们get

console.log(proxy.x) // set by proxy 1
console.log(proxy.x) // set by proxy 1 from proxy

现在很清楚了,proxy就是origin的代理,所有在proxy上的操作都会同步到origin上,而对origin的操作却不会同步到proxy上,而且proxy还有自己的行为。

可以这么想,proxy就是origin的秘书,所有的事务处理都要提交给秘书,秘书有自己的办事准则,可以直接提交给老板,也可以拒绝提交,或者添加一些其他的行为再提交。那这个秘书到底能代理老板做哪些事呢?

陷阱

语法

let p = new Proxy(target, handler);

初始化一个代理需要有两个参数

target:代理目标

handle:陷阱,是一个对象,我们的操作就像一只逃跑的动物,如果猎人在所有可以逃跑的路上全部放满了陷阱,那我们总是会落入一起一个的。本质就是一个对象,键描述我们的操作,值是函数,描述我们的行为,一共有13种陷阱。

0x003 set:设置属性

语法:

set(target, key, value)

target: 代理对象

key: 设置的属性

value: 设置的属性值

栗子:

let origin={}
let proxy=new Proxy(origin,{
 set:(target, key, value)=>{
 if(value>5)target[key]=value+10000
 }
})
proxy.x=1
proxy.y=10
console.log(proxy) // Proxy {y: 10010}
console.log(origin) // {y: 10010}

说明:
上面我们放置了一个set陷阱,当我们做set操作的时候,就会被捕捉到,我们判断value是否大于5,如果不大于5我们就不会做任何东西,但是如果大于5,就会给做赋值操作,并且还将他加上了10000。上面的栗子就相当于一个了。

get:访问属性

语法:

get(target, key)

target: 代理对象

key: 访问的属性

栗子:

let origin={
 x:1,
 y:2
}
let proxy=new Proxy(origin,{
 get:(target, key)=>{
 if(key==='x')return 'no'
 return target[key]
 }
})
console.log(proxy.x) // 'no'
console.log(proxy.y) // 2

deleteProperty:删除属性

语法:

deleteProperty(target, key)

target: 代理对象

key: 要删除的属性

栗子:

let origin={
 x:1,
 y:2
}
let proxy=new Proxy(origin,{
 deleteProperty:(target, key)=>{
 if(key==='x')return
 delete target[key]
 }
})
delete proxy.x
delete proxy.y
console.log(proxy) // {x:1}

has:判断是否包含某属性

语法:

has(target, key)

target: 代理对象

key: 要判断的属性

栗子:

let origin={
 x:1,
 y:2
}
let proxy=new Proxy(origin,{
 has:(target, key)=>{
 if(key==='x')return false
 return true
 }
})
console.log('x' in proxy) // false
console.log('y' in proxy) // true

ownKeys:获取自身属性值

  • 语法:

    ownKeys(target)
  • target: 代理对象

  • 栗子:

    let origin={
     x:1,
     y:2
    }
    let proxy=new Proxy(origin,{
     ownKeys:(target)=>{
     return ['y']
     }
    })
    console.log(Object.getOwnPropertyNames(proxy)) // ['y']
  • getPrototypeOf:获取prototype

  • 语法:

    getPrototypeOf(target)
  • target: 代理对象

  • 栗子

    let origin={
     x:1,
     y:2
    }
    let proxy=new Proxy(origin,{
     getPrototypeOf:(target)=>{
     return null
     }
    })
    console.log(Object.getPrototypeOf(p)) // null
  • setPrototypeOf:设置prototype

  • 语法:

    setPrototypeOf(target, prototype)
  • target: 代理对象

  • prototype: 要设置的prototype

  • 栗子

    let origin={
     x:1,
     y:2
    }
    let proxy=new Proxy(origin,{
     setPrototypeOf:(target, prototype)=>{
     throw 'no'
     }
    })
    Object.setPrototypeOf(proxy, {}) // Uncaught no
  • defineProperty :设置属性描述

  • 语法:

    defineProperty(target, prop, descriptor)
  • target: 代理对象

  • prop: 要设置描述的属性

  • descriptor: 描述

  • 栗子

    let origin={}
    let proxy=new Proxy(origin,{
     defineProperty:(target, prop, descriptor)=>{
     throw 'no'
     }
    })
    Object.defineProperty(proxy, 'x', {configurable: true}) // Uncaught no
  • getOwnPropertyDescriptor :获取自身属性描述

  • 语法:

    getOwnPropertyDescriptor(target, prop)
  • target: 代理对象

  • prop: 获取描述的属性

  • 栗子

    let origin={}
    let proxy=new Proxy(origin,{
     getOwnPropertyDescriptor:(target, prop)=>{
     throw 'no'
     }
    })
    Object.getOwnPropertyDescriptor(proxy, 'x') // Uncaught no
  • isExtensible:判断是否可扩展

  • 语法:

    isExtensible(target)
  • target: 代理对象

  • 栗子

    let origin={}
    let proxy=new Proxy(origin,{
     isExtensible:(target)=>{
     return false
     }
    })
    console.log(Object.isExtensible(proxy)); // false
  • preventExtensions :阻止扩展

  • 语法:

    preventExtensions(target)
  • target: 代理对象

  • 栗子:

    let origin={}
    let proxy=new Proxy(origin,{
     preventExtensions:(target)=>{
     return false;
     }
    })
    console.log(Object.preventExtensions(proxy)); // Uncaught TypeError: 'preventExtensions' on proxy: trap returned falsish
  • construct:构造

  • 语法:

    construct(target, argumentsList, newTarget)
  • target: 代理对象

  • argumentsList: 参数列表

  • newTarget: 新对象

  • 栗子:

    let Origin=function(){}
    let OriginProxy=new Proxy(Origin,{
     construct: function(target, argumentsList, newTarget) {
     throw 'error' 
     }
    })
    new OriginProxy() // Uncaught error
  • apply:调用

  • 语法:

    apply(target, thisArg, argumentsList)

    target: 代理对象

    thisArg: 上下文

    argumentsList: 参数列表

  • 栗子:

    let origin=function(){}
    let proxy=new Proxy(origin,{
     apply: function(target, thisArg, argumentsList) {
     throw 'error'
     }
    })
    origin() // Uncaught error
  • 下载本文
    显示全文
    专题