当前位置: 首页 > news >正文

js prototype之诡异

想必经常写js的人必然会经常性的用到prototype这个属性,我写这篇文章倒不是自己对prototype这个属性理解有多深刻,相反是因为自己理解肤浅,想通过写文章来加深理解。废话不多说。下面总结一下prototype的特征吧,也请高手砖头拍轻点。

  • 特征一:“对象”是不可访问prototype属性,只有“类型”才可以访问portotype属性。
var x=0;
var y='1';
var z=[];
var w={};
var u=function(){}
// 以上对象类型中,仅有u是有protottype属性的,方法可以被看做是一种类型。
Array
Object
Function
Number
String
//以上是js的内置对象,它们是“类型”,都含有prototype属性

可以简单这么理解,可以通过new关键字的对象都包含prototype属性。

  • 特征二:prototype是保存实例方法的真正所在。

  valueof()、toString()、Array.push,call,apply等方法实际上都是保存在prototype属性的名下,只是我们实际使用这些方法时通过各自的实例来进行访问的。这样我们就可以通过prototype来重写这些方法了。

  1. 重写方法
    Number.prototype.toString = function () {
         return "Number:" + this;
    }
    var x= 222;
    alert(x.toString());

     上边显示的结果是:"Number:222";

    Function.prototype.toString = function () {
         return "function { native code }";
    }
    function aaa() {
          return Math.max(1, 2);
    }
    alert(aaa.toString()); 
    //给当前的脚本库进行简单的加密处理
  2. 扩充方法
Array.prototype.max = function () {
    for (var i = 0, max = this[0]; this[i]; i++) {
         if (max < this[i])
         max = this[i];
    }
    return max;
}
// 此方法仅限正数比较,字符串或其他对象无法比较
  • 特征三:prototype属性是特定类型对象的实例所共有的,也就是共享的一片内存区域,这片内存区域是在当前特定类型声明时创建的(如果特定类型是自定义函数,那么这片共享的内存区域就是在这个自定义函数声明时就开始创建)。对于特定类型的对象的prototype属性,他们在内存有且只保存着一份,只要你对prototype属性的修改,都会影响到所有的实例对象。
  • prototype的属性让js实现了晚绑定和极晚绑定

请看下面的代码:

 1       var x={};
 2       var y={};
 3       console.log(x===y);
 4       // 这里是false,相信大家都知道为什么,因为它们在内存中所处的地址不一样
 5      
 6       var x={};
 7       var y=x;
 8       console.log(x===y);
 9       // 这里是true。
10         
11         function Nothing() { }
12         Nothing.prototype.noArray = [];
13         Nothing.prototype.noObject = {};
14         var nothing1 = new Nothing();
15         var nothing2 = new Nothing();
16         console.log(nothing1.noArray === nothing2.noArray);
17         console.log(nothing1.noObject === nothing2.noObject);
18         // 此处,两个都为true,为什么呢,nothing1和nothing2公用了一片内存区域,它们都是指向空这个唯一的空对象和空数组的。

再举一个例证来说明:

        function M() { }
        M.prototype.val = 0;
        var m1 = new M();
        var m2 = new M();
        console.log(m1.val);
        console.log(m2.val);
        // 这里会发现,两个值都是0
        M.prototype.val = 2;
        console.log(m1.val);
        console.log(m2.val);
        // 这里会发现,另个值都是2
  • 特征四:当我们尝试通过“对象.属性=value”来修改prototype中的属性时,其实并没有修改portotype属性,而是给当前创建了一同名的属性,prototype中的属性还是保留着。

还是拿上边的代码继续进行分析:

function Nothing() { }
Nothing.prototype.noArray = [];
Nothing.prototype.noObject = {};
var nothing1 = new Nothing();
var nothing2 = new Nothing();
nothing1.noArray = null;
console.log(nothing1.noArray === nothing2.noArray);
console.log(nothing1.noObject === nothing2.noObject);
// 这个时候发现,第一个是false,第二个是true

我们看一下调试器发现,

我们发现nothing1对象多了一个普通的属性。也就是说,当你尝试对prototype中的属性进行修改的时候,你其实并没有修改prototype中的属性,而是新追加了一个同名的“普通”属性。

