Proxy代理数据拦截方法

proxy

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

new Proxy(target,handler)

// target 是proxy 要包装的对象 (可以是数组、函数,也可以是另一个Proxy)
//handler 一个通常以函数作为属性的对象,用来定制拦截行为

基本的语法是:

const p = new Proxy(target,handler)

主要的方法有:
handler.has() 是针对 in 操作符的代理方法
handler.set() 方法是设置属性值操作的捕获器。
handler.get() 方法用于拦截对象的读取属性操作。
handler.defineProperty() 用于拦截对对象的 Object.defineProperty() 操作。
handler.deleteProperty() 方法用于拦截对对象属性的 delete 操作。

handler.has()方法

const obj = {
    name: '微芒不朽',
    occupation: '前端开发'
}

const handler = {
    has(target, key) {
        //判断是否存在该属性
        return key in target
    }
}


const p = new Proxy(obj, handler)
console.log(p.name) //微芒不朽
console.log(p.like) //undefined
console.log(p.occupation) //前端开发

handler.get()方法

const obj = {
    name: '微芒不朽',
    occupation: '前端开发'
}

const handler = {
    has(target, key) {
        //判断是否存在该属性
        return key in target
    },
    get(target, key) {
        if (key in target) {
            return target[key]
        } else {
            return new ReferenceError(key + '属性不存在')
        }
    }
}

const p = new Proxy(obj, handler)
console.log(p.name) //微芒不朽
console.log(p.like) //ReferenceError: like属性不存在
console.log(p.occupation) //前端开发

handler.set()方法

const obj = {
    name: '微芒不朽',
    occupation: '前端开发'
}

const handler = {
    set(target, key) {
        if (key in target) {
            return Reflect.set(...arguments)
        }
        throw new ReferenceError(key+'属性不存在')
    }
}

const p = new Proxy(obj, handler)
p.like = '编程' //Uncaught ReferenceError: like属性不存在
// console.log(p.like)
p.occupation = '测试'
console.log(p.occupation) //测试

handler.defineProperty() 方法

用于拦截对对象的 Object.defineProperty() 操作

const obj = {
    name: '微芒不朽',
    configurable: true,
    enumerable: true,
}

const handler = {
    defineProperty(target, key, descriptor) {
        console.log('属性', key)
        return true;
    }
}

const p = new Proxy({}, handler)
Object.defineProperty(p, 'like', obj) //属性 like

也可以使用 Reflect.defineProperty

const obj = {
    name: '微芒不朽',
    configurable: true,
    enumerable: true,
}

const handler1 = {
    defineProperty(target, key, descriptor) {
        console.log('属性1', descriptor)
        return Reflect.defineProperty(target, key, descriptor)
    }
}

const p1 = new Proxy({}, handler)
Object.defineProperty(p1, 'like', obj) //属性 like

handler.deleteProperty() 方法

主要拦截对象的 delete操作;

const obj = {
    name: '微芒不朽',
}

const handler = {
    deleteProperty(target, key) {
        console.log('删除' + key)
    }
}

const p = new Proxy({}, handler)
delete p.name //删除name

//也会拦截 Reflect.deleteProperty
Reflect.deleteProperty(obj,'name')

Proxy.revocable 撤销代理

proxy有一个唯一的静态方法,Proxy.revocable(target, handler)
Proxy.revocable()方法可以用来创建一个可撤销的代理对象。 该方法的返回值是一个对象,其结构为: {"proxy": proxy, "revoke": revoke}

  • proxy 表示新生成的代理对象本身,和用一般方式 new Proxy(target, handler) 创建的代理对象没什么不同,只是它可以被撤销掉。
  • revoke 撤销方法,调用的时候不需要加任何参数,就可以撤销掉和它一起生成的那个代理对象。
const obj = {
    name: '微芒不朽',
}
const handler = {}
const { proxy, revoke } = Proxy.revocable(obj, handler)
console.log(proxy.name) //微芒不朽
revoke() // 取值完成对proxy进行封闭,撤消代理
console.log(proxy.name) // Uncaught TypeError: illegal operation attempted on a revoked proxy

应用场景

Proxy实现一个格式检验器

场景:验证身份证、电话、姓名和邮箱

const obj = {
    certno: '420101198101012964',
    name: '微芒不朽1',
    tel: '123456789',
    mail: '13@qq.com',
}

const validators = {
    //校验证件号码
    certno(val) {
        return /(^\d{15}$)|(^\d{18}$)|(^\d{17}(\d|X|x)$)/.test(val)
    },
    //校验姓名,只能为汉字
    name(val) {
        return /^[\u0391-\uFFE5]+$/.test(val)
    },
    //检验电话或手机号
    tel(val) {
        return /^1\d{10}$|^0\d{2,3}-?\d{7,8}$/.test(val)
    },
    //检验邮箱
    mail() {
        return /^[a-zA-Z0-9_.-]+@[a-zA-Z0-9-]+(\.[a-zA-Z0-9-]+)*\.[a-zA-Z0-9]{2,6}$/.test(val)
    }
}

const validatorType = (target, validator) => {
   return new Proxy(target, {
        _validator: validator,
        set(target, key, value, proxy) {
            let validator = this._validator[key](value)
            if (validator) {
                return Reflect.set(target, key, value, proxy)
            } else {
                throw Error(`设置${key} 的值为 ${value},格式不正确`)
            }
        }
    })
}

const proxy = validatorType(obj, validators)
proxy.certno = '420101198101012964'
proxy.name = '微芒不朽1' //Uncaught Error: 设置name 的值为 微芒不朽1,格式不正确
proxy.tel = '123456789' //Uncaught Error: 设置tel 的值为 123456789,格式不正确
proxy.mail = '13@qq.com'

proxy 拦截私有属性

用proxy拦截日常定义的私有属性,使其不能更改;一般以下划线开头;

let obj = {
    _id:'1234567890',
    name:'微芒不朽'
}
const p = new Proxy(obj,{
    set(target,prop){
        if(prop[0]==='_'){
            throw Error( `${prop}为私有属性`)
        }
        return Reflect.set(target,prop)
    }
})

p.name = '加油啊'
console.log(p.name) //加油啊
p._id = '123'
console.log(p._id) //Uncaught Error: _id为私有属性

Reactive函数

用来绑定引用数据类型, 例如对象和数组等,实现响应式。 Proxy 本质上是对某个对象的劫持,这样它不仅仅可以监听对象某个属性值的变化,还可以监听对象属性的新增和删除 。而 reactive 是 vue3 中对数据进行劫持的核心 。

//判断是否为对象
function isObject(value) {
    return value != null && (typeof value === 'object' || typeof value === 'function')
}

function reactive(obj) {
    if (!isObject(obj)) {
        return obj
    }
    return new Proxy(obj, {
        get(target, key) {
            // TODO:收集依赖
            return Reflect.get(target, key)
        },
        set(target, key, value) {
            // TODO:触发依赖
            return Reflect.set(target, key, value)
        }
    })
}
const state = reactive({
    name: '微芒不朽'
})
console.log(state) //Proxy代理的对象
// Proxy { <target>: {…}, <handler>: {…} }
// <target>: Object { name: "微芒不朽" }
// name: "微芒不朽"
// <prototype>: Object { … }
// <handler>: Object { get: get(target, key), set: set(target, key, value)
//  }
// get: function get(target, key)
// set: function set(target, key, value)
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容