14-框架封装

一、 框架结构 ==> 难点

  • 1 jQuery对象的本质
  • 2 框架核心结构
  • 3 入口函数(构造函数)

1、jQuery的本质

jquery对外暴露了两个方法:jQuery和$,
这两方法实际上是同一个方法,
通过调用这两个方法,可以得到一个jQuery实例对象。
  • jQuery 实例对象是一个伪数组对象
  • jQuery和$实际上是一个工厂函数
  • 工厂函数直接调用,就可以得到一个实例

2、jQuery的基本结构

2.2.0版本

(function(w,factory){
    console.log('对模块做了支持');
    factory();
 }(window,function(){
    return jQuery;
 }));

1.7版本

(function(w){
   // 对外暴露的工厂函数
   function jQuery(){
       return new jQuery.fn.init();
   }
   //给原型提供一个简写的方法
   jQuery.fn = jQuery.prototype ={
   };
   // init是jQuery中真正的构造函数
   var init = jQuery.fn.init = function(){
   };
   // 替换构造函数的原型为jQuery工厂的原型
   init.prototype = jQuery.fn;
   // 把工厂通过两个变量暴露出来
   w.jQuery = w.$ = jQuery;
}(window));

3、jQuery插件实现机制

jQuery.fn.alert = function(msg){
   alert(msg);
};

在fn上添加

4、入口函数init

通过$工厂,最终到达了init构造函数这里,所有的初始化实例过程都在这里实现,所以把这里称之为入口函数

5、入口函数实现思路

==jq入口对不同参数处理的规律:==

  • 传入null、undefined、0、NaN、''返回空对象( 即空实例 )
  • 传入字符串,那么需要判断是html片段 还是 其它,
  • 如果是片段,则创建对应的DOM,然后添加到实例身上;
  • 否则按照选择器获取页面中的DOM,然后把获取到的DOM添加到实例身上。
  • 如果是数组或许伪数组,那么把每一项分别添加到实例身上。
  • 除了上面的数据类型,剩余的,统一添加到实例身上。

==入口函数实现逻辑:==

funciton init(selector){
    //传入的null、undefined、0、NaN、'' 返回空对象(即空实例)
    if(!selector){
        return this;
    }
    //传入的字符串,那么需要判断是html片段还是其他
    else if(typeof selector == 'string'){
        //如果是片段,则创建对应的DOM,然后添加到实例身上。如果字符串的第一个字母是<,最后一个字母是>,并且length>=3,就可以认为是html片段
        if(是html片段){
            /*
            1、先创建一个临时的div容器
            2、设置这个div的innerHTML为html片段
            3、然后遍历div的子元素分别添加到this身上,记住给实例补充length属性值
            4、可以使用数组的push来给实例添加,同时可以使用apply简化遍历过程
            */
        }
        // 否则按照选择器获取页面中的DOM,然后把获取到的DOM添加到实例身上
        else {

                /*
                * 实现的思路:
                * 1、使用querySelectorAll获取页面中的元素
                * 2、然后遍历获取到所有元素分别添加this身上,记住给实例补充length属性值,
                * 可以使用数组的push来给实例添加,同时可以使用apply简化遍历过程。
                * */
            }
    }
    /*
    * 判断是不是真假数组的思路:
    * 1、先把函数和window排除掉,
    * 2、然后通过toString来判断是不是真数组
    * 3、否则再判断是不是伪数组
    * 备注:下面的判断只用来判断是不是伪数组,不要用下面的判断条件判断真数组,
    *       例如:[ 0:1, , , ] 使用下面的判断,就会得到false,造成真数组的误判。
    * 3.1、 先看看这个对象有没有length属性,
    * 3.2、 如果有,看看length的值是不是为0,如果为0,OK是伪数组,
    * 3.3、 如果length的值不为0,看看这个数据有没有 length - 1这个属性,如果有,OK是伪数组。
    * 建议把这个是否是真假数组的判断封装为一个函数,在这里调用。
    * */
    else if (  ) {

                /*
                * 实现的思路:
                * 把真或伪数组中的每一项分别添加到实例身上,记住给实例补充length属性值,
                * 可以使用数组的push来给实例添加,同时可以使用apply简化遍历过程。
                * */
            }else {
                /*
                * 实现的思路:
                * 把这个参数直接添加到实例身子,length为1即可。
                * */
            }
}

6、代码块

代码块:一对大括号
如果把一对大括号赋值给其他变量,或者参与运算,那么大括号就变
成了字面量
例如:
    ({}).toString()  正确
    {}.toString();   错误

7、window的length属性代表页面中iframe的数量

window有一个window属性,指向自己

8、封装数组的trim方法

function trim(str){
    if(typeof str!== 'string'){
        return str;
    }
    if(str.trim){
        return str.trim();
    }
    return str.replace(/^\s+|\s+$/g,'');
}

第二天 核心方法 ==> 中等

  • 1 入口函数对函数的处理
  • 2 原型上的核心方法
  • 3 map与each

1、静态和实例方法的区别

  • 静态方法
jQuery.extend({
       isFunction: function( fn ) {
           return typeof fn === 'function';
       }
    });
  • 实例方法
jQuery.prototype.extend({
            alert: function( msg ) {
                alert( msg );
            }
        });
  • 静态方法的使用
jQuery.isFunction([])
  • 实例方法的使用
var $$ = new jQuery();
$$.alert( '实例调用' );

==构造函数不能使用自己原型中方法( Function例外 )==

==实例不能直接使用构造函数身上的静态方法==

2、IE8中apply有问题

