在JavaScript中,函数(Function)是一种特殊的对象,每个函数都具有一个prototype属性,这个属性是指向一个对象的引用,被称为原型对象。原型对象可以包含共享的方法和属性,使得由该函数创建的所有实例对象(即通过new关键字创建的实例)都可以访问到原型中的这些方法和属性。
当我们创建一个函数时,如果未指定该函数的prototype属性,JavaScript引擎会自动为这个函数创建一个原型对象,并为其设置一个constructor属性,该constructor属性指向该函数本身。也就是说,每个函数都有一个默认的原型对象,其constructor属性指向函数本身。
例如:
```javascript
function foo() {
this.name = 'foo';
}
alert(foo.prototype.constructor === foo); // true
```
在上述代码中,`foo.prototype`是函数`foo`的原型对象,其上有一个constructor属性,该属性指向`foo`函数。
需要注意的是,`foo.prototype`和`Function.prototype`是两个不同的对象。`Function.prototype`是所有函数的原型,也就是所有函数对象的共同祖先。而`foo.prototype`是具体函数`foo`的原型,它作为构造函数`foo`创建的对象的原型。因此,`foo.prototype`并不等于`Function.prototype`。
关于`Function`的原型链,可以这样理解:
- `Function`是构造函数,用于构造其他函数,因此它本身也是函数。
- `Function`的原型(`Function.prototype`)是所有函数共享的原型对象。
- `Function`同时也是`Function`的实例,因为`Function`继承自自身。即`Function instanceof Function`返回`true`。
- 同样,所有的函数都继承自`Function.prototype`,因为它们都是由`Function`构造出来的。
`new`关键字的作用是创建一个新的对象,并且将这个新对象的内部属性`[[prototype]]`(在ES6中称为`__proto__`)链接到函数的`prototype`属性指向的那个原型对象上。然后以这个新创建的对象为上下文执行函数体(相当于`this`指向了新对象),最后返回这个新对象。
例如:
```javascript
function Person(name) {
this.name = name;
}
Person.prototype.getName = function() {
return this.name;
};
var person = new Person('张三');
alert(person.getName()); // '张三'
```
在上面的代码中,`Person`是一个构造函数,使用`new Person('张三')`创建了一个新对象`person`,这个对象的原型是`Person.prototype`。因此,`person`可以访问原型对象上的`getName`方法。
此外,如果我们修改了`foo.prototype`,不会影响到`Function.prototype`。也就是说,我们可以在`foo.prototype`上添加任意方法,或者将`foo.prototype`修改为任意对象,这不会对其他函数的原型造成影响。`Function.prototype`是所有函数共享的原型,是全局性的,除非我们手动修改,否则不会被改变。
总结一下,理解JavaScript中的原型和prototype属性是理解JavaScript面向对象编程的关键。每个函数都有一个prototype属性,指向一个原型对象,所有由这个函数构造出的实例共享这个原型对象的方法和属性。原型链是实现继承的主要方式,而`new`关键字则是构造新实例、连接原型链的桥梁。通过原型链,我们可以实现代码的复用,以及更深入地理解JavaScript中的对象和继承。