一、构造函数
说构造函数的概念之前我们应该先了解类的概念,类与对象的概念息息相关,我们在讲对象的概念的时候说过,一只猫是一个对象,一个人是一个对象,一章桌子也是一个对象,那么我们平时说的猫,人,桌子在没有特定指出是哪一个对象的时候,我们其实说的就是类。我们身边有人类,猫也是 一个类,任何有相同属性和方法的对象我们都可以称他们为一个类。
在很多面向对象语言中都有类的概念,但是在javascript(es5)中没有类。而我们可以通过构造函数来模拟一个类。
function Cat(){
this.name='miaomiao';
this.age=12;
this.sayName=function(){
console.log('我们是猫类');
}
}
var cat=new Cat();
console.log(cat.name);
cat.sayName();
构造函数的函数名首字母大写,可以通过在函数中的this为类设置属性和方法,我们在上面的代码中为猫这个类定义了一个name属性,并赋值为"miaomiao",类创建完成后,我们可以通过new关键字创建这个类的实例。例如cat这个变量就是Cat类的一个实例。
我们可以通过传参的方式,在实例化对象的时候为对象设置属性
function Cat(name,age){
this.name=name;
this.age=age;
this.sayName=function(){
console.log('我是'+this.name);
}
}
var cat=new Cat('mioamiao',12);
cat.sayName();
console.log(cat);
一个构造函数可以生成多个对象的实例
function Person(name,age){
this.name=name;
this.age=age;
this.say=function(){
console.log('大家好');
}
}
var lily=Person('lily',12);
var lucy=Person('lucy',13);
console.log(lily);
console.log(lucy);
console.log(lily.say==lucy.say);
一个构造函数Person生成了两个对象实例lily和lucy,并且有两个属性和一个方法。但是,它们的hobby方法是不一样的。也就是说,每当你使用new来调用构造函数放回一个对象实例的时候,都会创建一个hobby方法。这既没有必要,又浪费资源,hobby方法完全可以被两个对象实例共享。
构造函数的缺点就是:同一个构造函数的对象实例之间无法共享属性或方法。
为了解决构造函数的对象实例之间无法共享属性的缺点,js提供了prototype属性。
//每一个构造函数都有一个prototype属性,这个属性指向一个原型对象,这个原型对象上的所有属性和方法都可以被这个构造函数的实例使用
function Person(name,age){
this.name=name;
this.age=age;
}
Person.prototype.hobby=function(){
return 'movie';
}
var girl=new Cat('rose',12);
var boy=new Cat('jack',18);
console.log(girl.hobby==boy.hobby);
//如果将hobby方法放在原型对象上,那么两个实例对象都共享着同一个方法。
二、变量和流量
我们之前一直使用var定义变量,在ES6版本中,我们可以使用let定义变量,下面我们来说说var与let的区别。
1)块级作用域
//S5只有全局作用域和函数作用域,没有块级作用域的概念,这带来了很多不合理的场景。看下面的代码
for(var i = 0;i<10;i++){
console.log(i); // 0-9
}
console.log(i); // 10
//因为没有块级作用域,所以我们在for语句的外面仍然能获取i的值,在实际开发中,这是一个不可理喻的场景,我们希望的是这个i只在for语句内有效,所以再ES6中添加了块级作用域的概念,我们可以用let声明变量,问题就解决了
for(let i = 0;i<10;i++){
console.log(i); // 0-9
}
console.log(i); // 报错 i is not defined
//因为用let声明变量,变量只在块级作用域下有效,所以再for语句之外输出i会报错。
2)不存在变量提升
我们使用var定义变量会发生“变量提升”,我们来看下面的代码
console.log(number); //undefied
var number = 100;
console.log(string); //报错
let string = "hello world";
变量提升会导致我们在变量声明之前使用变量也不会报错,但是这个变量的值是undefined,这是不合理的,因为然我们要使用这个变量,那就应该提前声明,如果没有提前声明,就应该给予错误提示。所以再ES6中我们使用let声明变量,如果在声明变量前使用变量就会报错。
关于报错,有些同学会感觉不报错要比报错好,但实际开发中,如果程序运行与我们的语气不符,我们更希望看到错误信息,这样有利于我们更快地发现问题。
3)不允许重复声明
在ES5中,我们可以多次声明同一个变量
var a = 10;
var a = 20;
let b = 10;
let b = 20; //报错
//但是多次声明是没有意义的,在ES6中,我们使用let声明变量限制了不能多次声明,如果多次声明同一个变量回报错。
4)常量
在ES6中,不仅有变量,还增加了常亮的概念,我们用const声明常亮,一旦声明,它的值就不能再改变
const PI = 3.1415926;
PI = 3 //报错
我们说常亮不能再改变,说的是不能重新为这个常亮赋值,但是如果常亮存储的是一个对象,那我们是可以改变这个对象的属性的
const obj = {name:'小明'};
obj.name = '小红';
console.log(obj.name); //小红