JavaScript 声明私有变量的两种方式
前言
JavaScript并不像别的语言,能使用关键字来声明私有变量。
我了解的JavaScript能用来声明私有变量的方式有两种,一种是使用闭包,一种是使用WeakMap。
闭包
闭包的描述有很多种,比如:
能访问其它函数作用域的函数;
内部函数访问外部函数作用域的桥梁;
......
使用闭包构建私有变量的逻辑在于:
1.在外部函数中声明变量和内部函数;
2.使用内部函数访问或者修改变量值;
3.在外部函数内返回内部函数;
functionoutside(){ letval=123; functioninside(){ returnval; } returninside; } console.log(outside()());//123
通过我上面的例子能够大致了解使用闭包构建私有变量的逻辑,但是不足以体现私有变量的重要性,一个const变量也能达到上述代码的效果:
//同样的能访问,但是不能修改,达到了上述代码的效果 constval=123; console.log(val);//123
接下来的代码,将具体体现私有变量的重要性:
functionperson(){ let_name='unknown'; let_age=18; let_sex='man'; functionsetName(name){ _name=name||'unknown'; } functiongetName(){ return_name; } functionsetAge(age){ if(typeofage==='number'){ _age=Math.floor(age); }else{ throwError("typeofage!=='number'"); } } functiongetAge(){ return_age; } functionsetSex(sex){ if(sex==='man'||sex===1){ _sex='man'; }elseif(sex==='woman'||sex===0){ _sex='woman'; }else{ throwError('inputerror'); } } functiongetSex(){ return_sex; } return{ setName:setName, getName:getName, setAge:setAge, getAge:getAge, setSex:setSex, getSex:getSex } } letxiaoming=person(); letxiaohong=person(); xiaoming.setName('xiaoming'); xiaohong.setName('xiaohong'); console.log('xiaomingname:'+xiaoming.getName());//xiaomingname:xiaoming console.log('xiaohongname:'+xiaohong.getName());//xiaohongname:xiaohong xiaoming.setAge(19.3333); xiaohong.setAge('16');//UncaughtError:typeofage!=='number' console.log('xiaomingage:'+xiaoming.getAge());//xiaomingage:19 console.log('xiaohongage:'+xiaohong.getAge());//xiaohongage:18 xiaoming.setSex(1); xiaohong.setSex('woman'); console.log('xiaomingsex:'+xiaoming.getSex());//xiaomingsex:man console.log('xiaohongsex:'+xiaohong.getSex());//xiaohongsex:woman
从上面的代码中,可以看出,如果想要设置或者获取_name、_age、_sex三个变量的值,只能通过固定的setName、getName、setAge、getAge、setSex、getSex等方法,而在所有的setter方法中,都对形参进行了判断。也就意味着,对对象的所有操作都将在掌控之中,这在某一层面上弱化了JavaScript作为弱类型语言上的一些负面影响。
WeakMap
如果对WeakMap不是很了解的可以先看WeakMap的详细介绍。
这里主要是利用WeakMap的key不可枚举这一知识点。
letnameWeakMap=newWeakMap(); letageWeakMap=newWeakMap(); letsexWeakMap=newWeakMap(); functionperson(){ let_hash=Object.create(null); nameWeakMap.set(_hash,'unknown'); ageWeakMap.set(_hash,18); sexWeakMap.set(_hash,'man'); functionsetName(name){ nameWeakMap.set(_hash,name||'unknown'); } functiongetName(){ returnnameWeakMap.get(_hash); } functionsetAge(age){ if(typeofage==='number'){ ageWeakMap.set(_hash,Math.floor(age)); }else{ throwError("typeofage!=='number'"); } } functiongetAge(){ returnageWeakMap.get(_hash); } functionsetSex(sex){ if(sex==='man'||sex===1){ sexWeakMap.set(_hash,'man'); }elseif(sex==='woman'||sex===0){ sexWeakMap.set(_hash,'woman'); }else{ throwError('inputerror'); } } functiongetSex(){ returnsexWeakMap.get(_hash); } return{ setName:setName, getName:getName, setAge:setAge, getAge:getAge, setSex:setSex, getSex:getSex } } letxiaoming=person(); letxiaohong=person(); xiaoming.setName('xiaoming'); xiaohong.setName('xiaohong'); console.log('xiaomingname:'+xiaoming.getName());//xiaomingname:xiaoming console.log('xiaohongname:'+xiaohong.getName());//xiaohongname:xiaohong xiaoming.setAge(19.3333); xiaohong.setAge('16');//UncaughtError:typeofage!=='number' console.log('xiaomingage:'+xiaoming.getAge());//xiaomingage:19 console.log('xiaohongage:'+xiaohong.getAge());//xiaohongage:18 xiaoming.setSex(1); xiaohong.setSex('woman'); console.log('xiaomingsex:'+xiaoming.getSex());//xiaomingsex:man console.log('xiaohongsex:'+xiaohong.getSex());//xiaohongsex:woman
同样达成了构建私有变量的效果。顺便提一句,class中构建私有变量用的就是WeakMap。
结尾
这篇文章只是记录我知道的关于JavaScript构建私有变量的方法以及作用,如有错误和遗漏,欢迎指出,不胜感谢。
以上就是JavaScript声明私有变量的两种方式的详细内容,更多关于JavaScript声明私有变量的资料请关注毛票票其它相关文章!