什么是面向对象编程(OOP)?用对象的思想去写代码,就是面向对象编程。
面向对象编程的特点
对象的组成
- vararr = [];
- arr.number = 10;//对象下面的变量:叫做对象的属性
- //alert( arr.number );
- //alert( arr.length );
- arr.test =function(){//对象下面的函数 : 叫做对象的方法alert(123);
- };
- arr.test();//方法
- arr.push();//方法arr.sort();
创建一个对象
- var obj = new Object(); //创建一个空的对象
- obj.name = '小明'; //属性
- obj.showName = function() { //方法
- alert(this.name); //this指向obj
- }
- obj.showName(); //小明
如果需要创建两个或多个对象
- var obj1 = new Object(); //创建一个空的对象
- obj1.name = '小明'; //属性
- obj1.showName = function() { //方法
- alert(this.name); //this指向obj
- }
- obj1.showName(); //小明
- var obj2 = new Object(); //创建一个空的对象
- obj2.name = '小灰'; //属性
- obj2.showName = function() { //方法
- alert(this.name); //this指向obj
- }
- obj2.showName(); //小灰
使用 Object 函数或对象字面量都可以创建面向对象,但需要创建多个对象时,会产生大量的重复代码,可通过工厂方式来解决这个问题
工厂方式 -------------------- 面向对象中的封装函数
- //工厂方式 : 封装函数
- function createPerson(name){
- varobj =new Object();
- obj.name = name;
- obj.showName =function(){
- alert( this.name );
- };
- return obj;
- }
- varp1 = createPerson('小明');
- p1.showName();
- varp2 = createPerson('小强');
- p2.showName();
创建对象用工厂方式来实现,可以传递参数,由于创建对象都是使用 Object 的原生构造函数来实现的,因此无法识别对象类型
构造函数模式 -------------------- 给一个对象添加方法
- //new 后面调用的函数叫构造函数
- function CreatePerson(name){
- this.name=name;
- this.showName=function(){
- alert(this.name);
- }
- }
- varp1=newCreatePerson('小明');//当new去调用一个函数时,函数中的this就是创建出来的对象而函数中的返回值就是this
- p1.showName();
- varp2=newCreatePerson('小强');
- p2.showName();
使用自定义的构造函数,定义对象类型的属性和方法,与工厂方式的区别:
上面例子中:CreatePerson 构造函数生成的两个对象 p1 与 p2 都是 CreatePerson 的实例
虽然构造函数解决了上面工厂方式的问题,但是它一样存在缺点,就是在创建对象时,每个对象都有一套自己的方法,每定义一个函数都实例化了一个对象
例如:
- function CreatePerson(name){
- this.name = name;
- this.showName =function(){
- alert( this.name );
- };
- }
- varp1 =newCreatePerson('小明');
- //p1.showName();
- varp2 =newCreatePerson('小强');
- //p2.showName();
- alert( p1.showName == p2.showName );//false 它们的值相同,地址不同
测试例子中的 p1.showName 与 p2.showName 是否会相等,弹出的结果是 false,说明 p1 和 p2 实例都包含一个不同的 showName 实例
再来举几个例子:
- vara = [1,2,3];
- varb = [1,2,3];
- alert( a == b );//false 值相同,地址不同
- vara = 5;
- varb = a;
- b += 3
- alert(b);//8alert(a);//5 基本类型 : 赋值的时候只是值的复制
- var a = [1, 2, 3];
- var b = a;
- b.push(4);
- alert(b); //[1,2,3,4]
- alert(a); //[1,2,3,4] 对象类型 : 赋值不仅是值的复制,而且也是引用的传递
- var a = [1, 2, 3];
- var b = a;
- b = [1, 2, 3, 4];
- alert(b); //[1,2,3,4]
- alert(a); //[1,2,3]
对比上面的几个例子,不难看出基本类型和对象类型的区别了,对象类型的赋值不仅是值的复制,也是引用的传递;提到了对象的引用应该很清楚上述 p1.showName==p2.showName 为何会返回结果是 false
原型模式(prototype) -------------------- 给一类对象添加方法
原型(prototype):重写对象下面公用的属性或方法,让公用的属性或方法在内存中只存在一份(提高性能),也就是说所有在原型对象中创建的属性或方法都直接被所有对象实例共享。
- var arr = [1, 2, 3, 4, 5];
- var arr2 = [2, 2, 2, 2, 2];
- Array.prototype.sum = function() { //原型prototype : 要写在构造函数的下面
- var result = 0;
- for (var i = 0; i < this.length; i++) {
- result += this[i];
- }
- return result;
- };
- alert(arr.sum()); //15
- alert(arr2.sum()); //10
原型优先级:如果在实例中添加了一个属性,而该属性与实例原型中的一个属性同名,该属性将会屏蔽原型中的那个属性
例子 1:
- var arr = [];
- arr.number = 10;
- Array.prototype.number = 20;
- alert(arr.number); //10
例子 2:
- Array.prototype.a = 12; //原型属性
- var arr = [1, 2, 3];
- alert(arr.a); //12
- arr.a = 5; //实例属性
- alert(arr.a); //5
工厂方式之原型
- function CreatePerson(name) { //普通方法
- this.name = name;
- }
- CreatePerson.prototype.showName = function() { //原型
- alert(this.name);
- }
- var p1 = new CreatePerson('小明');
- p1.showName();
- var p2 = new CreatePerson('小强');
- p2.showName();
- alert(p1.showName == p2.showName); //true
由上述例子中:p1.showName== p2.showName 弹出的结果是 true,可见原型解决了构造函数中 "每定义一个函数都实例化了一个对象" 的问题
原型的运用
选项卡实例:
- <!DOCTYPE html>
- <html>
- <head lang="en">
- <meta charset="UTF-8">
- <title>
- 选项卡
- </title>
- <style>
- #div1 div{ width:400px; height:300px; border:1px solid #ccc; overflow:
- hidden; display: none; margin: 15px 0; } #div1 input{ color: #fff; width:100px;
- height:40px; background: darkseagreen; border:none; font-size: 14px; letter-spacing:
- 5px; } #div1 p{ font-size: 20px; line-height: 24px; text-align: center;
- color:darkgreen; } #div1 .title{ padding: 0; font-weight: bold; } #div1
- .active{ background:sandybrown; color:#fff; }
- </style>
- <script>
- window.onload = function() {
- var oDiv = document.getElementById('div1');
- var aInput = oDiv.getElementsByTagName('input');
- var aDiv = oDiv.getElementsByTagName('div');
- var i = 0;
- for (i = 0; i < aInput.length; i++) {
- aInput[i].index = i;
- aInput[i].onmousemove = function() {
- for (var i = 0; i < aInput.length; i++) {
- aInput[i].className = '';
- aDiv[i].style.display = 'none';
- }
- aInput[this.index].className = 'active';
- aDiv[this.index].style.display = 'block';
- }
- }
- }
- </script>
- </head>
- <body>
- <div id="div1">
- <input class="active" type="button" value="五言律诗">
- <input type="button" value="七言律诗">
- <input type="button" value="五言绝句">
- <input type="button" value="七言绝句">
- <div style="display: block;">
- <p class="title">
- 落 花
- </p>
- <p class="author">
- 李商隐
- </p>
- <p>
- 高阁客竟去,小园花乱飞。
- </p>
- <p>
- 参差连曲陌,迢递送斜晖。
- </p>
- <p>
- 肠断未忍扫,眼穿仍欲归。
- </p>
- <p>
- 芳心向春尽,所得是沾衣。
- </p>
- </div>
- <div>
- <p class="title">
- 蜀 相
- </p>
- <p class="author">
- 杜甫
- </p>
- <p>
- 丞相祠堂何处寻,锦官城外柏森森。
- </p>
- <p>
- 映阶碧草自春色,隔叶黄鹂空好音。
- </p>
- <p>
- 三顾频烦天下计,两朝开济老臣心。
- </p>
- <p>
- 出师未捷身先死,长使英雄泪满襟。
- </p>
- </div>
- <div>
- <p class="title">
- 八阵图
- </p>
- <p class="author">
- 杜甫
- </p>
- <p>
- 功盖三分国,名成八阵图。
- </p>
- <p>
- 江流石不转,遗恨失吞吴。
- </p>
- </div>
- <div>
- <p class="title">
- 泊秦淮
- </p>
- <p class="author">
- 杜牧
- </p>
- <p>
- 烟笼寒水月笼沙,夜泊秦淮近酒家。
- </p>
- <p>
- 商女不知亡国恨,隔江犹唱后庭花。
- </p>
- </div>
- </div>
- </body>
- </html>
效果(鼠标经过按钮时选项卡切换):
面向对象选项卡:
- <!DOCTYPE html>
- <html>
- <head lang="en">
- <meta charset="UTF-8">
- <title>
- 选项卡
- </title>
- <style>
- #div1 div,#div2 div{ width:400px; height:300px; border:1px solid #ccc;
- overflow: hidden; display: none; margin: 15px 0; } #div1 input,#div2 input{
- color: #fff; width:100px; height:40px; background: darkseagreen; border:none;
- font-size: 14px; letter-spacing: 5px; } #div1 p,#div2 p{ font-size: 20px;
- line-height: 24px; text-align: center; color:darkgreen; } #div1 .title,#div2
- .title{ padding: 0; font-weight: bold; } #div1 .active,#div2 .active{ background:sandybrown;
- color:#fff; }
- </style>
- <script>
- window.onload = function() {
- var t1 = new TabSwitch('div1');
- t1.
- switch ();
- var t2 = new TabSwitch('div2'); //面向对象的复用性
- t2.
- switch ();
- t2.autoPlay();
- /*alert(t2.switch==t1.switch);//ture*/
- }
- function TabSwitch(id) {
- this.oDiv = document.getElementById(id);
- this.aInput = this.oDiv.getElementsByTagName('input');
- this.aDiv = this.oDiv.getElementsByTagName('div');
- this.iNow = 0; //自定义属性
- }
- TabSwitch.prototype.
- switch = function() { //原型
- for (var i = 0; i < this.aInput.length; i++) {
- var This = this; //将指向面向对象的this保存下来
- this.aInput[i].index = i;
- this.aInput[i].onmousemove = function() {
- This.tab(this); //This指向面向对象 this指向this.aInput[i]
- }
- }
- }
- TabSwitch.prototype.tab = function(obj) { //原型
- for (var i = 0; i < this.aInput.length; i++) {
- this.aInput[i].className = '';
- this.aDiv[i].style.display = 'none';
- }
- this.aInput[obj.index].className = 'active';
- this.aDiv[obj.index].style.display = 'block';
- }
- //自动播放
- TabSwitch.prototype.autoPlay = function() {
- var This = this;
- setInterval(function() {
- if (This.iNow == This.aInput.length - 1) {
- This.iNow = 0;
- } else {
- This.iNow++;
- }
- for (var i = 0; i < This.aInput.length; i++) {
- This.aInput[i].className = '';
- This.aDiv[i].style.display = 'none';
- }
- This.aInput[This.iNow].className = 'active';
- This.aDiv[This.iNow].style.display = 'block';
- },
- 1000);
- }
- </script>
- </head>
- <body>
- <div id="div1">
- <input class="active" type="button" value="五言律诗">
- <input type="button" value="七言律诗">
- <input type="button" value="五言绝句">
- <input type="button" value="七言绝句">
- <div style="display: block;">
- <p class="title">
- 落 花
- </p>
- <p class="author">
- 李商隐
- </p>
- <p>
- 高阁客竟去,小园花乱飞。
- </p>
- <p>
- 参差连曲陌,迢递送斜晖。
- </p>
- <p>
- 肠断未忍扫,眼穿仍欲归。
- </p>
- <p>
- 芳心向春尽,所得是沾衣。
- </p>
- </div>
- <div>
- <p class="title">
- 蜀 相
- </p>
- <p class="author">
- 杜甫
- </p>
- <p>
- 丞相祠堂何处寻,锦官城外柏森森。
- </p>
- <p>
- 映阶碧草自春色,隔叶黄鹂空好音。
- </p>
- <p>
- 三顾频烦天下计,两朝开济老臣心。
- </p>
- <p>
- 出师未捷身先死,长使英雄泪满襟。
- </p>
- </div>
- <div>
- <p class="title">
- 八阵图
- </p>
- <p class="author">
- 杜甫
- </p>
- <p>
- 功盖三分国,名成八阵图。
- </p>
- <p>
- 江流石不转,遗恨失吞吴。
- </p>
- </div>
- <div>
- <p class="title">
- 泊秦淮
- </p>
- <p class="author">
- 杜牧
- </p>
- <p>
- 烟笼寒水月笼沙,夜泊秦淮近酒家。
- </p>
- <p>
- 商女不知亡国恨,隔江犹唱后庭花。
- </p>
- </div>
- </div>
- <div id="div2">
- <input class="active" type="button" value="五言律诗">
- <input type="button" value="七言律诗">
- <input type="button" value="五言绝句">
- <input type="button" value="七言绝句">
- <div style="display: block;">
- <p class="title">
- 落 花
- </p>
- <p class="author">
- 李商隐
- </p>
- <p>
- 高阁客竟去,小园花乱飞。
- </p>
- <p>
- 参差连曲陌,迢递送斜晖。
- </p>
- <p>
- 肠断未忍扫,眼穿仍欲归。
- </p>
- <p>
- 芳心向春尽,所得是沾衣。
- </p>
- </div>
- <div>
- <p class="title">
- 蜀 相
- </p>
- <p class="author">
- 杜甫
- </p>
- <p>
- 丞相祠堂何处寻,锦官城外柏森森。
- </p>
- <p>
- 映阶碧草自春色,隔叶黄鹂空好音。
- </p>
- <p>
- 三顾频烦天下计,两朝开济老臣心。
- </p>
- <p>
- 出师未捷身先死,长使英雄泪满襟。
- </p>
- </div>
- <div>
- <p class="title">
- 八阵图
- </p>
- <p class="author">
- 杜甫
- </p>
- <p>
- 功盖三分国,名成八阵图。
- </p>
- <p>
- 江流石不转,遗恨失吞吴。
- </p>
- </div>
- <div>
- <p class="title">
- 泊秦淮
- </p>
- <p class="author">
- 杜牧
- </p>
- <p>
- 烟笼寒水月笼沙,夜泊秦淮近酒家。
- </p>
- <p>
- 商女不知亡国恨,隔江犹唱后庭花。
- </p>
- </div>
- </div>
- </body>
- </html>
效果(第二个选项卡加了一个自动切换功能):
面向对象中 this 的问题
一般会出现问题的情况有两种:
例子 1:
- //定时器
- function Aaa() {
- var _this = this; //将当前this值保存
- this.a = 12;
- setInterval(function() { //定时器中this指向window
- _this.show();
- },
- 1000);
- }
- Aaa.prototype.show = function() {
- alert(this.a);
- }
- var obj = new Aaa(); //12
例子 2:
- <!DOCTYPE html>
- <html>
- <head lang="en">
- <meta charset="UTF-8">
- <title>
- 面向对象中this的问题-----事件
- </title>
- <script>
- function Bbb() {
- var _this = this;
- this.b = 5;
- document.getElementById('btn1').onclick = function() { //点击事件
- _this.show();
- }
- }
- Bbb.prototype.show=function(){ alert(this.b); } window.onload=function(){
- var p2=new Bbb();
- }
- </script>
- </head>
- <body>
- <input id="btn1" type="button" value="按钮">
- </body>
- </html>
上面两个是分别对定时器和事件中 this 问题的解决方法,即将指向对象的 this 保存到了_this 中,在嵌套函数中调用对象的方法或属性时用 _this. 属性 或 _this. 方法
再来个实例:
拖拽效果:
- <!DOCTYPE html>
- <html>
- <head lang="en">
- <meta charset="UTF-8">
- <title>
- 最初写的拖拽效果
- </title>
- <style>
- #div1{ width:100px; height:100px; background: red; position: absolute;
- }
- </style>
- <script>
- window.onload = function() {
- var oDiv = document.getElementById('div1');
- oDiv.onmousedown = function(ev) {
- var oEvent = ev || event;
- var disX = 0;
- var disY = 0;
- var disX = oEvent.clientX - oDiv.offsetLeft;
- var disY = oEvent.clientY - oDiv.offsetTop;
- document.onmousemove = function(ev) {
- var oEvent = ev || event;
- oDiv.style.left = oEvent.clientX - disX + 'px';
- oDiv.style.top = oEvent.clientY - disY + 'px';
- };
- document.onmouseup = function() {
- document.onmousemove = null;
- document.onmouseup = null;
- };
- return false;
- }
- }
- </script>
- </head>
- <body>
- <div id="div1">
- </div>
- </body>
- </html>
面向对象的拖拽
- <!DOCTYPE html>
- <html>
- <head lang="en">
- <meta charset="UTF-8">
- <title>
- 面向对象写的拖拽效果
- </title>
- <style>
- #div1{ width:100px; height:100px; background: red; position: absolute;
- }
- </style>
- <script>
- window.onload = function() {
- var p = new Darg('div1');
- p.init();
- }
- function Darg(id) {
- this.oDiv = document.getElementById(id); //属性
- this.disX = 0; //属性
- this.disY = 0; //属性
- }
- Darg.prototype.init = function() { //原型 方法
- var This = this;
- this.oDiv.onmousedown = function(ev) {
- var oEvent = ev || event;
- This.fnDown(oEvent);
- return false;
- }
- }
- Darg.prototype.fnDown = function(ev) { //原型 方法
- var This = this;
- this.disX = ev.clientX - this.oDiv.offsetLeft;
- this.disY = ev.clientY - this.oDiv.offsetTop;
- document.onmousemove = function(ev) {
- var oEvent = ev || event;
- This.fnMove(oEvent);
- };
- document.onmouseup = function() {
- This.fnUp();
- };
- }
- Darg.prototype.fnMove = function(ev) { //原型
- this.oDiv.style.left = ev.clientX - this.disX + 'px';
- this.oDiv.style.top = ev.clientY - this.disY + 'px';
- }
- Darg.prototype.fnUp = function() { //原型
- document.onmousemove = null;
- document.onmouseup = null;
- }
- </script>
- </head>
- <body>
- <div id="div1">
- </div>
- </body>
- </html>
效果(拖动红色方块可以移到任何位置):
来源: http://www.cnblogs.com/jnslove/p/7028487.html