成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

前端基本功-示例代碼 (二)

wpw / 693人閱讀

摘要:前端基本功示例代碼一點(diǎn)這里前端基本功示例代碼二點(diǎn)這里一像素偽類實(shí)現(xiàn)對(duì)于老項(xiàng)目,有沒有什么辦法能兼容的尷尬問題了,個(gè)人認(rèn)為偽類是比較完美的方法了。

前端基本功-示例代碼 (一) 點(diǎn)這里
前端基本功-示例代碼 (二) 點(diǎn)這里

1.一像素

偽類 + transform 實(shí)現(xiàn)
對(duì)于老項(xiàng)目,有沒有什么辦法能兼容1px的尷尬問題了,個(gè)人認(rèn)為偽類+transform是比較完美的方法了。

原理是把原先元素的 border 去掉,然后利用 :before 或者 :after 重做 border ,并 transform 的 scale 縮小一半,原先的元素相對(duì)定位,新做的 border 絕對(duì)定位。

單條border樣式設(shè)置:

.scale-1px{
  position: relative;
  border:none;
}
.scale-1px:after{
  content: "";
  position: absolute;
  bottom: 0;
  background: #000;
  width: 100%;
  height: 1px;
  -webkit-transform: scaleY(0.5);
  transform: scaleY(0.5);
  -webkit-transform-origin: 0 0;
  transform-origin: 0 0;
}

四條boder樣式設(shè)置:

.scale-1px{
  position: relative;
  margin-bottom: 20px;
  border:none;
}
.scale-1px:after{
  content: "";
  position: absolute;
  top: 0;
  left: 0;
  border: 1px solid #000;
  -webkit-box-sizing: border-box;
  box-sizing: border-box;
  width: 200%;
  height: 200%;
  -webkit-transform: scale(0.5);
  transform: scale(0.5);
  -webkit-transform-origin: left top;
  transform-origin: left top;
}

最好在使用前也判斷一下,結(jié)合 JS 代碼,判斷是否 Retina 屏:

if(window.devicePixelRatio && devicePixelRatio >= 2){
  document.querySelector("ul").className = "scale-1px";
}

方法二

/*移動(dòng)端正常展示1px的問題 start*/
%border-1px{
    display: block;
    position:absolute;
    left: 0;
    width: 100%;
    content: " ";
}
.border-1px{
    position: relative;
    &::after{
        @extend %border-1px;
        bottom: 0;
        border-top: 1px solid #ccc;
    }
    &::before{
        @extend %border-1px;
        top: 0;
        border-bottom: 1px solid #ccc;
    }
}

@media (-webkit-min-device-pixel-ratio:1.5),(min-device-pixel-ratio:1.5){
    .border-1px{
        &::after{
            -webkit-transform: scaleY(0.7);
            transform: scaleY(0.7);
        }
    }
}

@media (-webkit-min-device-pixel-ratio:2),(min-device-pixel-ratio:2){
    .border-1px{
        &::after{
            -webkit-transform: scaleY(0.5);
            transform: scaleY(0.5);
        }
    }
}
/*移動(dòng)端正常展示1px的問題 end*/

方法三

    .hairline-border {
      box-shadow: 0 0 0 1px;
    }

    @media (min-resolution: 2dppx) {
      .hairline-border {
        box-shadow: 0 0 0 0.5px red;
      }
    }

    @media (min-resolution: 3dppx) {
      .hairline-border {
        box-shadow: 0 0 0 0.33333333px;
      }
    }

    @media (min-resolution: 4dppx) {
      .hairline-border {
        box-shadow: 0 0 0 0.25px;
      }
    }
2.動(dòng)畫
animation:mymove 5s infinite;

@keyframes mymove {
    from {top:0px;}
    to {top:200px;}
}

js實(shí)現(xiàn)一個(gè)持續(xù)的動(dòng)畫效果

//兼容性處理
window.requestAnimFrame = (function(){
  return  window.requestAnimationFrame       ||
          window.webkitRequestAnimationFrame ||
          window.mozRequestAnimationFrame    ||
          function(callback){
            window.setTimeout(callback, 1000 / 60);
          };
})();

var e = document.getElementById("e");
var flag = true;
var left = 0;

function render() {
    left == 0 ? flag = true : left == 100 ? flag = false : "";
    flag ? e.style.left = ` ${left++}px` :
        e.style.left = ` ${left--}px`;
}

