14-框架封装

2018-05-30  本文已影响10人  海棠未雨

一、 框架结构 ==> 难点

1、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入口对不同参数处理的规律:==

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

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、静态和实例方法的区别

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(){
    //逻辑代码
})

==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原型核心方法

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操作 ==> 中等

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

第五天 事件 ==> 难点

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

上一篇 下一篇

猜你喜欢

热点阅读