JS에서 ES6 프록시 프록시 사용법을 설명하는 기사(코드 공유)

奋力向前
풀어 주다: 2021-08-27 10:23:33
앞으로
1901명이 탐색했습니다.

이전 기사 "Vue에서 웹 프런트엔드 프로젝트 최적화에 대한 간략한 분석(코드 포함)"에서 Vue에서 웹 프런트엔드 프로젝트 최적화에 대해 배웠습니다. 다음 기사에서는 JS에서 ES6 프록시 프록시의 사용법을 소개합니다.

JS에서 ES6 프록시 프록시 사용법을 설명하는 기사(코드 공유)

프록시의 개념

프록시원래 영어 의미는 프록시입니다.ES6에서는 "에이전트"로 번역할 수 있습니다. 이는 주로 특정 작업의 기본 동작을 변경하는 데 사용되며 이는 언어 수준에서 수정하는 것과 동일하므로 일종의 "메타 프로그래밍"(메타 프로그래밍), 즉 프로그래밍입니다. 프로그래밍 언어.proxy英文原意是代理的意思,在ES6中,可以翻译为"代理器"。它主要用于改变某些操作的默认行为,等同于在语言层面做出修改,所以属于一种“元编程”(meta programming),即对编程语言进行编程。

proxy在目标对象的外层搭建了一层拦截,外界对目标对象的某些操作(后文会说明,有哪些操作可以拦截),必须通过这层拦截。语法

var proxy = new Proxy(target, handler);
로그인 후 복사

通过构造函数生成proxytarget参数是要拦截的目标对象,handler参数也是一个对象,用来定制拦截行为。

例子

var obj = new Proxy( {}, { get: function (target, key, receiver) { console.log(`getting ${key}!`); return Reflect.get(target, key, receiver); }, set: function (target, key, value, receiver) { console.log(`setting ${key}!`); return Reflect.set(target, key, value, receiver); }, } );
로그인 후 복사

一般将handle参数说成配置对象,在配置对象中,可以定义需要拦截的操作。如果配置对象为空,那么对proxy的操作将直通目标对象。

对proxy操作才有拦截效果,而不是目标对象。

Proxy实例的方法

当读取不存在的属性时候,抛出错误而不是返回undefined

var person = { name: "张三", }; var proxy = new Proxy(person, { get: function (target, property) { if (property in target) { return target[property]; } else { throw new ReferenceError('Property "' + property + '" does not exist.'); } }, }); proxy.name; // "张三" proxy.age; // 抛出一个错误
로그인 후 복사

拦截读取继承属性

let proto = new Proxy( {}, { get(target, propertyKey, receiver) { console.log("GET " + propertyKey); return target[propertyKey]; }, } ); let obj = Object.create(proto); obj.xxx; // "GET xxx"
로그인 후 복사

数组读取负数索引(负数索引表示倒着取数)

function createArray(...elements) { let handler = { get(target, propKey, receiver) { let index = Number(propKey); if (index < 0) { propKey = String(target.length + index); } return Reflect.get(target, propKey, receiver); }, }; let target = []; target.push(...elements); return new Proxy(target, handler); } let arr = createArray("a", "b", "c"); arr[-1]; // c
로그인 후 복사

实现数据的限制