(function animloop() {
    render();
    requestAnimFrame(animloop);
})();
3. 實(shí)現(xiàn)sum(2)(3)
// 寫一個(gè) function 讓下面兩行代碼輸出的結(jié)果都為 5
console.log(sum(2, 3));
console.log(sum(2)(3));

實(shí)現(xiàn)

function sum() {
    var cache;
    if (arguments.length === 1) {
        cache = arguments[0];
        return function ( number ) {return cache + number;};
    }
    else return arguments[0] + arguments[1];
};
4. 函數(shù)柯里化
函數(shù)柯里化指的是將能夠接收多個(gè)參數(shù)的函數(shù)轉(zhuǎn)化為接收單一參數(shù)的函數(shù),并且返回接收余下參數(shù)或結(jié)果的新函數(shù)的技術(shù)。

函數(shù)柯里化的主要作用和特點(diǎn)就是參數(shù)復(fù)用、提前返回和延遲計(jì)算/執(zhí)行。

1. 參數(shù)復(fù)用

引導(dǎo)

// 普通函數(shù)
function add(x,y){
    return x + y;
}

add(3,4);   //5

// 實(shí)現(xiàn)了柯里化的函數(shù)
// 接收參數(shù),返回新函數(shù),把參數(shù)傳給新函數(shù)使用,最后求值
let add = function(x){
    return function(y){
        return x + y;
    }
};

add(3)(4);  // 7

通用的柯里化函數(shù)

感覺currying就是返回函數(shù)的函數(shù),在此函數(shù)閉包中定義了私有域變量。

function curry(fn) {
    let slice = Array.prototype.slice,  // 將slice緩存起來
        args = slice.call(arguments, 1);   // 這里將arguments轉(zhuǎn)成數(shù)組并保存
        
    return function() {
        // 將新舊的參數(shù)拼接起來
        let newArgs = args.concat(slice.call(arguments));    
        return fn.apply(null, newArgs); // 返回執(zhí)行的fn并傳遞最新的參數(shù)
    }
}
if (typeof Function.prototype.bind === "undefined"){
  Function.prototype.bind = function (thisArgs){
    var fn = this,
        slice = Array.prototype.slice,
        args = slice.call(arguments, 1);
    return function (){
      let newArgs = args.concat(slice.call(arguments))
      return fn.apply(thisArgs, newArgs);
    }
  }
}

ES6版的柯里化函數(shù)

function curry(fn, ...allArgs) {
    const g = (...allArgs) => allArgs.length >= fn.length ?
        fn(...allArgs) : (...args) => g(...allArgs, ...args)

    return g;
}

// 測試用例
const foo = curry((a, b, c, d) => {
    console.log(a, b, c, d);
});
foo(1)(2)(3)(4);    // 1 2 3 4
const f = foo(1)(2)(3);
f(5);               // 1 2 3 5
function trueCurrying(fn, ...args) {
    if (args.length >= fn.length) {
        return fn(...args)
    }
    return function (...args2) {
        return trueCurrying(fn, ...args, ...args2)
    }
}
// 比較多次接受的參數(shù)總數(shù)與函數(shù)定義時(shí)的入?yún)?shù)量,
//當(dāng)接受參數(shù)的數(shù)量大于或等于被 Currying 函數(shù)的傳入?yún)?shù)數(shù)量時(shí),
//就返回計(jì)算結(jié)果,否則返回一個(gè)繼續(xù)接受參數(shù)的函數(shù)。
//注意這點(diǎn)和上邊的區(qū)別

題目:需要寫一個(gè)函數(shù),滿足

curry(fn)(1)(2)(3) //6
var fn = function(a,b,c) {
    return a+b+c;
}

function curry(fn) {
    var arr = [],
    mySlice = arr.slice
    fnLen = fn.length;

    function curring() {
        arr = arr.concat(mySlice.call(arguments));
        if(arr.length < fnLen) {
            return curring;
        }
        return fn.apply(this, arr);
    }
  return curring;
}
curry(fn)(1)(2)(3);//6

本小題來自:幾個(gè)讓我印象深刻的面試題(一)

