前言
JavaScript並不像別的語言,能使用關鍵字來宣告私有變數。
我瞭解的JavaScript能用來宣告私有變數的方式有兩種,一種是使用閉包,一種是使用WeakMap。
閉包
閉包的描述有很多種,比如:
能訪問其它函式作用域的函式;
內部函式訪問外部函式作用域的橋樑;
......
使用閉包構建私有變數的邏輯在於:
1.在外部函式中宣告變數和內部函式;
2.使用內部函式訪問或者修改變數值;
3.在外部函式內返回內部函式;
function outside(){
let val = 123;
function inside(){
return val;
}
return inside;
}
console.log(outside()());//123
通過我上面的例子能夠大致瞭解使用閉包構建私有變數的邏輯,但是不足以體現私有變數的重要性,一個const變數也能達到上述程式碼的效果:
//同樣的能訪問,但是不能修改,達到了上述程式碼的效果
const val = 123;
console.log(val);//123
接下來的程式碼,將具體體現私有變數的重要性:
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
從上面的程式碼中,可以看出,如果想要設定或者獲取 _name、_age、_sex三個變數的值,只能通過固定的 setName、getName、setAge、getAge、setSex、getSex等方法,而在所有的setter方法中,都對形參進行了判斷。也就意味著,對物件的所有操作都將在掌控之中,這在某一層面上弱化了JavaScript作為弱型別語言上的一些負面影響。
WeakMap
如果對WeakMap不是很瞭解的可以先看WeakMap的詳細介紹。
這裡主要是利用WeakMap的key不可列舉這一知識點。
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
同樣達成了構建私有變數的效果。順便提一句,class中構建私有變數用的就是WeakMap。
結尾
這篇文章只是記錄我知道的關於JavaScript構建私有變數的方法以及作用,如有錯誤和遺漏,歡迎指出,不勝感謝。