apply方法可以该改变this指向,同时可以把数组或伪数组平铺传入给函数,
但是IE8,apply只能平铺真数组或者内置的伪数组,我们自定义的伪数组会报错。

IE8中需要先把自定义伪数组转换为真数组,才能借用apply。
借用数组的slice方法,通过一个伪数组得到一个真数组

//先把伪数组截取为真数组
[].splce.call(伪数组) 
//实现需求
[].push.apply(空对象,[].splice.call(伪数组));

3、入口函数

为了防止获取不到页面的元素,所以要把函数传入到jQuery,
然后在这个回调函数中编写代码。这个回调函数会在页面DOM解析完毕之后执行。
$(function(){
    //逻辑代码
})
  • html5新增了一个DOMContentLoaded事件,兼容IE9,
  • 这个事件会在DOM解析完毕后触发,
  • 通常这个事件要比onload快很多,
  • 但是也有很少的例外。
  • ==备注==:如果发生了例外,DOMContentLoaded事件和onload事件触发的间隔时间相差不会很大,
  • 所以可以认为DOMContentLoaded 比 onload要快,只监听DOMContentLoaded即可。

==IE9+==

document.addEventListener( 'DOMContentLoaded', function() {
    var spans = document.querySelectorAll( 'span' );
    console.log( spans, 'DOMContentLoaded');
} );

==IE8兼容==

document.attachEvent( 'onreadystatechange', function() {
    if ( document.readyState === 'complete' ) {
        /*var spans = document.querySelectorAll( 'span' );
        console.log( spans, 'DOMContentLoaded');*/
        /*逻辑代码*/
    }

4、jQ原型核心方法

  • 1、jquery 获取版本号
  • 2、selector 代表所有实例默认的选择器,也代表实例是一个jQuery类型的对象
  • 3、length 代表所有实例默认的长度
  • 4、toArray 把实例转换为数组返回
  • 5、get 获取指定下标的元素,获取的是原生DOM
  • 6、each 遍历实例,把遍历到的数据分别传给回调使用
  • 7、map 遍历实例,把遍历到的数据分别传给回调使用,然后把回调的返回值收集起来组成一个数组返回
  • 8、slice 截取实例的部分元素,构成一个新的jQuery实例返回。
  • 9、first 获取实例中的第一个元素,是jQuery类型的实例对象。
  • 10、last 获取实例中的最后一个元素,是jQuery类型的实例对象。
  • 11、eq 获取指定下标的元素,获取的是jQuery类型的实例对象。
  • 12、push 给实例添加新元素
  • 13、sort 对实例中的元素进行排序
  • 14、splice 按照指定下标指定数量删除元素,也可以替换删除的元素。

5.each方法

function each(obj,fn){
    var i,len,key;
    if('length' in obj){
        for(i = 0,len=obj.length;i<len;i++){
        //修改this指向为value,并添加中断
            if(fn.call(obj[i],i,obj[i]==false)){
                break;
            }
        }
    }else{
        for(key in obj){
            if(fn.call(obj[key],key,obj[key])==false){
                break;
            }
        }
    }
}

each( obj, function( key, val ) {
     // 当得到val为222这个的属性时,就不用再遍历之后的属性了
     if ( val == 333 ) {
        return false;
     }
     console.log( key, val );
} );

6、map实现

function map( obj, fn ) {
    /*
     * 1、先判断obj是不是数组或者伪数组,
     * 2、如果是,则通过i的方式遍历这个对象
     * 3、如果不是,则通过for in的方式遍历这个对象
     * 4、在遍历的过程中,把每一次遍历到key和val分别传给回调。
     * 5、在给回调传参的时候,需要收集回调的返回值,最后把所有的返回值构成新数组返回。
     * */
    var i, len, key, result = [];

    if( 'length' in obj ) {
        for ( i = 0, len = obj.length; i < len; i++ ) {
            result.push( fn.call( obj[ i ], obj[ i ], i ) );
        }
    }else {
        for ( key in obj ) {
            result.push( fn.call( obj[ key ], obj[ key ], key ) );
        }
    }

    return result;
}
console.log(map(obj, function (val, key) {
    console.log(val, key, this);
}));

第三天 DOM操作 ==> 中等

  • 1 创建DOM
  • 2 删除DOM
  • 3 追加DOM

第四天 属性样式操作 ==> 简单

  • 1 class属性操作
  • 2 公共属性操作
  • 3 样式操作

第五天 事件 ==> 难点

  • 1 事件绑定
  • 2 事件解除

第六天 ajax和插件 ==> 难点

  • 1 ajax
  • 2 插件实现原理
  • 3 折线图饼图做成框架插件
  • 4 刮刮乐插件
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容

  •   JavaScript 与 HTML 之间的交互是通过事件实现的。   事件,就是文档或浏览器窗口中发生的一些特...
    霜天晓阅读 3,490评论 1 11
  • 第3章 基本概念 3.1 语法 3.2 关键字和保留字 3.3 变量 3.4 数据类型 5种简单数据类型:Unde...
    RickCole阅读 5,122评论 0 21
  • 1.几种基本数据类型?复杂数据类型?值类型和引用数据类型?堆栈数据结构? 基本数据类型:Undefined、Nul...
    极乐君阅读 5,514评论 0 106
  • 咚咚咚,咚咚咚……正吃晚饭的李强听到有敲门声,声音由小变大,由弱变强,越来越急,这个时间点来找人,说白了,就是来挤...
    钱乐活阅读 520评论 18 11
  • 一、9patch图片的概念 9patch图片是andriod app开发里一种特殊的图片形式,文件的扩展名为:.9...
    笑说余生阅读 138,009评论 21 120