2. 提前返回
var addEvent = function(el, type, fn, capture) {
    if (window.addEventListener) {
        el.addEventListener(type, function(e) {
            fn.call(el, e);
        }, capture);
    } else if (window.attachEvent) {
        el.attachEvent("on" + type, function(e) {
            fn.call(el, e);
        });
    } 
};

上面的方法有什么問題呢?很顯然,我們每次使用addEvent為元素添加事件的時(shí)候,(eg. IE6/IE7)都會(huì)走一遍if...else if ...,其實(shí)只要一次判定就可以了,怎么做?–柯里化。改為下面這樣子的代碼:

var addEvent = (function(){
    if (window.addEventListener) {
        return function(el, sType, fn, capture) {
            el.addEventListener(sType, function(e) {
                fn.call(el, e);
            }, (capture));
        };
    } else if (window.attachEvent) {
        return function(el, sType, fn, capture) {
            el.attachEvent("on" + sType, function(e) {
                fn.call(el, e);
            });
        };
    }
})();

初始addEvent的執(zhí)行其實(shí)值實(shí)現(xiàn)了部分的應(yīng)用(只有一次的if...else if...判定),而剩余的參數(shù)應(yīng)用都是其返回函數(shù)實(shí)現(xiàn)的,典型的柯里化。

對(duì)比:惰性加載

