`
rainsilence
  • 浏览: 161547 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Javascript new 原理及模拟new

阅读更多

js的new可以看成是一个代理模式的代理类。包裹了new 后面的函数

处理顺序为
1.创建一个function对象,并将prototype设置为传入函数
2.执行传入函数
3.判断传入函数返回值,如果为null,则返回第一步的function对象。

 

实现代码:

模拟一个new。封装在newInstance方法里。

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>Insert title here</title>
<script type="text/javascript">
    function Animal(name) {
        this.name = name;
    }
    Animal.prototype.sayName = function() {
        alert("My name is " + this.name);
    }
    function newInstance(fn) {
        var Class = function() {};
        Class.prototype = fn.prototype;
        var slice = Array.prototype.slice;
        var args = slice.call(arguments);
        args.splice(0, 1);
        var instance = new Class();
        instance.constructor = fn;
        var result = fn.apply(instance, args);
        return (result != null) ? result : instance;
    }
    // 以上代码等同于new Animal("Jack")
    var cat = newInstance(Animal, "Jack");
    cat.sayName();
    alert(cat instanceof Animal); // true
</script>


</head>
<body>
    
</body>
</html>
 

 

分享到:
评论
18 楼 rainsilence 2011-05-23  
int08h 写道
prototypal inheritance能理解的人自然理解了也没什么好提的,不理解的人自然无法理解提不出啥,当然回不了那贴子了……
不过我对instanceof失效倒是感觉蛮遗憾的,一个语法层面上的东西,现在使用一个函数去弥补,总觉得别扭
比如我的某绘图工具里就有
if (shape instanceof Rectangle) {

}
else if (shape instanceof Circle) {

}
else {
    var drawer = getDrawerFromConfig(shape);
    drawer.draw();
}

哈哈,遇到知音啦。。我写得绘图工具里一开始也是这样
if (shape instanceof Rectangle) {

}
else if (shape instanceof Circle) {

}
else {


这样的代码可以用设计模式中的合成模式来解决。
以上代码最后可以替换为
shape.draw(drawer);
然后不同的shape去调用drawer不同的方法(利用shape中的Composite方法)。但是Circle这样的类就无法单单以DTO的身份出现了,而是作为一个部分

可惜这段经典的代码我写得时候是有版权的。虽然我脑子里有备份,但是作为职业道德无法发在这里。
17 楼 int08h 2011-05-23  
prototypal inheritance能理解的人自然理解了也没什么好提的,不理解的人自然无法理解提不出啥,当然回不了那贴子了……
不过我对instanceof失效倒是感觉蛮遗憾的,一个语法层面上的东西,现在使用一个函数去弥补,总觉得别扭
比如我的某绘图工具里就有
if (shape instanceof Rectangle) {

}
else if (shape instanceof Circle) {

}
else {
    var drawer = getDrawerFromConfig(shape);
    drawer.draw();
}
16 楼 rainsilence 2011-05-23  
zhanghh321 写道
我不是很理解楼主的意图:
我这么写
<script>
function f(){

}
var o2=new f();
alert(o2 instanceof f);
</script>
结果打印的也是true啊


我这篇文章的意图在于证明new是代理模式的一个实现。你这个是常规最基本的new的使用。
15 楼 zhanghh321 2011-05-23  
我不是很理解楼主的意图:
我这么写
<script>
function f(){

}
var o2=new f();
alert(o2 instanceof f);
</script>
结果打印的也是true啊
14 楼 rainsilence 2011-05-14  
我觉得我另外一篇文章的技术含量远远超过这篇,并且是一个无需使用function&prototype的继承方案。。怎么没人关注。。。
http://www.iteye.com/topic/1026530
13 楼 rainsilence 2011-05-13  
danny.chiu 写道
既然模拟new,具体实现里面又用到了new,有点理解不通,这篇文章目的是什么呢?在下笨,想请大家棒喝


你的想法很好啊。。
这篇文章的目的在于证明new是一种代理模式的思路。
如果一定要保证不出现new,可以变通实现
new Class改成
Object.create(Class.prototype);
12 楼 rainsilence 2011-05-13  
<div class="quote_title">frederick_hai 写道</div>
<div class="quote_div">
<div class="quote_title">rainsilence 写道</div>
<div class="quote_div">
<pre name="code" class="html">       <span style="color: #ff0000;"> return (result == null) ? result : instance;</span>
</pre>
  这句话逻辑错了,应该是return(result == null)? instance : result; 
<p> </p>
</div>
<p> </p>
</div>
<p>恩,是啊,写的时候是return(result)? result : instance;</p>
<p>为了大家能看清楚,匆忙改的。。</p>
<p>所以测试很重要啊</p>
11 楼 frederick_hai 2011-05-13  
<div class="quote_title">rainsilence 写道</div>
<div class="quote_div">
<pre name="code" class="html">       <span style="color: #ff0000;"> return (result == null) ? result : instance;</span>
</pre>
  这句话逻辑错了,应该是return(result == null)? instance : result; 
<p> </p>
</div>
<p> </p>
10 楼 kill_all 2011-05-13  
模拟new,用来了解Javascript里面new的机制。。。。。。。
thx楼主
9 楼 danny.chiu 2011-05-13  
既然模拟new,具体实现里面又用到了new,有点理解不通,这篇文章目的是什么呢?在下笨,想请大家棒喝
8 楼 rainsilence 2011-05-12  
lipeng88213 写道
 
引用
var result = fn.apply(instance, args);


调用instance.fn()吗?怎么理解这句话?


调用fn,并且将fn中的this绑定到instance上。
7 楼 lipeng88213 2011-05-12  
 
引用
var result = fn.apply(instance, args);


调用instance.fn()吗?怎么理解这句话?
6 楼 rainsilence 2011-05-12  
danny.chiu 写道
引用
处理顺序为

1.创建一个function对象,并将prototype设置为传入函数

2.执行传入函数

3.判断传入函数返回值,如果为null,则返回第一步的function对象。


学习了,请问下出处哈!

以前也忘了是哪本书,说法和您的不太一样,因为类实例化后得到的实际上是一个对象,他的意思可以这样用代码模拟:

function A() {}

// 实例化函数

var a = {};
A.apply(a, Array.prototype.slice.call(arguments));
return a;


具体还有其他细节,比如类的prototype,回去找下,但这两种思路很不一样,请rainsilence大哥斧正。



如int08h所言。。你这样做不管是instanceof还是isPrototypeOf,都无法判断类型了。

Moreover:

Array.prototype.slice.call只是用来做数组复制,因为arguments不是完整的array。但是apply函数支持直接传递arguments。所以如果不是要删除元素的,可以不必调用。
5 楼 rainsilence 2011-05-12  
int08h 写道
楼上的方法会丢失掉instanceof判定类型的能力,即:
new A() instanceof A <-- true
A() instanceof A <-- false

具体的构造过程其实是无法简单模拟的,因为涉及到2个私有属性[[Class]]和[[Construct]]
http://bclary.com/2004/11/07/#a-13.2.2


你说的是这个吧:
引用
13.2.2 [[Construct]]
When the [[Construct]] property for a Function object F is called, the following steps are taken:

1. Create a new native ECMAScript object.

2. Set the [[Class]] property of Result(1) to "Object".

3. Get the value of the prototype property of F.

4. If Result(3) is an object, set the [[Prototype]] property of Result(1) to Result(3).

5. If Result(3) is not an object, set the [[Prototype]] property of Result(1) to the original Object prototype object as described in 15.2.3.1.

6. Invoke the [[Call]] property of F, providing Result(1) as the this value and providing the argument list passed into [[Construct]] as the argument values.

7. If Type(Result(6)) is Object then return Result(6).

8. Return Result(1).


Step1,Step2可以看成我代码中的
var Class = function() {};

Step3, Step4, Step5(我这里缺少prototype==null时的判断。不过这里传进来的一般都是function,所以不需要)
Class.prototype = fn.prototype;

Step6
var result = fn.apply(instance, args); 

Step7 Step8
return (result == null) ? result : instance;

虽然缺少了Class和Construct属性。但是,这两个属性都是function的。从对象上来看,不会有什么不同。为了防止对象.constructor得到函数本身,特地加了一句,这样。就算得到了constructor,也是传入function参数的(具有Class属性),虽然Construct属性不是很完整,或者说可能不是完全一样,但是做的事情是一样的,外表上看不出来。。。。
4 楼 int08h 2011-05-11  
楼上的方法会丢失掉instanceof判定类型的能力,即:
new A() instanceof A <-- true
A() instanceof A <-- false

具体的构造过程其实是无法简单模拟的,因为涉及到2个私有属性[[Class]]和[[Construct]]
http://bclary.com/2004/11/07/#a-13.2.2
3 楼 danny.chiu 2011-05-11  
引用
处理顺序为

1.创建一个function对象,并将prototype设置为传入函数

2.执行传入函数

3.判断传入函数返回值,如果为null,则返回第一步的function对象。


学习了,请问下出处哈!

以前也忘了是哪本书,说法和您的不太一样,因为类实例化后得到的实际上是一个对象,他的意思可以这样用代码模拟:

function A() {}

// 实例化函数

var a = {};
A.apply(a, Array.prototype.slice.call(arguments));
return a;


具体还有其他细节,比如类的prototype,回去找下,但这两种思路很不一样,请rainsilence大哥斧正。

2 楼 rainsilence 2011-05-10  
对,
function Test() {
    return new Object();
}

// false
alert(new Test() instanceof Test)


function Test() {
}

// true
alert(new Test() instanceof Test);
1 楼 int08h 2011-05-10  
判断函数返回值这里,只要返回的是null/undefined或者所有基元类型,即不是Object的话,都会返回第一步的function对象

相关推荐

    计算机发展与计算机应用概述.pdf

    计算机发展与计算机应用概述.pdf

    计算机二级公共基础知识全集合.pdf

    计算机二级公共基础知识全集合.pdf

    计算机机试答案.pdf

    计算机机试答案.pdf

    基于STM32F103的750W全桥逆变器并离网设计方案及其实现

    内容概要:本文详细介绍了基于STM32F103RCT6的750W全桥逆变器设计方案,涵盖硬件电路设计、软件编程以及保护机制等方面。硬件部分包括主控芯片的选择、PWM配置、Boost升压电路、PCB布局优化等;软件部分涉及并离网切换的状态机设计、过流保护、风扇控制算法、并机功能实现等。文中还分享了许多实战经验和调试技巧,如死区时间配置、电流采样方法、并网同步算法等。 适合人群:具有一定电子电路和嵌入式开发基础的技术人员,尤其是从事逆变器及相关电力电子产品开发的工程师。 使用场景及目标:适用于希望深入了解逆变器工作原理和技术实现的开发者,特别是那些需要掌握并离网切换、高效电源管理及可靠保护机制的人群。目标是帮助读者构建一个稳定可靠的逆变器系统,能够应对各种复杂的工作环境。 其他说明:本文不仅提供了详细的理论讲解,还有丰富的代码片段和实践经验分享,有助于读者更好地理解和应用相关技术。

    基于Simulink的单相全桥逆变器仿真与优化:MATLAB环境下的详细实现

    内容概要:本文详细介绍了如何利用Simulink在MATLAB环境中搭建单相全桥逆变器的仿真模型。首先,通过构建H桥结构,连接直流电源和RL负载,并引入PWM控制器进行开关管的控制。接着,针对仿真过程中遇到的各种问题,如谐波失真、开关管直通等问题,提出了具体的解决方案,包括加入LC滤波器、设置死区时间和优化PWM参数等。此外,还探讨了通过MATLAB脚本自动化测试不同参数组合的方法,以及如何提高电压利用率和降低谐波失真。最终,通过对仿真结果的分析,验证了所提方法的有效性和优越性。 适合人群:电力电子工程师、科研人员、高校学生等对逆变器仿真感兴趣的群体。 使用场景及目标:适用于研究和开发高效、稳定的逆变器系统,旨在通过仿真手段减少实验成本,优化设计方案,提高系统的性能指标。 其他说明:文中提供了详细的建模步骤和技术细节,帮助读者更好地理解和掌握相关技术和方法。同时,强调了仿真参数的选择和优化对于获得理想仿真结果的重要性。

    计算机红外通信.pdf

    计算机红外通信.pdf

    软考考试学习必备资料.md

    软考考试学习必备资料.md

    基于cornerstonejs开发移动端

    基于cornerstonejs开发移动端

    JavaScript网页设计高级案例:构建交互式图片画廊#JavaScript

    构建交互式图片画廊

    在学习Wpf的过程中,手搓了一个2048

    源码

    Bosch Rexroth IndraWorks Ds IndraWorks Ds 14V16.310.0

    Bosch Rexroth IndraWorks Ds IndraWorks Ds 14V16.310.0

    java面向对象 - 类与对象

    java面向对象 - 类与对象

    电机控制领域无感FOC算法的AT32平台实现及其鲁棒性优化

    内容概要:本文详细介绍了基于AT32平台的无感FOC(Field-Oriented Control)控制算法,特别是针对永磁同步电机(PMSM)和无刷直流电机(BLDC)的位置速度观测器实现。文章首先展示了启动策略的独特之处,即跳过传统前馈强拖阶段,直接利用矢量控制环和观测器协同启动。接着深入探讨了磁链观测器的核心算法,包括磁链积分、反正切求角度以及速度估算部分使用的改良版PLL。此外,文中还提到了容差配置模块,用于提高系统的鲁棒性和稳定性。最后,强调了模块间良好的解耦设计,使得各功能模块拥有明确的输入输出接口,增强了代码的可维护性和移植性。 适合人群:从事电机控制系统开发的技术人员,尤其是对无感FOC算法感兴趣的工程师。 使用场景及目标:适用于需要高精度、快速响应的电机控制系统开发项目,旨在提升系统的鲁棒性和稳定性,特别是在电机参数存在偏差的情况下依然能够保持良好性能。 其他说明:文章不仅提供了详细的代码实现,还分享了许多实用的经验和技术细节,如启动策略、磁链观测器的物理本质、速度估算方法等,有助于读者更好地理解和应用无感FOC算法。

    计算机机房de设置与维护.pdf

    计算机机房de设置与维护.pdf

    《Java 面试进阶指北 》 质量很高,专为面试打造

    《Java 面试进阶指北 》 质量很高,专为面试打造

    外转子开关磁阻电机多目标优化的NSGA-II算法实现与Matlab代码解析

    内容概要:本文详细介绍了外转子开关磁阻电机(ER-SRM)的多目标优化方法,主要采用NSGA-II算法进行优化。文章首先解释了为什么ER-SRM比传统内转子电机更难以优化,接着展示了如何利用NSGA-II算法解决这一难题。文中提供了详细的Matlab代码,包括种群初始化、交叉变异操作、非支配排序以及目标函数的定义。此外,还讨论了优化过程中的一些注意事项,如初始种群多样性的保持、交叉变异参数的选择、目标函数的设计等。最后,通过具体的案例和图表展示了优化结果及其应用价值。 适合人群:从事电机设计与优化的研究人员和技术人员,尤其是对外转子开关磁阻电机感兴趣的读者。 使用场景及目标:适用于需要同时优化电机效率、转矩波动和制造成本等多种目标的情况。通过NSGA-II算法,可以在多个相互冲突的目标间找到最佳平衡点,从而提高电机的整体性能。 其他说明:文章不仅提供了完整的Matlab代码实现,还分享了许多实践经验,如参数设置的经验公式、常见错误及解决方案等。这对于理解和掌握NSGA-II算法的实际应用非常有帮助。

    "慢行智远"是一款专业的串口数据采集与波形分析软件 软件支持多通道波形显示、数据记录、协议解析等功能,界面友好,操作简便,是您进行串口通信与数据分析的得力助手

    慢行智远V2.0"是一款专业的串口数据采集与信号分析软件,集成了多通道数据采集、实时波形显示、FFT频谱分析、FIR滤波处理等高级功能。软件提供直观的用户界面,支持亮色/暗色两种主题,具备强大的数据处理与可视化能力。核心功能包括: 全面的串口通信支持(多种波特率、数据位、停止位、校验位配置) 多通道(最多4通道)波形实时显示与分析 高级信号处理(FFT频谱分析、FIR滤波、信号平滑等) 智能数据管理(断行数据处理、大数据量优化) 数据记录与导出(文本、CSV、图像多种格式) 自适应界面设计(支持高DPI显示、暗色主题) 适用人群 嵌入式开发工程师:需要通过串口调试单片机、开发板等嵌入式设备 电子工程师:进行电路测试、信号采集与分析的专业人员 工业自动化技术人员:监测工业设备数据、进行状态分析 科研教育工作者:用于实验数据采集、科学研究与教学演示 医疗设备开发人员:分析生物电信号、开发医疗监测设备 物联网开发者:调试传感器网络、分析传感器数据 硬件测试工程师:进行产品质量检测、性能评估 使用场景及目标 研发调试场景 单片机开发:实时监控传感器数据、调试通信协议、观察系统运行状态等等

    计算机基础- 图.pdf

    计算机基础- 图.pdf

    基于MATLAB和YALMIP的孤岛微电网MILP调度优化:最小化甩负荷与发电浪费

    内容概要:本文详细介绍了如何利用MATLAB和YALMIP工具箱构建并优化孤岛微电网的混合整数线性规划(MILP)调度模型。主要内容涵盖模型搭建的关键步骤,如定义决策变量、设置约束条件(尤其是电池充放电互斥约束)、处理光伏出力预测、设定目标函数以及选择求解器参数。文中强调了模型的实际应用场景,即在光伏板发电、电池储能和用户用电之间寻找最佳平衡,确保最小化甩负荷和发电浪费。此外,作者分享了一些实用技巧,如通过调整甩负荷惩罚系数α来优化调度策略,以及如何有效配置GUROBI求解器以缩短计算时间。 适合人群:从事电力系统优化、微电网调度研究的专业人士,以及对混合整数线性规划感兴趣的科研人员和技术开发者。 使用场景及目标:适用于需要精确控制发电、储电和用电的孤岛微电网系统。目标是在满足用户电力需求的同时,最大化利用可再生能源,减少化石燃料消耗,并延长电池使用寿命。 其他说明:文中提供了大量MATLAB代码片段,帮助读者更好地理解和实现具体的建模方法。同时,作者还提到了一些常见的陷阱和优化建议,有助于提高模型性能和求解效率。

    2025大模型时代的新能源汽车自动驾驶发展趋势.pdf

    2025大模型时代的新能源汽车自动驾驶发展趋势

Global site tag (gtag.js) - Google Analytics