如何使用 rxjs 在 angular2 中实现输入 keyup 事件的去抖动服务

IT技术 javascript angular typescript rxjs rxjs5
2021-03-01 16:47:39

我正在尝试在输入按键事件时调用服务。

HTML

<input placeholder="enter name" (keyup)='onKeyUp($event)'>

下面是onKeyUp()函数

onKeyUp(event) {
    let observable = Observable.fromEvent(event.target, 'keyup')
        .map(value => event.target.value)
        .debounceTime(1000)
        .distinctUntilChanged()
        .flatMap((search) => {
            // call the service
        });
    observable.subscribe((data) => {
        // data
    });
}

从浏览器的网络选项卡中发现,它在每个按键事件上调用按键功能(正如它应该做的那样),但我试图实现的是每个按键之间的去抖动时间为 1 秒服务电话。此外,如果我移动箭头键移动,则会触发该事件。

链接

4个回答

所以这个链确实是正确的,但问题是你正在创建一个 Observable 并在每个keyup事件上订阅它这就是为什么它多次打印相同的值。只是有多个订阅,这不是您想要做的。

显然有更多的方法可以正确地做到这一点,例如:

@Component({
  selector: 'my-app',
  template: `
    <div>
      <input type="text" (keyup)='keyUp.next($event)'>
    </div>
  `,
})
export class App implements OnDestroy {

  public keyUp = new Subject<KeyboardEvent>();

  private subscription: Subscription;

  constructor() {
    this.subscription = this.keyUp.pipe(
      map(event => event.target.value),
      debounceTime(1000),
      distinctUntilChanged(),
      mergeMap(search => of(search).pipe(
        delay(500),
      )),
    ).subscribe(console.log);
  }

  ngOnDestroy(): void {
    this.subscription.unsubscribe();
  }
}

查看您更新的演示:http : //plnkr.co/edit/mAMlgycTcvrYf7509DOP

2019 年 1 月:针对 RxJS 6 更新

这有效,但有点误导。keyUp 不是 Subject<string> 类型。如果你想要 Subject<string> 你必须在输入中指定(keyup)='keyUp.next($event.target.value)'
2021-04-29 16:47:39
如果用户点击“Enter”,您希望在不等待谴责的情况下执行什么逻辑。有没有办法忽略谴责?
2021-05-12 16:47:39
@BlackEagle 只需为keyup.enter 事件添加与模板中相同的功能,例如 (keyup.enter)="console.log($event.target.value)"
2021-05-18 16:47:39

@marlin 提供了一个很好的解决方案,它在 angular 2.x 中运行良好,但在 angular 6 中,他们已经开始使用 rxjs 6.0 版本,并且语法略有不同,所以这里是更新的解决方案。

import {Component} from '@angular/core';
import {Observable, of, Subject} from 'rxjs';
import {debounceTime, delay, distinctUntilChanged, flatMap, map, tap} from 'rxjs/operators';

@Component({
    selector: 'my-app',
    template: `
        <div>
            <input type="text" (keyup)='keyUp.next($event)'>
        </div>
     `,
})
export class AppComponent {
    name: string;

    public keyUp = new Subject<string>();

    constructor() {
        const subscription = this.keyUp.pipe(
            map(event => event.target.value),
            debounceTime(1000),
            distinctUntilChanged(),
            flatMap(search => of(search).pipe(delay(500)))
        ).subscribe(console.log);
    }
}
@FrancescoBorzì 您可以使用 map((event: any) => (<HTMLInputElement>event.target).value),或设置事件的类型
2021-05-06 16:47:39
为什么会有额外的延迟(500)?事件不是已经被 1000 的反弹时间延迟了吗?
2021-05-18 16:47:39
错误 TS2339:“字符串”类型上不存在属性“目标”。
2021-05-21 16:47:39

好吧,这是一个基本的去抖动器。

ngOnInit ( ) {
        let inputBox = this.myInput.nativeElement;
        let displayDiv = this.myDisplay.nativeElement;

        let source = Rx.Observable.fromEvent ( inputBox, 'keyup' )
            .map ( ( x ) => { return x.currentTarget.value; } )
            .debounce ( ( x ) => { return Rx.Observable.timer ( 1000 ); } );

        source.subscribe (
            ( x ) => { displayDiv.innerText = x; },
            ( err ) => { console.error ( 'Error: %s', err ) },
            () => {} );
    }
}

如果您设置两个指示的视图子项(输入和显示),您将看到去抖动工作。不确定这是否对您没有任何作用,但这种基本形式是(据我所知)去抖动的直接方法,我经常使用这个起点,内部文本集只是一个示例,它可以拨打服务电话或其他任何您需要它做的事情。

我建议您查看 Angular2 教程,该教程显示了一个与您所要求的内容类似的干净且解释清楚的示例。

https://angular.io/docs/ts/latest/tutorial/toh-pt6.html#!#observables