//创建一个类  其实就是个对象
var Student={
name:"robot",
height:1.6,
run:function(){
console.log(this.name+'runing');
}
}
function createStudent(name){
//基于Student原型创建一个新对象 Object.create()可以传入一个对象,创建出基于这个对象原型链的对象
var s = Object.create(Student);
// 初始化新对象:
s.name = name;
return s;
}
var xiaoming=createStudent("小明");
xiaoming.run();
console.log(xiaoming.__proto__==Student);//true

以上是一个原型链继承的例子

javascript创建对象

JavaScript对每个创建的对象都会设置一个原型,指向它的原型对象。

当我们用obj.xxx访问一个对象的属性时,JavaScript引擎先在当前对象上查找该属性,如果没有找到,就到其原型对象上找,如果还没有找到,就一直上溯到Object.prototype对象,最后,如果还没有找到,就只能返回undefined

例如,创建一个Array对象:

var arr = [1, 2, 3];

其原型链是:

arr ----> Array.prototype ----> Object.prototype ----> null

Array.prototype定义了indexOf()shift()等方法,因此你可以在所有的Array对象上直接调用这些方法。

当我们创建一个函数时:

function foo() {
return 0;
}

函数也是一个对象,它的原型链是:

foo ----> Function.prototype ----> Object.prototype ----> null

由于Function.prototype定义了apply()等方法,因此,所有函数都可以调用apply()方法。

容易想到,如果原型链很长,那么访问一个对象的属性就会因为花更多的时间查找而变得更慢,因此要注意不要把原型链搞得太长。

构造函数

除了直接用{ ... }创建一个对象外,JavaScript还可以用一种构造函数的方法来创建对象。它的用法是,先定义一个构造函数:

function Student(name) {
this.name = name;
this.hello = function () {
alert('Hello, ' + this.name + '!');
}
}

你会问,咦,这不是一个普通函数吗?

这确实是一个普通函数,但是在JavaScript中,可以用关键字new来调用这个函数,并返回一个对象:

var xiaoming = new Student('小明');
xiaoming.name; // '小明'
xiaoming.hello(); // Hello, 小明!

注意,如果不写new,这就是一个普通函数,它返回undefined。但是,如果写了new,它就变成了一个构造函数,它绑定的this指向新创建的对象,并默认返回this,也就是说,不需要在最后写return this;

新创建的xiaoming的原型链是:

xiaoming ----> Student.prototype ----> Object.prototype ----> null

也就是说,xiaoming的原型指向函数Student的原型。如果你又创建了xiaohongxiaojun,那么这些对象的原型与xiaoming是一样的:

xiaoming ↘
xiaohong -→ Student.prototype ----> Object.prototype ----> null
xiaojun ↗

new Student()创建的对象还从原型上获得了一个constructor属性,它指向函数Student本身:

xiaoming.constructor === Student.prototype.constructor; // true
Student.prototype.constructor === Student; // true Object.getPrototypeOf(xiaoming) === Student.prototype; // true xiaoming instanceof Student; // true

注意,用构造函数构造出来的对象是没有prototype属性的

构造函数有prototype属性

现在我们就认为xiaomingxiaohong这些对象“继承”自Student

不过还有一个小问题,注意观察:

xiaoming.name; // '小明'
xiaohong.name; // '小红'
xiaoming.hello; // function: Student.hello()
xiaohong.hello; // function: Student.hello()
xiaoming.hello === xiaohong.hello; // false

xiaomingxiaohong各自的name不同,这是对的,否则我们无法区分谁是谁了。

xiaomingxiaohong各自的hello是一个函数,但它们是两个不同的函数,虽然函数名称和代码都是相同的!

如果我们通过new Student()创建了很多对象,这些对象的hello函数实际上只需要共享同一个函数就可以了,这样可以节省很多内存。

要让创建的对象共享一个hello函数,根据对象的属性查找原则,我们只要把hello函数移动到xiaomingxiaohong这些对象共同的原型上就可以了,也就是Student.prototype

修改代码如下:

function Student(name) {
this.name = name;
} Student.prototype.hello = function () {//这是为了节省内存
alert('Hello, ' + this.name + '!');
};

new创建基于原型的JavaScript的对象就是这么简单!

忘记写new怎么办

如果一个函数被定义为用于创建对象的构造函数,但是调用时忘记了写new怎么办?

在strict模式下,this.name = name将报错,因为this绑定为undefined,在非strict模式下,this.name = name不报错,因为this绑定为window,于是无意间创建了全局变量name,并且返回undefined,这个结果更糟糕。

所以,调用构造函数千万不要忘记写new。为了区分普通函数和构造函数,按照约定,构造函数首字母应当大写,而普通函数首字母应当小写,这样,一些语法检查工具如jslint将可以帮你检测到漏写的new

最后,我们还可以编写一个createStudent()函数,在内部封装所有的new操作。一个常用的编程模式像这样:

function Student(props) {
this.name = props.name || '匿名'; // 默认值为'匿名'
this.grade = props.grade || 1; // 默认值为1
} Student.prototype.hello = function () {
alert('Hello, ' + this.name + '!');
}; function createStudent(props) {
return new Student(props || {})
}

这个createStudent()函数有几个巨大的优点:一是不需要new来调用,二是参数非常灵活,可以不传,也可以这么传:

var xiaoming = createStudent({
name: '小明'
}); xiaoming.grade; //

如果创建的对象有很多属性,我们只需要传递需要的某些属性,剩下的属性可以用默认值。由于参数是一个Object,我们无需记忆参数的顺序。如果恰好从JSON拿到了一个对象,就可以直接创建出xiaoming

最新文章

  1. C#开发奇技淫巧三:把dll放在不同的目录让你的程序更整洁
  2. SEO优化小技巧
  3. I Hate It HDOJ---1754
  4. linux 中环境变量配置错误导致部分命令不能使用包括vi
  5. php memcache 扩展 php -m 与 phpinfo() 不同
  6. CF518D. Ilya and Escalator [概率DP]
  7. 【iOS】swift-如何理解 if let 与guard?
  8. springboot增删改查
  9. IDEA打开maven项目dependencies红线
  10. [转帖]linux下的CPU、内存、IO、网络的压力测试
  11. maven error: element dependency can not have character children
  12. ThinkPHP的数据操作
  13. JS强制刷新页面、清除缓存刷新
  14. VS编程,WPF中两个滚动条 ScrollViewer 同步滚动的一种方法
  15. Matlab 基础
  16. block本质探寻四之copy
  17. linux 查询管道过滤,带上标题字段
  18. Python使用opencv
  19. SQL正则表达式
  20. AppScan 8.0.3安全漏洞扫描总结

热门文章

  1. Linux 网络工具netcat(nc)的应用
  2. Struts2的学习自我总结
  3. 记笔记的软件(vnote)
  4. instanceof关键字 与 getClass()
  5. Visual Studio新建类自动添加注释
  6. iOS 定义多个参数函数的写法
  7. 集合源码阅读——ArrayList
  8. footer始终在页面最底部的方法(问题待检验)
  9. CUDA中使用多维数组
  10. IEAD工具教你创建maven项目