let addEvent = function(ele, type, fn) {
    if (window.addEventListener) {
        addEvent = function(ele, type, fn) {
            ele.addEventListener(type, fn, false);
        }
    } else  if (window.attachEvent) {
        addEvent = function(ele, type, fn) {
            ele.attachEvent("on" + type, function() {
                fn.call(ele)
            });
        }
    }

    addEvent(ele, type, fn);
3. 延遲計(jì)算/運(yùn)行

ES5中的bind方法

if (!Function.prototype.bind) {
    Function.prototype.bind = function(context) {
        var self = this,
            args = Array.prototype.slice.call(arguments);
            
        return function() {
            return self.apply(context, args.slice(1));    
        }
    };
}

推薦閱讀:從一道面試題認(rèn)識(shí)函數(shù)柯里化

參考文章:ES6版的柯里化函數(shù)、JS中的柯里化(currying)

5.手寫一個(gè) bind 方法

帶一個(gè)參數(shù):

Function.prototype.bind = function(context) {
    let self = this,
        slice = Array.prototype.slice,
        args = slice.call(arguments);
        
    return function() {
        return self.apply(context, args.slice(1));    
    }
};

帶多個(gè)參數(shù):

//ES3實(shí)現(xiàn)
if(!Function.prototype.bind){
    Function.prototype.bind = function(o, args){
        
        var self = this,
            boundArgs = arguments;//注:arguments是指sum.bind(null,1)中的參數(shù)null和1
        
        return function(){                 //此時(shí)返回的只是一個(gè)函數(shù)
            var args = [], i;
            for(var i=1; i< boundArgs.length; i++){  
                 args.push(boundArgs[i]);
            }
            for(var i =0; i< arguments.length; i++){
                 args.push(arguments[i]);//注:這里的arguments是指result(2)中的參數(shù)2
            }
            return self.apply(o, args);    
        }

    }

}

或者

// 代碼來自書籍 《javaScript 模式》
if (typeof Function.prototype.bind === "undefined"){
  Function.prototype.bind = function (thisArgs){
    var fn = this,
        slice = Array.prototype.slice,
        args = slice.call(arguments, 1);
    return function (){
      return fn.apply(thisArgs, args.concat(slice.call(arguments)));
    }
  }
}
//注:前后arguments不是一回事哦~


//調(diào)用
var sum = function(x,y){ return x+y };
var result = sum.bind(null,1);
result(2);   // 3

或者

Function.prototype.bind = function(){
    var fn = this;
    var args = Array.prototye.slice.call(arguments);
    var context = args.shift();

    return function(){
        return fn.apply(context, args.concat(Array.prototype.slice.call(arguments)));
    };

本節(jié)參考文章:js中的bind

其他文章:JavaScirpt 的 bind 函數(shù)究竟做了哪些事

6.經(jīng)典面試問題:new 的過程

首先來看一下,函數(shù)聲明的過程

// 實(shí)際代碼
function fn1() {}

// JavaScript 自動(dòng)執(zhí)行
fn1.protptype = {
    constructor: fn1,
    __proto__: Object.prototype
}

fn1.__proto__ = Function.prototype
var a = new myFunction("Li","Cherry");
//偽代碼
new myFunction{
    var obj = {};
    obj.__proto__ = myFunction.prototype;
    var result = myFunction.call(obj,"Li","Cherry");
    return typeof result === "object"? result : obj;
}

創(chuàng)建一個(gè)空對(duì)象 obj;

將新創(chuàng)建的空對(duì)象的隱式原型指向其構(gòu)造函數(shù)的顯示原型。

使用 call 改變 this 的指向

如果無返回值或者返回一個(gè)非對(duì)象值,則將 obj 返回作為新對(duì)象;如果返回值是一個(gè)新對(duì)象的話那么直接直接返回該對(duì)象。

所以我們可以看到,在 new 的過程中,我們是使用 call 改變了 this 的指向。

var NEW = function(func) {
    var o = Object.create(func.prototype)
    var k = func.call(o)
    if (typeof k === "object") {
        return k
    } else {
        return o
    }
}
7.javascript里面的繼承怎么實(shí)現(xiàn),如何避免原型鏈上面的對(duì)象共享

什么是原型鏈

當(dāng)一個(gè)引用類型繼承另一個(gè)引用類型的屬性和方法時(shí)候就會(huì)產(chǎn)生一個(gè)原型連。

ES5:寄生組合式繼承:通過借用構(gòu)造函數(shù)來繼承屬性和原型鏈來實(shí)現(xiàn)子繼承父。

    function ParentClass(name) {
      this.name = name;
    }
    ParentClass.prototype.sayHello = function () {
      console.log("I"m parent!" + this.name);
    }
    function SubClass(name, age) {
      //若是要多個(gè)參數(shù)可以用apply 結(jié)合 ...解構(gòu)
      ParentClass.call(this, name);
      this.age = age;
    }
    SubClass.prototype.sayChildHello = function (name) {
      console.log("I"m child " + this.name)
    }
    
    SubClass.prototype = Object.create(ParentClass.prototype);
    SubClass.prototype.constructor = SubClass;

    let testA = new SubClass("CRPER")

    // Object.create()的polyfill
    /*
    function pureObject(obj){
        //定義了一個(gè)臨時(shí)構(gòu)造函數(shù)
         function F() {}
         //將這個(gè)臨時(shí)構(gòu)造函數(shù)的原型指向了傳入進(jìn)來的對(duì)象。
         F.prototype = obj;
         //返回這個(gè)構(gòu)造函數(shù)的一個(gè)實(shí)例。該實(shí)例擁有obj的所有屬性和方法。
         //因?yàn)樵搶?shí)例的原型是obj對(duì)象。
         return new F();
    }
    */
    
    或
    
    function subClass() {
        superClass.apply(this, arguments);
        this.abc = 1;
    }

    function inherits(subClass, superClass) {
        function Inner() {}
        
        Inner.prototype = superClass.prototype;
        subClass.prototype = new Inner();
        subClass.prototype.constructor = subClass;
    }
    
    inherits(subClass, superClass);
    
    subClass.prototype.getTest = function() {
        console.log("hello")
    };

ES6: 其實(shí)就是ES5的語法糖,不過可讀性很強(qiáng)..

    class ParentClass {
      constructor(name) {
        this.name = name;
      }
      sayHello() {
        console.log("I"m parent!" + this.name);
      }
    }

    class SubClass extends ParentClass {
      constructor(name) {
        super(name);
      }
      sayChildHello() {
        console.log("I"m child " + this.name)
      }
      // 重新聲明父類同名方法會(huì)覆寫,ES5的話就是直接操作自己的原型鏈上
      sayHello(){
        console.log("override parent method !,I"m sayHello Method")
      }
    }

    let testA = new SubClass("CRPER")
8.繼承 JS 內(nèi)置對(duì)象(Date)

寫在前面,本節(jié)只記錄了如何繼承Date對(duì)象...的解決方案,具體問題和解析過程請(qǐng)看原文

ES5

// 需要考慮polyfill情況
Object.setPrototypeOf = Object.setPrototypeOf ||
function(obj, proto) {
    obj.__proto__ = proto;

    return obj;
};

/**
 * 用了點(diǎn)技巧的繼承,實(shí)際上返回的是Date對(duì)象
 */
function MyDate() {
    // bind屬于Function.prototype,接收的參數(shù)是:object, param1, params2...
    var dateInst = new(Function.prototype.bind.apply(Date, [Date].concat(Array.prototype.slice.call(arguments))))();

    // 更改原型指向,否則無法調(diào)用MyDate原型上的方法
    // ES6方案中,這里就是[[prototype]]這個(gè)隱式原型對(duì)象,在沒有標(biāo)準(zhǔn)以前就是__proto__
    Object.setPrototypeOf(dateInst, MyDate.prototype);

    dateInst.abc = 1;

    return dateInst;
}

// 原型重新指回Date,否則根本無法算是繼承
Object.setPrototypeOf(MyDate.prototype, Date.prototype);

MyDate.prototype.getTest = function getTest() {
    return this.getTime();
};

let date = new MyDate();

// 正常輸出,譬如1515638988725
console.log(date.getTest());

ES6

class MyDate extends Date {
    constructor() {
        super();
        this.abc = 1;
    }
    getTest() {
        return this.getTime();
    }
}

let date = new MyDate();

// 正常輸出,譬如1515638988725
console.log(date.getTest());

注意:這里的正常輸出環(huán)境是直接用ES6運(yùn)行,不經(jīng)過babel打包,打包后實(shí)質(zhì)上是轉(zhuǎn)化成ES5的,所以效果完全不一樣,會(huì)報(bào)錯(cuò)的

9.簡易雙向數(shù)據(jù)綁定

  
  

10.JavaScript實(shí)現(xiàn)發(fā)布-訂閱模式

發(fā)布-訂閱模式又叫觀察者模式,它定義對(duì)象間的一種一對(duì)多的依賴關(guān)系,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí),所有依賴于它的對(duì)象都將得到通知。JavaScript開發(fā)中我們一般用事件模型來代替?zhèn)鹘y(tǒng)的發(fā)布-訂閱模式

示例1

    function Dep() {//發(fā)布者
        this.subs = [];
    }
    
    Dep.prototype.addSub = function (sub) {
        this.subs.push(sub);
    }
    
    Dep.prototype.notify = function () {
        this.subs.forEach(sub=>sub.update());
    }
    
    function Watcher(fn) {//訂閱者
        this.fn = fn;
    }
    
    Watcher.prototype.update = function () {
         this.fn();
    }
    
    var dep = new Dep();
    dep.addSub(new Watcher(function () {
        console.log("okokok");
    }))
    dep.notify();

推薦閱讀:Javascript設(shè)計(jì)模式之發(fā)布-訂閱模式

示例2

function Event(){
    this.list={},
    this.on=function(key,cb){//訂閱事件
        if(!this.list[key]){
            this.list[key] = []
        }
        this.list[key].push(cb)
    },
    this.emit = function(){//觸發(fā)事件
        var key = Array.prototype.shift.call(arguments)
        var e = this.list[key]
        if(!e){
            return
        }
        var args = Array.prototype.slice.call(arguments)
        for(var i = 0;i

嘗試一下:

var a = new Event()
a.on("a",function(x){console.log(x)})
a.emit("a",1)//1

推薦閱讀:從單向到雙向數(shù)據(jù)綁定

示例3

var myBus = (function() {
    var clienlist = {},
        addlisten, trigger, remove;
    /**
     * 增加訂閱者
     * @key {String} 類型
     * @fn {Function} 回掉函數(shù)
     * */
    addlisten = function(key, fn) {
        if(!clienlist[key]) {
            clienlist[key] = [];
        }
        clienlist[key].push(fn);
    };
    /**
     * 發(fā)布消息
     * */
    trigger = function() {
        var key = [].shift.call(arguments), //取出消息類型
            fns = clienlist[key]; //取出該類型的對(duì)應(yīng)的消息集合
        if(!fns || fns.length === 0) {
            return false;
        }
        for(var i = 0, fn; fn = fns[i++];) {
            fn.apply(this, arguments);
        }
    };
    /**
     * 刪除訂閱
     * @key {String} 類型
     * @fn {Function} 回掉函數(shù)
     * */
    remove = function(key, fn) {
        var fns = clienlist[key]; //取出該類型的對(duì)應(yīng)的消息集合
        if(!fns) { //如果對(duì)應(yīng)的key沒有訂閱直接返回
            return false;
        }
        if(!fn) { //如果沒有傳入具體的回掉,則表示需要取消所有訂閱
            fns && (fns.length = 0);
        } else {
            for(var l = fns.length - 1; l >= 0; l--) { //遍歷回掉函數(shù)列表
                if(fn === fns[l]) {
                    fns.splice(l, 1); //刪除訂閱者的回掉
                }
            }
        }
    };
    return {
        $on: addlisten,
        $emit: trigger,
        $off: remove
    }
})();

推薦閱讀:寫一個(gè)簡單vue 中間件,$emit、$on

示例4
這個(gè)示例更像示例2示例3的總結(jié),我也放這里吧,多看幾種寫法也多少開闊一下思路或全當(dāng)復(fù)習(xí)

賣燒餅的店主可以把小明、小龍的電話記錄下來,等店里有燒餅了在通知小龍小明來拿這就是所謂的發(fā)布-訂閱模式,代碼如下:

/*燒餅店*/        
var Sesamecakeshop={
    clienlist:[],//緩存列表
    addlisten:function(fn){//增加訂閱者
        this.clienlist.push(fn);
    },
    trigger:function(){//發(fā)布消息
        for(var i=0,fn;fn=this.clienlist[i++];){
            fn.apply(this,arguments);
        }
    }
}

/*小明發(fā)布訂閱*/
Sesamecakeshop.addlisten(function(price,taste){
    console.log("小明發(fā)布的"+price+"元,"+taste+"味道的");
});
/*小龍發(fā)布訂閱*/
Sesamecakeshop.addlisten(function(price,taste){
    console.log("小龍發(fā)布的"+price+"元,"+taste+"味道的");
});        

Sesamecakeshop.trigger(10,"椒鹽");

從代碼中可以看出,只有小明,小龍預(yù)定了燒餅,燒餅店就可以發(fā)布消息告訴小龍與小明。但是有個(gè)問題不知道大家發(fā)現(xiàn)了沒有。小明只喜歡椒鹽味道的。而小龍只喜歡焦糖味道的。上面的代碼就滿足不了客戶的需求,給客戶一種感覺就是,不管我喜歡不喜歡,你都會(huì)發(fā)給我。如果發(fā)布比較多,客戶就會(huì)感到厭煩,甚至?xí)雱h除訂閱。下邊是對(duì)代碼進(jìn)行改良大家可以看看。

/*燒餅店*/        
var Sesamecakeshop={
    clienlist:{},/*緩存列表*/
    /**
     * 增加訂閱者
     * @key {String} 類型
     * @fn {Function} 回掉函數(shù)
     * */
    addlisten:function(key,fn){
        if(!this.clienlist[key]){
            this.clienlist[key]=[];
        }
        this.clienlist[key].push(fn);
    },
    /**
     * 發(fā)布消息
     * */
    trigger:function(){
        var key=[].shift.call(arguments),//取出消息類型
            fns=this.clienlist[key];//取出該類型的對(duì)應(yīng)的消息集合
        if(!fns || fns.length===0){
            return false;
        }
        for(var i=0,fn;fn=fns[i++];){
            fn.apply(this,arguments);
        }
    },
    /**
     * 刪除訂閱
     * @key {String} 類型
     * @fn {Function} 回掉函數(shù)
     * */
    remove:function(key,fn){
        var fns=this.clienlist[key];//取出該類型的對(duì)應(yīng)的消息集合
        if(!fns){//如果對(duì)應(yīng)的key沒有訂閱直接返回
            return false;
        }
        if(!fn){//如果沒有傳入具體的回掉,則表示需要取消所有訂閱
            fns && (fns.length=0);
        }else{
            for(var l=fns.length-1;l>=0;l--){//遍歷回掉函數(shù)列表
                if(fn===fns[l]){
                    fns.splice(l,1);//刪除訂閱者的回掉
                }
            }
        }
    }
}

/*小明發(fā)布訂閱*/
Sesamecakeshop.addlisten("焦糖",fn1=function(price,taste){
    console.log("小明發(fā)布的"+price+"元,"+taste+"味道的");
});
/*小龍發(fā)布訂閱*/
Sesamecakeshop.addlisten("椒鹽",function(price,taste){
    console.log("小龍發(fā)布的"+price+"元,"+taste+"味道的");
});        

Sesamecakeshop.trigger("椒鹽",10,"椒鹽");

Sesamecakeshop.remove("焦糖",fn1);//注意這里是按照地址引用的。如果傳入匿名函數(shù)則刪除不了        

Sesamecakeshop.trigger("焦糖",40,"焦糖");

推薦必讀:發(fā)布-訂閱模式

11.扁平化后的數(shù)組

如:[1, [2, [ [3, 4], 5], 6]] => [1, 2, 3, 4, 5, 6]

    var data =  [1, [2, [ [3, 4], 5], 6]];

    function flat(data, result) {
        var i, d, len;
        for (i = 0, len = data.length; i < len; ++i) {
            d = data[i];
            if (typeof d === "number") {
                result.push(d);
            } else {
                flat(d, result);
            }
        }
    }

    var result = [];
    flat(data, result);

    console.log(result);
12.冒泡排序
解析:

比較相鄰的兩個(gè)元素,如果前一個(gè)比后一個(gè)大,則交換位置。

第一輪的時(shí)候最后一個(gè)元素應(yīng)該是最大的一個(gè)。

按照步驟一的方法進(jìn)行相鄰兩個(gè)元素的比較,這個(gè)時(shí)候由于最后一個(gè)元素已經(jīng)是最大的了,所以最后一個(gè)元素不用比較。

js代碼實(shí)現(xiàn)

function bubble_sort(arr){
  for(var i = 0;i < arr.length - 1; i++){
    for(var j = 0;j < arr.length - i - 1;j++){
      if(arr[j] > arr[j+1]){
        [arr[j], arr[j+1]] = [arr[j + 1], arr[j]]
      }
    }
  }
}

var arr = [3,1,5,7,2,4,9,6,10,8];
bubble_sort(arr);
console.log(arr);
13.快速排序

快速排序是對(duì)冒泡排序的一種改進(jìn)

解析:

第一趟排序時(shí)將數(shù)據(jù)分成兩部分,一部分比另一部分的所有數(shù)據(jù)都要小。

然后遞歸調(diào)用,在兩邊都實(shí)行快速排序。

js代碼實(shí)現(xiàn)

function quick_sort(arr){
  if(arr.length <= 1){
    return arr;
  }
  var pivotIndex = Math.floor(arr.length / 2);
  var pivot = arr.splice(pivotIndex, 1)[0];

  var left = [];
  var right = [];
  for (var i = 0;i < arr.length; i++) {
    if(arr[i] < pivot){
      left.push(arr[i]);
    } else {
      right.push(arr[i]);
    }
  }

  return quick_sort(left).concat([pivot],quick_sort(right));
}

var arr=[5,6,2,1,3,8,7,1,2,3,4,7];
console.log(quick_sort(arr));
14.選擇排序
// 選擇排序:大概思路是找到最小的放在第一位,找到第二小的放在第二位,以此類推 算法復(fù)雜度O(n^2)
選擇demo:
function selectionSort(arr) {
    let len = arr.length;
    let minIndex;
    for (let i = 0; i < len - 1; i++) {
        minIndex = i;
        for (let j = i + 1; j < len; j++) {
            if (arr[j] < arr[minIndex]) {     //尋找最小的數(shù)
                minIndex = j;                 //將最小數(shù)的索引保存
            }
        }
        [arr[i], arr[minIndex]] = [arr[minIndex], arr[i]];
    }
return arr;
}

本節(jié)參考文章:2018前端面試總結(jié)...

15.插入排序
解析:

從第一個(gè)元素開始,該元素可以認(rèn)為已經(jīng)被排序

取出下一個(gè)元素,在已經(jīng)排序的元素序列中從后向前掃描

如果該元素(已排序)大于新元素,將該元素移到下一位置

重復(fù)步驟3,直到找到已排序的元素小于或者等于新元素的位置

將新元素插入到下一位置中

重復(fù)步驟2

js代碼實(shí)現(xiàn)

function insert_sort(arr){
  var i=1,
  j,key,len=arr.length;
  for(;i-1){
      if(arr[j]>key){
        arr[j+1]=arr[j];
      }else{
        break;
      }
    }

    arr[j+1]=key;
  }

  return arr;
}

或

function insert_sort(arr) {
    let len = arr.length;
    let preIndex, current;
    for (let i = 1; i < len; i++) {
        preIndex = i - 1;
        current = arr[i];
        while (preIndex >= 0 && arr[preIndex] > current) {
            arr[preIndex + 1] = arr[preIndex];
            preIndex--;
        }
        arr[preIndex + 1] = current;
    }
    return arr;
}

insert_sort([2,34,54,2,5,1,7]);
華米科技 招前端

聯(lián)系: 于夢(mèng)中(wx:tsw0618) 內(nèi)推,備注來意,簡歷請(qǐng)甩 [email protected]

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/101083.html

相關(guān)文章

  • 前端本功-示例代碼 ()

    摘要:前端基本功示例代碼一點(diǎn)這里前端基本功示例代碼二點(diǎn)這里一像素偽類實(shí)現(xiàn)對(duì)于老項(xiàng)目,有沒有什么辦法能兼容的尷尬問題了,個(gè)人認(rèn)為偽類是比較完美的方法了。 前端基本功-示例代碼 (一) 點(diǎn)這里前端基本功-示例代碼 (二) 點(diǎn)這里 1.一像素 偽類 + transform 實(shí)現(xiàn)對(duì)于老項(xiàng)目,有沒有什么辦法能兼容1px的尷尬問題了,個(gè)人認(rèn)為偽類+transform是比較完美的方法了。 原理是把原先元素...

    Java3y 評(píng)論0 收藏0
  • 前端修煉の道 | 你知道哪些復(fù)合選擇器?

    摘要:后代選擇器后代選擇器又稱包含選擇器,用于選擇指定元素的后代元素。這些選擇器既可以是基本選擇器,也可以是一個(gè)復(fù)合選擇器。注意元素選擇器及和屬性選擇器之間沒有空格。 showImg(https://segmentfault.com/img/bVblJEJ?w=900&h=383); 復(fù)合選擇器是通過基本選擇器進(jìn)行組合后構(gòu)成的,常用的復(fù)合選擇器有: 交集選擇器 并集選持器 后代選擇器 子元...

    Java_oldboy 評(píng)論0 收藏0
  • 前端工具【0】—— Emmet插件

    摘要:介紹是許多流行文本編輯器的插件,它極大地改進(jìn)了和工作流程為大部分流行的編輯器都提供了安裝插件,核心是縮寫語法鍵不同編輯器可自行設(shè)置,以下是我整理的常用知識(shí)點(diǎn)。 介紹:Emmet是許多流行文本編輯器的插件,它極大地改進(jìn)了HTML和CSS工作流程 、為大部分流行的編輯器都提供了安裝插件,核心是縮寫語法+tab鍵(不同編輯器可自行設(shè)置),以下是我整理的常用知識(shí)點(diǎn)。 一、特性 1、支持定制: ...

    zhichangterry 評(píng)論0 收藏0
  • 前端工具【0】—— Emmet插件

    摘要:介紹是許多流行文本編輯器的插件,它極大地改進(jìn)了和工作流程為大部分流行的編輯器都提供了安裝插件,核心是縮寫語法鍵不同編輯器可自行設(shè)置,以下是我整理的常用知識(shí)點(diǎn)。 介紹:Emmet是許多流行文本編輯器的插件,它極大地改進(jìn)了HTML和CSS工作流程 、為大部分流行的編輯器都提供了安裝插件,核心是縮寫語法+tab鍵(不同編輯器可自行設(shè)置),以下是我整理的常用知識(shí)點(diǎn)。 一、特性 1、支持定制: ...

    fobnn 評(píng)論0 收藏0
  • 20K前端大佬面試(關(guān)于如何回答ajax跨域問題)

    摘要:在接觸前端開發(fā)起,跨域這個(gè)詞就一直以很高的頻率在我們學(xué)習(xí)工作中重復(fù)出現(xiàn),最近在工作中遇到了跨域的相關(guān)問題,這里我把它總結(jié)記錄一下。 在接觸前端開發(fā)起,跨域這個(gè)詞就一直以很高的頻率在我們學(xué)習(xí)工作中重復(fù)出現(xiàn),最近在工作中遇到了跨域的相關(guān)問題,這里我把它總結(jié)記錄一下。關(guān)于跨域,有N種類型,現(xiàn)在我只專注于ajax請(qǐng)求跨域(ajax跨域只是屬于瀏覽器同源策略中的一部分,其它的這里不做介紹),內(nèi)容...

    Yangyang 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<