JavaScript 聲明私有變量的兩種方式
前言
JavaScript并不像別的語言,能使用關(guān)鍵字來聲明私有變量。
我了解的JavaScript能用來聲明私有變量的方式有兩種,一種是使用閉包,一種是使用WeakMap。
閉包
閉包的描述有很多種,比如:
能訪問其它函數(shù)作用域的函數(shù);
內(nèi)部函數(shù)訪問外部函數(shù)作用域的橋梁;
......
使用閉包構(gòu)建私有變量的邏輯在于:
1.在外部函數(shù)中聲明變量和內(nèi)部函數(shù);
2.使用內(nèi)部函數(shù)訪問或者修改變量值;
3.在外部函數(shù)內(nèi)返回內(nèi)部函數(shù);
function outside(){ let val = 123; function inside(){ return val; } return inside; } console.log(outside()());//123
通過我上面的例子能夠大致了解使用閉包構(gòu)建私有變量的邏輯,但是不足以體現(xiàn)私有變量的重要性,一個const變量也能達(dá)到上述代碼的效果:
//同樣的能訪問,但是不能修改,達(dá)到了上述代碼的效果 const val = 123; console.log(val);//123
接下來的代碼,將具體體現(xiàn)私有變量的重要性:
function person(){ let _name = 'unknown'; let _age = 18; let _sex = 'man'; function setName(name){ _name = name || 'unknown'; } function getName(){ return _name; } function setAge(age){ if(typeof age === 'number'){ _age = Math.floor(age); }else{ throw Error("typeof age !== 'number'"); } } function getAge(){ return _age; } function setSex(sex){ if(sex === 'man' || sex === 1){ _sex = 'man'; }else if(sex === 'woman' || sex === 0){ _sex = 'woman'; }else{ throw Error('input error'); } } function getSex(){ return _sex; } return { setName : setName, getName : getName, setAge : setAge, getAge : getAge, setSex : setSex, getSex : getSex } } let xiaoming = person(); let xiaohong = person(); xiaoming.setName('xiaoming'); xiaohong.setName('xiaohong'); console.log('xiaoming name : ' + xiaoming.getName());//xiaoming name : xiaoming console.log('xiaohong name : ' + xiaohong.getName());//xiaohong name : xiaohong xiaoming.setAge(19.3333); xiaohong.setAge('16');//Uncaught Error: typeof age !== 'number' console.log('xiaoming age : ' + xiaoming.getAge());//xiaoming age : 19 console.log('xiaohong age : ' + xiaohong.getAge());//xiaohong age : 18 xiaoming.setSex(1); xiaohong.setSex('woman'); console.log('xiaoming sex : ' + xiaoming.getSex());//xiaoming sex : man console.log('xiaohong sex : ' + xiaohong.getSex());//xiaohong sex : woman
從上面的代碼中,可以看出,如果想要設(shè)置或者獲取 _name、_age、_sex三個變量的值,只能通過固定的 setName、getName、setAge、getAge、setSex、getSex等方法,而在所有的setter方法中,都對形參進(jìn)行了判斷。也就意味著,對對象的所有操作都將在掌控之中,這在某一層面上弱化了JavaScript作為弱類型語言上的一些負(fù)面影響。
WeakMap
如果對WeakMap不是很了解的可以先看WeakMap的詳細(xì)介紹。
這里主要是利用WeakMap的key不可枚舉這一知識點(diǎn)。
let nameWeakMap = new WeakMap(); let ageWeakMap = new WeakMap(); let sexWeakMap = new WeakMap(); function person(){ let _hash = Object.create(null); nameWeakMap.set(_hash,'unknown'); ageWeakMap.set(_hash,18); sexWeakMap.set(_hash,'man'); function setName(name){ nameWeakMap.set(_hash,name || 'unknown'); } function getName(){ return nameWeakMap.get(_hash); } function setAge(age){ if(typeof age === 'number'){ ageWeakMap.set(_hash,Math.floor(age)); }else{ throw Error("typeof age !== 'number'"); } } function getAge(){ return ageWeakMap.get(_hash); } function setSex(sex){ if(sex === 'man' || sex === 1){ sexWeakMap.set(_hash,'man'); }else if(sex === 'woman' || sex === 0){ sexWeakMap.set(_hash,'woman'); }else{ throw Error('input error'); } } function getSex(){ return sexWeakMap.get(_hash); } return { setName : setName, getName : getName, setAge : setAge, getAge : getAge, setSex : setSex, getSex : getSex } } let xiaoming = person(); let xiaohong = person(); xiaoming.setName('xiaoming'); xiaohong.setName('xiaohong'); console.log('xiaoming name : ' + xiaoming.getName());//xiaoming name : xiaoming console.log('xiaohong name : ' + xiaohong.getName());//xiaohong name : xiaohong xiaoming.setAge(19.3333); xiaohong.setAge('16');//Uncaught Error: typeof age !== 'number' console.log('xiaoming age : ' + xiaoming.getAge());//xiaoming age : 19 console.log('xiaohong age : ' + xiaohong.getAge());//xiaohong age : 18 xiaoming.setSex(1); xiaohong.setSex('woman'); console.log('xiaoming sex : ' + xiaoming.getSex());//xiaoming sex : man console.log('xiaohong sex : ' + xiaohong.getSex());//xiaohong sex : woman
同樣達(dá)成了構(gòu)建私有變量的效果。順便提一句,class中構(gòu)建私有變量用的就是WeakMap。
結(jié)尾
這篇文章只是記錄我知道的關(guān)于JavaScript構(gòu)建私有變量的方法以及作用,如有錯誤和遺漏,歡迎指出,不勝感謝。
以上就是JavaScript 聲明私有變量的兩種方式的詳細(xì)內(nèi)容,更多關(guān)于JavaScript 聲明私有變量的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
理解Javascript_14_函數(shù)形式參數(shù)與arguments
在'執(zhí)行模型詳解'的'函數(shù)執(zhí)行環(huán)境'一節(jié)中對arguments有了些許的了解,那么讓我們深入的分析一下函數(shù)的形式參數(shù)與arguments的關(guān)系。2010-10-10JavaScript創(chuàng)建對象的幾種方式及關(guān)于this指向問題
這篇文章主要介紹了JavaScript創(chuàng)建對象的幾種方式及關(guān)于this指向問題,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值。需要的小伙伴可以參考一下2022-07-07JavaScript中“過于”犀利地for/in循環(huán)使用示例
Java中的增強(qiáng)for循環(huán)很是好用,但是JavaScript中為我們提供的for/in循環(huán)已然不是這么簡單了,下面有個簡單的示例,大家不妨參考下2013-10-10實(shí)例講解JavaScript中call、apply、bind方法的異同
這篇文章主要以實(shí)例講解的方式為大家總結(jié)了JavaScript中call、apply、bind方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-09-09JS中的常見數(shù)組遍歷案例詳解(forEach,?map,?filter,?sort,?reduce,?ever
這篇文章主要介紹了JS中的常見數(shù)組遍歷方法詳解(forEach,?map,?filter,?sort,?reduce,?every),本篇講用實(shí)際案例詳解他們的語法和用法,需要的朋友可以參考下2023-05-05javaScript嗅探執(zhí)行神器-sniffer.js
本文主要介紹了javaScript嗅探執(zhí)行神器-sniffer.js的相關(guān)知識。具有很好的參考價值,下面跟著小編一起來看下吧2017-02-02