函数节流和函数防抖

最近编写vue前端项目时,了解了新的专业名词:函数防抖、函数节流;虽是第一次知道这个词,但平常开发中其实早已接触使用过了;

防抖与节流是处理频繁事件的技术:用以限制事件频繁地发生可能造成的问题,如:

  • 多次执行同一事件,造成事件叠加带来不必要的影响(iOS典型的例子是点击按钮Push到其他vc时会连续跳转多次)
  • 频繁渲染界面等带来的性能问题
  • 频繁调用接口请求数据带来的流量压力

节流(throttle)

节流是指高频事件触发,但在n秒内只会执行一次,所以节流会稀释函数的执行频率;

应用场景:
频繁点击按钮只响应一次事件

防抖(debounce)

防抖是指触发高频事件后n秒内函数只会执行一次,如果n秒内高频事件再次被触发,则重新计算时间;

应用场景:
输入框输入文字时实时搜索功能

节流和防抖的区别

简单来说节流是控制频率,防抖是控制次数
单从字面理解的话,防抖节流容易搞混且不太好理解;下面引用一张图说明:

以上图基础 举个例子:
Button点击事件有可能存在极短时间内,频繁的点击了多次的情况;针对这种情况分别做节流和防抖处理interval设置为1000ms,实际的效果如下:

  • 函数节流
    在1000ms内连续点击了3次,button响应事件a、x、y;最终只会立即执行a,丢弃x、y;如果在m ms内连续点击了n 次,最终会执行时间 m/1000 + 1 个事件;即不管有多少事件interval计算是固定的;
  • 函数防抖
    在1000ms内连续点击了2次b、c;最终会已最后一次点击c为准重新计时,当时间达到interval才执行c;而前面的b会丢弃;如果在m ms内连续点击了n 次,且每每2个点击间隔都没有超过1000ms最终只会在n点击1000ms后执行第n个事件(只有一个事件);即interval计算是动态的,每次都会以最后一个重新计时;

代码实现

前端实现

// 
export default {
  // 防抖
  debounce: function (fn, interval) {
    // 时间间隔ms
    var interval = interval || 200;
    var timer;
    // 闭包
    return function () {
      // 考虑作用域,上下文环境,apply需要用到this对象
      var self = this;
      // 接收的参数用 ES6 中的 rest 参数统一存储到变量 args 中。arguments就是传入的参数数组,而且个数可以不确定的传回给fn(不确定函数到底有多少个参数,用arguments来接收)
      var args = arguments;
      // 判断还在定时,说明当前正在一个计时过程中,并且又触发了相同事件。所以要取消当前的计时,重新开始计时
      if (timer) {
        clearTimeout(timer);
      }
      timer = setTimeout(function () {
        timer = null;
        // 执行方法
        fn.apply(self, args);
      }, interval);
    };
  },

  // 节流 
  throttle: function (func, interval) {
    var timer = null
    var startTime = 0
    return function () {
      // 结束时间
      var curTime = Date.now()
      // 间隔时间 = 延迟的时间 - (结束时间戳 - 开始时间戳)
      var seconds = curTime - startTime
      var self = this
      var args = arguments
      clearTimeout(timer)
      if (seconds > interval) {
        // 证明可以触发了
        func.apply(self, args)
        // 重新计算开始时间
        startTime = Date.now()
      }
    }
  }
}

使用,以实时搜索为例:

// html
     <van-search
        v-model="searchValue"
        placeholder="输入姓名查询"
        @input="onInput"
      />

// js
  methods: {
    onInput: debounceThrottleTool.debounce(function() {
      this.queryUsers()
    }, 800)
  }

iOS实现

知道了思路后,其实不同语言实现起来都不难了;iOS这边可以直接使用dispatch实现(当然也有其他方式)

@implementation MMThrottler

- (instancetype)init {
    self = [super init];
    if (self) {
        _interval = .2;
        _queue = dispatch_get_main_queue();
        _previousDate = NSDate.distantPast;
        _semaphore = dispatch_semaphore_create(1);
    }
    return self;
}

- (void)execute:(void(^)())action {
    __weak typeof(self) selfWeak = self;
    dispatch_semaphore_wait(_semaphore, DISPATCH_TIME_FOREVER);
    
    if (_blockItem) {
        // 取消上一次事件
        dispatch_block_cancel(_blockItem);
    }
    
    _blockItem = dispatch_block_create(DISPATCH_BLOCK_BARRIER, ^{
        selfWeak.previousDate = NSDate.date;
        action();
    });
    
    NSTimeInterval seconds = [NSDate.date timeIntervalSinceDate:_previousDate];
    NSTimeInterval delaySeconds = seconds > _interval ? 0 : _interval;
    if (seconds > _interval) {
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delaySeconds * NSEC_PER_SEC)), _queue, _blockItem);
    }
    
    dispatch_semaphore_signal(_semaphore);
}

@end
@implementation MMDebouncer

- (instancetype)init {
    self = [super init];
    if (self) {
        _interval = .5;
        _queue = dispatch_get_main_queue();
        _semaphore = dispatch_semaphore_create(1);
    }
    return self;
}

- (void)execute:(void(^)())action {
    dispatch_semaphore_wait(_semaphore, DISPATCH_TIME_FOREVER);
    
    if (_blockItem) {
        // 取消上一次事件
        dispatch_block_cancel(_blockItem);
    }
    
    _blockItem = dispatch_block_create(DISPATCH_BLOCK_BARRIER, ^{
        action();
    });
    
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(_interval * NSEC_PER_SEC)), _queue, _blockItem);
    
    dispatch_semaphore_signal(_semaphore);
}

@end

使用,以实时搜索为例

#pragma mark - UITextFieldDelegate -
- (void)searchTextDidChange {
    WeakObj(self)
    [self.debouncer execute:^{
        [selfWeak queryUsers];
    }];
}

响应式框架

如果项目使用了响应式框架,这些框架一般都封装了函数节流、函数防抖功能;
RxSwift可以很简单的使用节流和防抖:

textfield.rx.text.orEmpty.changed
            .debounce(0.3, scheduler: MainScheduler.instance)
            .asObservable()
            .subscribe(onNext: { [weak self] response in

            }).disposed(by: rx.disposeBag)

textfield.rx.text.orEmpty.changed
            .throttle(0.3, scheduler: MainScheduler.instance)
            .asObservable()
            .subscribe(onNext: { [weak self] response in

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

推荐阅读更多精彩内容

  • 我们在一个项目中按钮连续多次点击,会造成一些问题,比如连点按钮会造成多次push页面,或者造成多次网络请求。那么就...
    Da_Yao阅读 116评论 0 0
  • 戏言 什么是节流?顾名思义,为了节省流量呗,怎样节省流量?那就是少发送请求呗。 什么是防抖?就是防...
    杰_5ae8阅读 223评论 0 0
  • 函数节流 还记得上篇文章中说到的图片懒加载吗?我们在文章的最后实现了一个页面滚动时按需加载图片的方式,即在触发滚动...
    柏丘君阅读 2,833评论 1 19
  • 什么是函数节流与函数防抖? 举个例子,我们知道目前的一种说法是当 1 秒内连续播放 24 张以上的图片时,在人眼的...
    李轻舟阅读 1,295评论 0 11
  • 概念解释 在一定时间内,代码执行的次数不一定要非常多,执行的代码越多,带来的效果也是一样,反而会因为执行次数过多而...
    辣瓜瓜阅读 1,537评论 0 2