JavaScript中的普通函数与构造函数比较


Posted in Javascript onApril 07, 2015

问题

什么是构造函数?
构造函数与普通函数区别是什么?
用new关键字的时候到底做了什么?
构造函数有返回值怎么办?
构造函数能当普通函数调用吗?

以下是我的一些理解,理解错误的地方恳请大家帮忙指正,谢谢!

this
this永远指向当前正在被执行的函数或方法的owner。例如:

function test(){
  console.log(this);
}
test();
//Window {top: Window, window: Window, location: Location, external: Object, chrome: Object…}

上面这段代码中,我们在页面中定义了一个test()函数,然后在页面中调用。函数定义在全局时,其owner就是当前页面,也就是window对象。

this指向的几种情况

1.全局中调用

    this.name //this指向window对象
   
2.函数调用

    test();//test()函数中的this也指向window对象
   
3.对象的方法调用

    obj1.fn();//obj1对象的fn()方法中的this指向obj1
   
4.调用构造函数
    var dog=new Dog();//构造函数内的this指向新创建的实例对象,也就是这里的dogcall和apply

call和apply的作用一样,只是接受参数的方式不一样,call接受的是多个单个参数,apply接受的是参数数组。
call和apply的作用简单地可以说成,当一个对象实例缺少一个函数/方法时,可以调用其他对象的现成函数/方法,其方式是通过替换其中的this为这个对象实例,改变函数运行时的上下文。
例如:

function Dog(){
  this.sound="汪汪汪";
}
Dog.prototype.bark=function(){
  alert(this.sound);
}

现在我有另外一个cat对象:

var cat={sound:'喵喵喵'}

我也想让这个cat对象可以调用bark方法,这时候就不用重新为它定义bark方法了,可以用call/apply调用Dog类的bark方法:

Dog.prototype.bark.call(cat);

或者:

dog.bark.call(cat);

加点东西,变成一个带参数的栗子:

function Dog(){
  this.sound="汪汪汪";
}
Dog.prototype.bark=function(words){
  alert(this.sound+" "+words);
}
var dog=new Dog();
dog.bark("有小偷");//alert:汪汪汪  有小偷
Dog.prototype.bark.call(cat,"饿了");//alert:喵喵喵  饿了

普通函数
这是一个简单的普通函数:

function fn(){
  alert("hello sheila");
}
fn();//alert:hello sheila

普通函数与构造函数相比有四个明显特点:

1.不需要用new关键字调用

    fn();2.可以用return语句返回值

function fn(a,b){
    return a+b;
  }
  alert(fn(2,3));//alert:5

3.函数内部不建议使用this关键字
我们说不建议使用,当然硬要用是可以的,只是要注意这时候发生了什么。如果在普通函数内部使用this关键字定义变量或函数,因为这时候this指向的是window全局对象,这样无意间就会为window添加了一些全局变量或函数。

function greeting(){
    this.name="sheila";
    alert("hello "+this.name);
  }
  greeting();//alert:hello sheila
  alert(window.name);//alert:sheila

4.函数命名以驼峰方式,首字母小写

构造函数
在JavaScript中,用new关键字来调用定义的构造函数。默认返回的是一个新对象,这个新对象具有构造函数定义的变量和函数/方法。

举个栗子:

function Prince(name,age){
  this.gender="male";
  this.kind=true;
  this.rich=true;
  this.name=name;
  this.age=age;
}
Prince.prototype.toFrog=function(){
  console.log("Prince "+this.name+" turned into a frog.");
}
var prince=new Prince("charming",25);
prince.toFrog();//Prince charming turned into a frog.
prince.kind;//true

与普通函数相比,构造函数有以下明显特点:

1.用new关键字调用

    var prince=new Prince("charming",25);

2.函数内部可以使用this关键字
在构造函数内部,this指向的是构造出的新对象。用this定义的变量或函数/方法,就是实例变量或实例函数/方法。需要用实例才能访问到,不能用类型名访问。

 prince.age;//25
    Prince.age;//undefined

3.默认不用return返回值
构造函数是不需要用return显式返回值的,默认会返回this,也就是新的实例对象。当然,也可以用return语句,返回值会根据return值的类型而有所不同,细节将在下文介绍。

4.函数命名建议首字母大写,与普通函数区分开。
不是命名规范中的,但是建议这么写。

使用new关键字实例化的时候发生了什么?
以上文中的Prince()函数举个栗子:

1.第一步,创建一个空对象。

var prince={}

2.第二步,将构造函数Prince()中的this指向新创建的对象prince。
3.第三步,将prince的_proto_属性指向Prince函数的prototype,创建对象和原型间关系
4.第四步,执行构造函数Prince()内的代码。

构造函数有return值怎么办?
构造函数里没有显式调用return时,默认是返回this对象,也就是新创建的实例对象。
当构造函数里调用return时,分两种情况:

1.return的是五种简单数据类型:String,Number,Boolean,Null,Undefined。
这种情况下,忽视return值,依然返回this对象。

2.return的是Object
这种情况下,不再返回this对象,而是返回return语句的返回值。

function Person(name){
    this.name=name;
    return {name:"cherry"}
  }
  var person=new Person("sheila");
  person.name;//cherry
  p;//Object {name: "cherry"}
Javascript 相关文章推荐
JQuery的AJAX实现文件下载的小例子
May 15 Javascript
node.js中的fs.ftruncate方法使用说明
Dec 15 Javascript
jQuery使用fadein方法实现渐出效果实例
Mar 27 Javascript
js实现向右横向滑出的二级菜单效果
Aug 27 Javascript
浏览器检测JS代码(兼容目前各大主流浏览器)
Feb 21 Javascript
Angularjs上传图片实例详解
Aug 06 Javascript
关于meta viewport中target-densitydpi属性详解(推荐)
Aug 18 Javascript
如何理解Vue的render函数的具体用法
Aug 30 Javascript
浅谈使用mpvue开发小程序需要注意和了解的知识点
May 23 Javascript
微信小程序实现slideUp、slideDown滑动效果及点击空白隐藏功能示例
Dec 11 Javascript
解决Vue在Tomcat8下部署页面不加载的问题
Nov 12 Javascript
如何在wxml中直接写js代码(wxs)
Nov 14 Javascript
jQuery控制cookie过期时间的方法
Apr 07 #Javascript
JavaScript随机生成信用卡卡号的方法
Apr 07 #Javascript
JavaScript实现信用卡校验方法
Apr 07 #Javascript
jQuery控制网页打印指定区域的方法
Apr 07 #Javascript
JavaScript使用指针操作实现约瑟夫问题实例
Apr 07 #Javascript
jquery.form.js实现将form提交转为ajax方式提交的方法
Apr 07 #Javascript
JavaScript中字符串分割函数split用法实例
Apr 07 #Javascript
You might like
PHP4之真OO
2006/10/09 PHP
怎样在PHP中通过ADO调用Asscess数据库和COM程序
2006/10/09 PHP
关于PHP开发的9条建议
2015/07/27 PHP
PHP编程实现阳历转换为阴历的方法实例
2017/08/08 PHP
读jQuery之三(构建选择器)
2011/06/11 Javascript
dojo学习第一天 Tab选项卡 实现
2011/08/28 Javascript
javascript列表框操作函数集合汇总
2013/11/28 Javascript
jquery中的$(document).ready()使用小结
2014/02/14 Javascript
简介JavaScript中valueOf()方法的使用
2015/06/05 Javascript
JS操作JSON方法总结(推荐)
2016/06/14 Javascript
javascript封装addLoadEvent实现页面同时加载执行多个函数的方法
2016/07/25 Javascript
js 判断各种数据类型的简单方法(推荐)
2016/08/29 Javascript
移动端Ionic App 资讯上下循环滚动的实现代码(跑马灯效果)
2017/08/29 Javascript
利用10行js代码实现上下滚动公告效果
2017/12/08 Javascript
vue 微信授权登录解决方案
2018/04/10 Javascript
用react-redux实现react组件之间数据共享的方法
2018/06/08 Javascript
D3.js实现拓扑图的示例代码
2018/06/30 Javascript
python socket网络编程步骤详解(socket套接字使用)
2013/12/06 Python
python生成指定尺寸缩略图的示例
2014/05/07 Python
Python模块搜索概念介绍及模块安装方法介绍
2015/06/03 Python
Python编程求质数实例代码
2018/01/31 Python
使用Python读取大文件的方法
2018/02/11 Python
selenium+PhantomJS爬取豆瓣读书
2019/08/26 Python
Python语法之精妙的十个知识点(装B语法)
2020/01/18 Python
详解Scrapy Redis入门实战
2020/11/18 Python
css 如何让背景图片拉伸填充避免重复显示
2013/07/11 HTML / CSS
Lulu & Georgia官方网站:购买地毯、家具、抱枕、壁纸、床上用品等
2018/03/19 全球购物
经贸日语专业个人求职信
2013/12/13 职场文书
演讲比赛获奖感言
2014/02/02 职场文书
公司请假条格式
2014/04/11 职场文书
给校长的建议书100字
2014/05/16 职场文书
村党支部书记四风问题个人对照检查材料思想汇报
2014/10/06 职场文书
自愿离婚协议书范文2014
2014/10/12 职场文书
毕业欢送晚会主持词
2019/06/25 职场文书
如何把新闻人物写得立体、鲜活?
2019/08/14 职场文书
php 防护xss,PHP的防御XSS注入的终极解决方案
2021/04/01 PHP