let validator = { set: function (obj, prop, value) { if (prop === "age") { if (!Number.isInteger(value)) { throw new TypeError("The age is not an integer"); } if (value > 200) { throw new RangeError("The age seems invalid"); } } // 对于age以外的属性,直接保存 obj[prop] = value; }, }; let person = new Proxy({}, validator); person.age = 100; person.age; // 100 person.age = "young"; // 报错 person.age = 300; // 报错
로그인 후 복사

防止内部属性“_”被外部读写(通常我们以下划线开头,表示其实内部属性)

var handler = { get(target, key) { invariant(key, "get"); return target[key]; }, set(target, key, value) { invariant(key, "set"); target[key] = value; return true; }, }; function invariant(key, action) { if (key[0] === "_") { throw new Error(`Invalid attempt to ${action} private "${key}" property`); } } var target = {}; var proxy = new Proxy(target, handler); proxy._prop; // Error: Invalid attempt to get private "_prop" property proxy._prop = "c"; // Error: Invalid attempt to set private "_prop" property
로그인 후 복사

拦截——函数调用、callapply操作

var twice = { apply(target, ctx, args) { return Reflect.apply(...arguments) * 2; }, }; function sum(left, right) { return left + right; } var proxy = new Proxy(sum, twice); proxy(1, 2); // 6 proxy.call(null, 5, 6); // 22 proxy.apply(null, [7, 8]); // 30
로그인 후 복사

不对...in...循环生效

var handler = { has(target, key) { if (key[0] === "_") { return false; } return key in target; }, }; var target = { _prop: "foo", prop: "foo" }; var proxy = new Proxy(target, handler); "_prop" in proxy; // false
로그인 후 복사

不对for...in...循环生效

let stu1 = { name: "张三", score: 59 }; let stu2 = { name: "李四", score: 99 }; let handler = { has(target, prop) { if (prop === "score" && target[prop] < 60) { console.log(`${target.name} 不及格`); return false; } return prop in target; }, }; let oproxy1 = new Proxy(stu1, handler); let oproxy2 = new Proxy(stu2, handler); "score" in oproxy1; // 张三 不及格 // false "score" in oproxy2; // true for (let a in oproxy1) { console.log(oproxy1[a]); } // 张三 // 59 for (let b in oproxy2) { console.log(oproxy2[b]); } // 李四 // 99
로그인 후 복사

拦截object.keys()

프록시는 대상 객체의 외부 레이어에 차단 레이어를 구축합니다. 외부 세계에서 대상 객체에 대한 특정 작업(가로채울 수 있는 작업은 나중에 설명함)은 이 레이어를 통과해야 합니다. 차단의. 구문
let target = { a: 1, b: 2, c: 3, }; let handler = { ownKeys(target) { return ["a"]; }, }; let proxy = new Proxy(target, handler); Object.keys(proxy); // [ 'a' ]
로그인 후 복사

은 생성자를 통해프록시를 생성합니다.target매개변수는 가로챌 대상 객체이고핸들러 code> 매개변수는 차단 동작을 사용자 정의하는 데 사용되는 개체이기도 합니다.

rrreee일반적으로handle매개변수를 구성 개체라고 합니다. 구성 개체에서는 가로채야 하는 작업을 정의할 수 있습니다. 구성 개체가 비어 있으면프록시에 대한 작업이 대상 개체로 직접 이동합니다.

차단 효과는 대상 개체가 아닌 프록시 작업에만 적용됩니다. 프록시 인스턴스 방법존재하지 않는 속성을 읽을 때 정의되지 않음을 반환하는 대신 오류를 발생시킵니다.rrreee상속된 속성을 읽는 인터셉트rrreee 배열 읽기 음수 인덱스(음수 인덱스는 숫자를 거꾸로 가져오는 것을 의미) rrreee 내부 속성 " _"를 외부에서 읽고 쓰는 것을 방지하기 위해 데이터 제한을 구현합니다. rrreee(보통 밑줄로 시작하여 실제로 내부 속성) rrreee인터셉션 - 함수 호출, call, apply작업 rrreee은 ...in...에서 적용되지 않습니다. > 루프 rrreee잘못된 for...in...루프가 적용됩니다rrreee object.keys()메서드 차단rrreee이 문서의 출처 RYF 주소: https:/ /es6.ruanyifeng.com/#docs/proxy추천 학습: JS 고급 튜토리얼

위 내용은 JS에서 ES6 프록시 프록시 사용법을 설명하는 기사(코드 공유)의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
js
원천:chuchur.com
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿
회사 소개 부인 성명 Sitemap
PHP 중국어 웹사이트:공공복지 온라인 PHP 교육,PHP 학습자의 빠른 성장을 도와주세요!