JavaScript 之 對象屬性的特性 和defineProperty方法

来源:https://www.cnblogs.com/jiaobaba/archive/2019/11/19/11891276.html
-Advertisement-
Play Games

對象是無序屬性的集合,而這些屬性在創建是都帶有一些特征值(可以理解為屬性的屬性,天生自帶的),這些特征值是為了實現JavaScript引擎用的,因此JavaScript不能直接訪問。 JavaScript通過這些特征值來定義屬性的行為(屬性是否刪除,枚舉,修改等)。 例如,在全局定義的屬性是會掛載到 ...


對象是無序屬性的集合,而這些屬性在創建是都帶有一些特征值(可以理解為屬性的屬性,天生自帶的),這些特征值是為了實現JavaScript引擎用的,因此JavaScript不能直接訪問。

JavaScript通過這些特征值來定義屬性的行為(屬性是否刪除,枚舉,修改等)。

例如,在全局定義的屬性是會掛載到window上的。當想刪除window上的這個屬性,是不可以的。也就是說window上的屬性是不可配置的。delete window.obj     //false  

Function.prototype當你修改成其他值,其原始值並沒有改變。是不可寫的。

在比如,我們的for in是可以枚舉原型鏈上屬性的,但所有的原型頂端都是Object.prototype.但for in 並沒有枚舉出來。所以Object.prototype是不可枚舉的。

屬性分為兩種類型:1數據屬性 2訪問器屬性。例如:一般我們自己在對象設置的屬性預設是數據屬性,而Window上的那麼屬性是訪問器屬性。

怎樣知道這個屬性到底是數據屬性還是訪問器屬性?

使用Object.getOwnPropertyDescriptor(屬性所在的對象,屬性)方法。 返回一個對象,當時訪問器屬性時,該對象屬性有enumerable,configurable,get,set。當時數據屬性,該對象返回的屬性有value,writable,enumerable,configurable.

var obj  = {name:'zwq',age:18};
console.log(Object.getOwnPropertyDescriptor(obj,'name'));  //name屬性屬性
//{value: "zwq", writable: true, enumerable: true, configurable: true}
console.log(Object.getOwnPropertyDescriptor(window,'name')); //window上的name屬性時訪問器屬性
//enumerable: true, configurable: true, get: ƒ, set: ƒ}

 

數據屬性

  • 數據上行包含一個數據值的位置,可以讀取和寫入值,數據屬性有4個描述其行為的屬性,由於是這些值不能直接訪問,是內部值,所以該規範把他們放在兩對括弧中。
  • 屬性是否可配置:[[Configurable]]:表能否通過delete刪除屬性,能夠修改屬性的特性,能否把屬性修改為訪問器屬性
  • 屬性是否可枚舉:[[Enumerable]]:表能否通過for-in迴圈返回屬性
  • 屬性是否可修改:[[Writable]]
  • 屬性的數據值 :[[value]]讀取屬性的時候,從這個位置讀,寫入屬性的時候,把新值保存到這個位置。

普通定義的屬性 預設值前三個都是true,最後一個是undefined。

當我們想到修改屬性預設的特性,使用Object.defineProperty(屬性所在對象,屬性的名字,描述符對象)方法。

當使用Object.defineProperty方法第二個參數屬性的名字不存在時,該方法會創建這個屬性,並且該屬性的特性除了value剩下的特性的預設值都是false。也就是說當你想讓這個用Object.defineProperty方法創建的屬性跟正常的屬性一樣可枚舉,配置,寫入,必須把這個屬性值的特性都改為true。否則就是false。

 

     var obj  = {name:'zwq',age:18};
