最新公告
  • 欢迎您光临起源地模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • [译]在 Angular 中使用拦截器的方式 Top 10

    正文概述 掘金(程序猿小新)   2020-12-29   564

    有许多种方式使用拦截器,我确定我们大多数人使用的很浅显。在这篇文章中,我将介绍在 Angular 中我最喜欢的 10 种使用拦截器的方式。

    我使例子尽可能的简洁。我希望他们能够启发你们去思考使用拦截器的新方式。这篇文章不是关于拦截器教学的,因为已经有很多好的文章了。但是,在开始倒数之前,让我们以一些基础的知识点开始。

    HttpInterceptor 101

    HttpInterceptor 是在 Angular 4.3 引入。它提供一种方式拦截 HTTP 请求和响应,在传递他们之前转换或者处理他们。

    这是因为我们可能想要在某个请求第一次没成功后重试。不变性确保了拦截器链能够多次重新处理相同的请求。

    你可以使用多个拦截器,但是这个记心中:

    在示例 APP 中,我们提供了全部拦截器,但是一次仅使用一个。这通过检查路径实现(代码在这里)。如果不是我们找的请求,我们通过 next.handle(req) 传递给下一个拦截器。

    [译]在 Angular 中使用拦截器的方式 Top 10

    拦截器的另一个好处是他们能够一起处理请求和响应。我们将看到,这给我们很好的可能性。

    更多深度知识,可以看 Max Koretskyi aka Wizard 这篇很棒的文章:

    • Angular 拦截器内部指南和 HttpClient 机制

    在示例的 HTTP 请求中,我使用了 JSONPlaceholder 这个网站。如果你想看代码,你可以从这里找到:

    • 10.x 版本,译者版本

    GitHub 示例代码 10.x版本 ? StackBlitz 线上运行 ?

    • 8.x 版本,作者版本

    GitHub 示例代码 8.x版本 ? StackBlitz 线上运行 ?

    现在,让我们开始倒数吧!

    [译]在 Angular 中使用拦截器的方式 Top 10

    10.URL

    操纵 URL。当我大声说出来时,听起来有些风险,但是让我们看下在拦截器下做这个事情是多么简单。

    例如,我们想从 HTTP 变为 HTTPS。

    就像克隆请求同时使用 https:// 替换 http:// 一样简单。然后我们将克隆的 HTTPS 请求发送到下一个 handler。

    // 克隆请求,同时使用 https:// 替换 http://
    const httpsReq = req.clone({
        url: req.url.replace('http://', 'https://')
    });
    return next.handler(httpsReq)
    

    这个例子中,我们设置 URL 为 HTTP,但是当我们检查请求时,我们能看见它已经变成了 HTTPS。

    const url = `http://jsonplaceholder.typicode.com/todos/1`;
    this.response = this.http.get(url);
    

    [译]在 Angular 中使用拦截器的方式 Top 10

    自动化魔术 https,为什么这个不高明呢。通常你可以通过 web 服务器设置这些事情。或者你想在开发环境从 HTTP 切换到 HTTPS,你可以使用这个 CLI:

    ng serve -ssl
    

    类似,你可以修改一点 URL,并且成它为 API 前缀拦截器:

    req.clone({
        url: environment.serverUrl + request.url
    });
    

    或者你可以再次通过 CLI 来实现:

    ng serve - serve-path=<path> - base-href <path>/
    

    感谢 David Herges 的 CLI 提示。

    9.Loader

    当我们等待响应时,每个人都希望看见命运的纺纱轮(表示旋转的loading)。只要在请求活跃的时候,我们在拦截器中统一设置,这样我们就能够看见 loader 了。

    首先,我们能够使用 loader 服务,这样就有了展示和隐藏 loader 的功能。在处理请求前,我们调用展示方法并通过 finalize 完成后隐藏 loader。

    const loaderService = this.injector.get(LoaderService);
    
    loaderService.show();
    
    return next.handle(req).pipe(
        delay(5000),
        finalize(() => loaderService.hide())
    )
    

    这个例子很简单,在真实的解决方案中,我们应该考虑到会有多个 HTTP 请求被拦截。这可以通过一个请求(+1)响应(-1)的计数器来解决这个问题。

    当然,我添加了一个延迟来让我们有时间能够看到 loader。

    [译]在 Angular 中使用拦截器的方式 Top 10

    全局的 loader 听起来是个不错的主意,但是这个为什么不在列表中呢?可能它适合特定的应用,但是如果你同时加载多个,你可能想要对 loader 定制化。

    我将给你留下一些思考的空间。如果你用 switchMap 去取消请求将会发生什么?

    8.转换

    当 API 返回一个我们不赞同的格式,我们能够使用拦截器去格式化成我们想要的样子。

    这能够从 XML 转换到 JSON,或者像例子中的属性名字从大驼峰拼写到小驼峰拼写。如果后端不关心 JSON/JS 转换,我们能够使用拦截器将全部属性名重命名为小驼峰。

    检查是否有 npm 包能够为你完成繁重的工作。在这个例子中我使用 loadsh 的 mapKeys 和 camelCase。

    return next.handle(req).pipe(
        map((event: HttpEvent<any>) => {
            if (event instanceof HttpResponse) {
                let camelCaseObject = mapKeys(event.body, (v,k) => camelCase(k));
                const modEvent = event.clone({ body: camelCaseObjec });
    
                return modEvent;
            }
        })
    )
    

    这个事情通常是后端来做,所以我通常不这么做。但是将这个加入到的兵器库,这样你需要的时候就能够使用了。

    [译]在 Angular 中使用拦截器的方式 Top 10

    7.Headers

    通过操纵 headers 我们能够做许多事,例如:

    • 认证(authentication)/ 授权(authorization)
    • 缓存行为;例如,If-Modified-Since
    • XSRF 保护

    我们能够通过拦截器轻而易举的添加 headers。

    cosnt modified = req.clone({
        setHeaders: { "X-Man": "wolverine" }
    });
    
    return next.handle(modified);
    

    然后我们在开发者工具中就能够看到它被添加到了请求头中。

    [译]在 Angular 中使用拦截器的方式 Top 10

    Angular 使用拦截器来防范 跨站请求伪造(XSRF)。通过读取 cookie 中的 XSRF-TOKEN 并设置一个 X-XSRF-TOKEN 请求头来实现。仅仅运行在你的域名中的代码才能够读取 cookie,这样后端能够确定 HTTP 请求来自己客户端程序而不是攻击者。

    如你所见,在拦截器中能够直接操纵 headers。接下来我们将看到更多操纵 headers 的例子。

    6.通知

    这里有很多不同的例子用于展示消息。在我的例子中,每次从服务器获得 201 创建状态时,我会展示“Object created.”。

    return next.hadle(req).pipe(
        tap((event: HttpEvent<any>) => {
            if (event instanceof HttpResponse && event.status === 201) {
                this.toastr.success('Object created.')
            }
        })
    );
    

    或者我们检查对象的类型来展示“Type created”。或者通过将数据和消息包裹到对象中创建一个定制的消息。

    {
        data: T,
        message: string;
    }
    

    [译]在 Angular 中使用拦截器的方式 Top 10

    当发生错误的时候,我们也可以通过拦截器展示通知。

    5.Errors

    在这个拦截器中我们实现了两个关于错误的用例。

    首先,我们能够重试 HTTP 请求。例如,网络中断在移动端场景很常见,再试一次可能会成功。值得考虑的事情是在放弃之前的重试次数。我们应该在重试前等待吗,或者立即重试?

    对于这点,我们使用 RxJS 中的 retry 操作符重新订阅 observable。HttpClient 方法调用的重新订阅具有再次发出 HTTP 请求的效果。

    这种行为的更高级的示例:

    • 当发生错误时,基于自定义标准重试 observable 序列
    • 当使用幂等回退时 rxjs 的能力

    第二点,检查异常的状态。基于状态,决定应该做什么。

    return next.handle(req).pipe(
        retry(2),
        catchError((error: HttpErrorResponse) => {
            if (error.status !== 401) {
                // 401 在 auth.interceptor 中处理了
                this.toastr.error(error.message);
            }
            return throwError(error)
        })
    )
    

    这个例子中,在检查错误状态前,我们重试了两次。如果状态不是 401,我们已弹出(toastr)的形式展示错误。所有的错误将重新抛出来进一步处理。

    [译]在 Angular 中使用拦截器的方式 Top 10

    [译]在 Angular 中使用拦截器的方式 Top 10

    更多的错误处理的知识,你可以在这里阅读我早期的文章:

    • 期待意外 -- Angular 错误处理的最佳实践

    4.分析

    因为拦截器能够同时处理请求和响应,能够在一次完整的 HTTP 操作中计时和记日志。所以我们能够捕获请求和响应时间,记录经过的时间结果。

    const started = Date.now()
    let ok: string;
    
    return next.handle(req).pipe(
        tap(
            (event: HttpEvent<any>) => ok = event instanceof HttpResponse ? 'successed' : '',
            (error: HttpEventResponse) => ok = 'failed'
        ),
    
        // 响应 observable 结束或者完成的时候记日志
        finalize(() => {
            const elapsed = Date.now() - started;
            const msg = `${req.method} "${req.urlWithParams}" ${ok} in ${elapsed} ms.`
            console.log(msg);
        })
        )
    

    这有许多中可能,例如,我们能记录分析日志到数据库中做统计。这个例子中,我们输出到 console。

    [译]在 Angular 中使用拦截器的方式 Top 10

    3. 伪造后端

    当没有后端服务时,可以在开发中模拟或者伪装后端。你也可以将其用于 StackBlitz 中托管的代码。

    我们基于请求模拟返回,然后返回一个 HttpResponse observable。

    const body = {
        firstName: 'mock',
        lastName: 'Faker',
    };
    
    return of(new HttpResponse(
        { status: 200, body: body }
    ));
    

    [译]在 Angular 中使用拦截器的方式 Top 10

    2. 缓存

    因为拦截器能够自己处理请求,没有转发到 next.handle(),所以我们利用这一点来缓存请求。

    在 key-value map 构成的缓存中,我们使用 URL 作为 key。如果我们响应在 map 中,我们能够通过 next handler 返回这个 observable 响应。

    当你已经有响应缓存时,你不需要一路走到后端,这个提升了性能。

    import { Injectable } from '@angular/core';
    import { HttpEvent, HttpRequest, HttpHandler, HttpInterceptor, HttpResponse } from '@angular/common/http';
    import { Observable, of } from 'rxjs';
    import { tap, shareReplay } from 'rxjs/operators';
    
    @Injectable()
    export class CacheInterceptor implements HttpInterceptor {
        private cache = new Map<string, any>();
    
        intercept(request: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {
            if (request.method !== 'Get') {
                return next.handle(request);
            }
    
            const cacheResponse = this.cache.get(request.url);
            if (cacheResponse) {
                return of(cacheResponse);
            }
    
            return next.handle(request).pipe(
                tap(event => {
                    if (event instanceof HttpResponse) {
                        this.cache.set(request.url, event);
                    }
                })
            )
        }
    }
    
    

    如果我们运行这个请求,清空响应然后再次运行将使用缓存。

    [译]在 Angular 中使用拦截器的方式 Top 10

    [译]在 Angular 中使用拦截器的方式 Top 10

    如果数据更新了,你需要使缓存失效,这会引入一些复杂性。但是现在不用担心!缓存生效的时候是真的爽!

    关于缓存的更多知识可以读 Dominic E. 的这篇很帮的文章:

    • RxJS 高级缓存

    1.认证

    清单中的第一个是认证!对于很多应用来说他是基本的,我们已经有了适当的认证系统。这是拦截器最常见的用例之一,有充分的理由。恰到好处!

    有几个和权限相关的事情我们能做:

    1. 添加 bearer token
    2. 重新刷新 token
    3. 重定向到登录页

    当我们发送 bearer token 时,我们也应该有些过滤。如果我们还没 token,我们可能在登录,并不需要添加 token。或者如果我们调用其他域名,我们也不希望添加 token。例如,如果向 Slack 发送错误信息。

    相比于其他拦截器这个会有点复杂。这是一个带有一些解释性注释的例子:

    import { Injectable } from '@angular/core';
    import { HttpEvent, HttpInterceptor, HttpHandler, HttpRequest, HttpErrorResponse } from '@angular/common/http';
    import { throwError, Observable, BehaviorSubject, of } from 'rxjs';
    import { catchError, filter, take, switchMap } from 'rxjs/operators';
    
    @Injectable()
    export class AuthInterceptor implements HttpInterceptor {
        private AUTH_HEADER = 'Authorization';
        private token = 'secrettoken';
        private refreshTokenInProgress = false;
        private refreshTokenSubject: BehaviorSubject<any> = new BehaviorSubject<any>(null);
    
        intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {
            if (!req.headers.has('Content-type')) {
                req = req.clone({
                    headers: req.headers.set('Content-Type', 'application/json')
                });
            }
    
            req = this.addAuthenticationToken(req);
    
            return next.handle(req).pipe(
                cacheError((error: HttpErrorResponse) => {
                    if (error && error.status === 401) {
                        if (this.refreshTokenInProgress) {
                            return this.refreshTokenSubject.pipe(
                                filter(result => result !== null),
                                take(1),
                                switchMap(() => next.handle(this.addAuthenticationToken(req)))
                            );
                        } else {
                            this.refreshTokenInProgress = true;
    
                            // 设置 refreshTokenSubject 为 null,这样随后的 API 将等到新的 token 被取回时才调用。
                            this.refreshTokenSubject.next(null);
    
                            return this.refreshAccessToken().pipe(
                                switchMap((success: boolean) => {
                                    this.refreshTokenSubject.next(success);
                                    return next.handle(this.addAuthenticationToken(req));
                                }),
    
                                // 当我们调用刷新 token 方法完成时,重置 refreshTokenInProgress 为 false,
                                // 这是为了下次 token 需要再次被刷新
                                finalize(() => this.refreshTokenInProgress = false)
                            );
                        }
                    } else {
                        return throwError(error)
                    }
                })
            )
        }
    
        private refreshAccessToken(): Observable<any> {
            return of('secret token');
        }
    
        private addAuthenticationToken(request: HttpRequest<any>): HttpRequest<any> {
            // 如果还没有 token,不应该在 header 中设置 token。
            // 首先我们应该从存储 token 的地方取回
            if (!this.token) {
                return request;
            }
    
            // 如果访问外部域名不应该添加 token
            if (!request.url.match(/www.mydomain.com\//)) {
                return request;
            }
    
            return request.clone({
                headers: request.headers.set(this.AUTH_HEADER, 'Bearer ' + this.token)
            })
        }
    }
    

    大吉大利,今晚吃鸡!?

    总结

    拦截器是 Angular 4.3 中一个重要的功能,这里我们看到了很多很棒的功能。现在发挥你的创造力,我相信你可以想出一些有趣的东西!

    记住,通过使用拦截器,你可以像蝙蝠侠一样棒!

    [译]在 Angular 中使用拦截器的方式 Top 10

    • 10.x 版本,译者版本

    GitHub 示例代码 10.x版本 ? StackBlitz 线上运行 ?

    • 8.x 版本,作者版本

    GitHub 示例代码 8.x版本 ? StackBlitz 线上运行 ?

    感谢 Angular In Depth 提供想法和帮助编辑文档。希望我没有忘记任何一个人,谢谢 Max Koretskyi aka Wizard,Tim Deschryver,Alex Okrushko,Alexander Poshtaruk,Lars Gyrup Brink Nielsen,Nacho Vazquez Calleja,thekiba & Alexey Zuev!

    资源

    • Angular 拦截器内部指南和 HttpClient 机制 by Max Koretskyi,aka Wizard

    • Angular 文档

    • 许多其他好文章中的点点滴滴

    译者参考

    • 驼峰拼写法
    • XSRF/CSRF

    感谢阅读

    感谢你阅读到这里,翻译的不好的地方,还请指点。希望我的内容能让你受用,再次感谢。by llccing 千里


    起源地下载网 » [译]在 Angular 中使用拦截器的方式 Top 10

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元