仔细分析上述代码,就会得出下面这条结论了。

  • 特征五:当访问一个对象的某个属性的时候,优先访问的是“普通”属性,其次再访问prototype中的属性。

这样就能解释上面代码中第一个为什么是false了。同样,当删除对象属性时,优先删除的仍然是这个“普通属性”。

        function Nothing() {}
        Nothing.prototype.noArray = [];
        Nothing.prototype.noObject = {};

        var nothing1 = new Nothing();
        var nothing2 = new Nothing();
        nothing1.noArray = null;
        console.log(nothing1.noArray === nothing2.noArray);
        console.log(nothing1.noObject === nothing2.noObject);
        // 第一个结果是false,第二个结果是true
        delete nothing1.noArray;
        console.log(nothing1.noArray === nothing2.noArray);
        // 这时候nothing1.noArray是[],结果为true;
        delete Nothing.prototype.noArray;
        console.log(nothing1.noArray === nothing2.noArray);
        // 这时候nothing1.noArray 和nothing2.noArray都为undefined,结果为true

转载于:https://www.cnblogs.com/dacuotecuo/p/3447639.html

相关文章:

  • 仿博彩业的转盘功能演示
  • 混合敏捷研发(一)SpecDD:混合的敏捷方法
  • 如何在Android模拟器上安装apk文件
  • 【转】iOS 宏(define)与常量(const)的正确使用-- 不错
  • 程序自动生成Dump文件
  • Python:urllib 和urllib2之间的区别
  • 华为实习日记——第十二天
  • Js 读写cookies
  • ExpandableListView(可展开的列表组件)的说明以及其用法
  • 搭建mcollective高可用,使puppet架构更加安全、稳定
  • MyEclipse中add jars和add external jars的区别带来的svn checkout的问题
  • JS实现网页换肤功能效果
  • android firmware 利用UDP socket发送Magic Packet--c语言版本
  • angular controller as syntax vs scope
  • 并发编程实战的阅读(锁的重入)
  • es6
  • JavaScript学习总结——原型
  • maven工程打包jar以及java jar命令的classpath使用
  • Meteor的表单提交:Form
  • Ruby 2.x 源代码分析:扩展 概述
  • Spring技术内幕笔记(2):Spring MVC 与 Web
  • 编写高质量JavaScript代码之并发
  • 关键词挖掘技术哪家强(一)基于node.js技术开发一个关键字查询工具
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 基于阿里云移动推送的移动应用推送模式最佳实践
  • 将 Measurements 和 Units 应用到物理学
  • 理解在java “”i=i++;”所发生的事情
  • 前言-如何学习区块链
  • 扫描识别控件Dynamic Web TWAIN v12.2发布,改进SSL证书
  • 使用iElevator.js模拟segmentfault的文章标题导航
  • 小程序开发中的那些坑
  • 一个6年java程序员的工作感悟,写给还在迷茫的你
  • 一些基于React、Vue、Node.js、MongoDB技术栈的实践项目
  • 正则学习笔记
  • ​低代码平台的核心价值与优势
  • ###51单片机学习(1)-----单片机烧录软件的使用,以及如何建立一个工程项目
  • #include
  • #QT(串口助手-界面)
  • (Git) gitignore基础使用
  • (MATLAB)第五章-矩阵运算
  • (poj1.3.2)1791(构造法模拟)
  • (附源码)springboot工单管理系统 毕业设计 964158
  • (黑马C++)L06 重载与继承
  • (蓝桥杯每日一题)love
  • (十三)Flask之特殊装饰器详解
  • (四)Controller接口控制器详解(三)
  • (原)记一次CentOS7 磁盘空间大小异常的解决过程
  • (转)nsfocus-绿盟科技笔试题目
  • .bat批处理(十一):替换字符串中包含百分号%的子串
  • .net core 连接数据库,通过数据库生成Modell
  • .net MySql
  • .NET 程序如何获取图片的宽高(框架自带多种方法的不同性能)
  • .net反混淆脱壳工具de4dot的使用
  • [2018/11/18] Java数据结构(2) 简单排序 冒泡排序 选择排序 插入排序
  • [ACM] hdu 1201 18岁生日