Object.defineProperty(obj,'name',{ 修改name屬性的特性,值為haha,並且name屬性不能修改值 value:'haha', writable:false //預設值是true,改為false,不可寫。 })
Object.defineProperty(obj,'sex',{創建一個sex屬性,這個屬性不可枚舉 value:'woman', writable:true, configurable:true, })

 

 

 

 

 訪問器屬性

訪問器屬性不包含writable和value,他包含的是一對getter和setter函數,在讀取訪問器屬性是,會調用getter函數,並返回有效的值,在寫入訪問器屬性時(修改屬性)會調用setter函數並傳入新值。訪問器包含4個特性

 

  • 屬性是否可配置:[[Configurable]]:表能否通過delete刪除屬性,能夠修改屬性的特性,能否把屬性修改為訪問器屬性
  • 屬性是否可枚舉:[[Enumerable]]:表能否通過for-in迴圈返回屬性
  • [[Get]]:在讀取屬性時調用的函數。預設值是undefined。
  • [[Set]]:在寫入(或修改)屬性時調用的函數。預設值undefined。

定義訪問器屬性,同樣也必須商用Object.defineProperty().

     function Person(){
            this._name = 'zwq',
            this.age =  18
        }
        var person = new Person();
        
        
        Object.defineProperty(person,'name',{
            set(newValue){
                console.log('set');
                this._name = newValue   //設置或修改屬性時,會調用set函數,把設置的值通過參數傳進去後,用一個變數或屬性保存。並且當調用get,return就是返回的這個值
            },
            get(){
                return this._name;   //當讀取屬性時 返回return的值
            }
            
        })

 不一定非要同時指定getter和setter,只指定getter意味著屬性是不能寫。

 

vue的雙向數據劫持綁定(主要應用於表單中)的原理就是利用Object.defineProperty來檢測數據的變化。

雙向劫持綁定時當視圖(頁面的某一元素)發生改變時,數據跟著改變,當數據改變時,視圖也跟著改變。例如下麵的輸入框裡面的內容改變時,數據(對象或數組)改變。檢測數據改變。底下的div文本根據數據的改變而改變。

 

 

 上面我們介紹到,當數據改變時會觸發set方法。由此我們就可以檢測數據的變化。

    //檢測對象的變化。
     var input = document.getElementById('Oinput');
        var view = document.getElementById('view');

        var data = {
            valueObj :{
                value:'zwq'
            }
        }
        //當輸入框數據發生改變時,數據跟著改變
        input.oninput = function(){
            data.valueObj.value = this.value;
        }
        
        // 更新視圖
        function upData(){   
            view.innerText = data.valueObj.value;
        }
        
        upData(data);
        obServe(data);

        // 監控某個對象是否發生改變
        function obServe(data){
           //判斷當前傳的是否是對象,如果不是,直接return
            if(!data || !(data instanceof Object)){return data}
      //獲取所有屬性名。使用keys方法可以獲取所有屬性名(包括原型上的)並保存帶數組中 var arrProperty = Object.keys(data);
       //遍曆數組,調用defindRective檢測每一個屬性值的改變  arrProperty.forEach(function(key){
          defindRective(data,key,data[key]); //傳入3個參數,當前對象,當前屬性,當前屬性值 }) } function defindRective(obj,key,val){ obServe(val); //使用遞歸,當想上面的數組,對象套對象的形式,由於裡面的對象是一個引用值,無法檢測裡面的數據變化,所以使用遞歸。 Object.defineProperty(obj,key,{ //核心:使用Object,definPropert的set檢測數據的改變。 set(newValue){ console.log(5); if(newValue == val) return val; val = newValue; upData(); //當數據變化,跟新視圖 }, get(){ return val; } }) }

 

// 監測數組,將數組原型重寫
// 當操作數組的arr push pop unshift slice...才會檢測
        let {push} = Array.prototype;
        var arr = [1,2,3];
        function upData(){
            console.log('更新');
        }
        Object.defineProperty(Array.prototype,'push',{
            value:(function(){
                return (...arg) => {
                    upData();
                    push.apply(arr,arg);
                }
            })()
        })
 arr.push(8,9);

 

 

由於使用Object.defineProperty檢測數組和對象變化要分開實現。而且,當添加數據時,不會檢測到的。

所以ES6中添加了Proxy來實現。

Proxy&reflect簡介:植入代理的模式,以簡潔易懂的方式控制對外部對象的訪問,利用set,get方法控制屬性的讀寫功能,還有其餘的has,desProperty。。等方法。但proxy相容性不好,使用時要註意。

proxy是一個構造函數,通過代理的方式將你想到代理的對象傳給構造函數,並且需要傳入參數對象對讀和寫進行控制。使用new方法實例化代理的對象,

此後,當修改,或添加屬性都使用代理的對象。

 

let data = {
    value:'zwq',
}
// let data = [1,2];
let oProxyData = new Proxy(data,{
    set(target,key,value,receiver){  //傳入4個參數 對象 屬性 屬性值 代理的對象
        // target[key] = value;
        Reflect.set(target,key,value); //等同於上一步
        upData();
  },
    get(target,key,receiver){
        // console.log(target,key);
        Reflect.get(target, key);
    },
    has(target,key){ //當使用in時觸發當前函數。
        return key in target;   //in --檢測對象能否訪問該屬性,能訪問返回true,不能false,無論是在實例還是原型中。
    }
});

console.log('valu' in oProxyData);

function upData(){
    console.log("更新啦");
}
oProxyData.value = 20;    





您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 原文:https://blog.csdn.net/fasfaf454/article/details/51438743 1、獲取手機系統信息( CPU,廠商名稱等)adb shell "cat /system/build.prop | grep "product""2、獲取手機系統版本adb she ...
  • 複合選擇器介紹 複合選擇器其實很好理解,說白了就跟我們生活中的有血緣關係家庭成員一樣,通過標簽或者 屬性或 屬性,去找對應的有血緣關係的某個選擇器,具體的大家往下看哦。 如果是初學者對基本的選擇器不是很瞭解的可以看筆者之前寫過的基本選擇器文章, "CSS基本選擇器是什麼?基本選擇器是如何工作" ,在 ...
  • 總結獲取url中查詢參數的兩種方式 通過正則表達式獲取單個參數 url中的所有查詢參數可以通過 欄位獲取,以字元串的形式返回。並有固定的格式 ,所以可以正則表達式匹配。 分析下需要匹配的格式: + , 其中需要獲取的是value部分,但是也需要 param= 參與匹配,但是不能參與返回結果,這裡有 ...
  • 本文介紹瞭如何使用Bootstrap實現響應式佈局,如何定製自己的響應式框架,Bootstrap響應式底層原理和模擬實現。 ...
  • css預處理器:用一種專門的編程語言,為CSS增加了一些編程的特性,將CSS作為目標生成文件,然後開發者就只要使用這種語言進行編碼工作,可以讓你的CSS更加簡潔、適應性更強、可讀性更佳,更易於代碼的維護等諸多好處。 ...
  • 有些時間我們希望能按需動態載入js文件,而不是直接在HTML中寫script標簽。 以下為示例代碼: 1 var js = document.createElement('script'); 2 js.async = true; 3 js.src = jsSrc; 4 js.onload = fun ...
  • [TOC] jQuery的補充 基本篩選器 屬性選擇器 表單選擇器 篩選器方法 下一個元素 上一個元素 父親元素 兒子和兄弟元素 操作標簽 樣式操作 樣式類 位置操作 尺寸 文本操作 HTML代碼 文本值 值 屬性操作 用於 和`radio` == 和`attr`的區別== 全稱 (屬性) 全稱 ( ...
  • [TOC] js BOM(瀏覽器對象模型) 由上圖可知,window對象是 BOM 的核心對象,所有對象都是通過它延伸出來的,也可以稱為window的子對象。 DOM是BOM的一部分 window對象 所有瀏覽器都支持window對象,它表示瀏覽器的介面。 常用window方法 ::瀏覽器視窗的內部 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...