JavaScript中是如何定義私有變量的


前言

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三個變量的值,只能通過固定的 setNamegetNamesetAgegetAgesetSexgetSex等方法,而在所有的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構建私有變量的方法以及作用,如有錯誤和遺漏,歡迎指出,不勝感謝。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM