国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?

這篇具有很好參考價值的文章主要介紹了75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?。希望對大家有所幫助。如果存在錯誤或未考慮完全的地方,請大家不吝賜教,您也可以點擊"舉報違法"按鈕提交疑問。

75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?

區(qū)別

  • 攔截器不依賴與servlet容器,過濾器依賴與servlet容器。
  • 攔截器只能對action請求(DispatcherServlet 映射的請求)起作用,而過濾器則可以對幾乎所有的請求起作用。
  • 攔截器可以訪問容器中的Bean(DI),而過濾器不能訪問(基于spring注冊的過濾器也可以訪問容器中的bean)。

執(zhí)行順序

過濾器 和 攔截器的觸發(fā)時機也不同,我們看下邊這張圖。
75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?,Spring最新全家桶面試1000道必考題,spring,java,后端,springmvc,ssm,spring boot,spring cloud
過濾器Filter是在請求進入容器后,但在進入servlet之前進行預(yù)處理,請求結(jié)束是在servlet處理完以后。

攔截器 Interceptor 是在請求進入servlet后,在進入Controller之前進行預(yù)處理的,Controller 中渲染了對應(yīng)的視圖之后請求結(jié)束。

知識延伸

過濾器 (Filter)

過濾器的配置比較簡單,直接實現(xiàn)Filter 接口即可,也可以通過@WebFilter注解實現(xiàn)對特定URL攔截,看到Filter 接口中定義了三個方法。

  • init() :該方法在容器啟動初始化過濾器時被調(diào)用,它在 Filter 的整個生命周期只會被調(diào)用一次?!缸⒁狻梗哼@個方法必須執(zhí)行成功,否則過濾器會不起作用。
  • doFilter() :容器中的每一次請求都會調(diào)用該方法, FilterChain 用來調(diào)用下一個過濾器 Filter。
  • destroy(): 當(dāng)容器銷毀 過濾器實例時調(diào)用該方法,一般在方法中銷毀或關(guān)閉資源,在過濾器 Filter 的整個生命周期也只會被調(diào)用一次。

使用過濾器的代碼

@Component
public class MyFilter implements Filter {
    
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

        System.out.println("Filter 前置");
    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {

        System.out.println("Filter 處理中");
        filterChain.doFilter(servletRequest, servletResponse);
    }

    @Override
    public void destroy() {

        System.out.println("Filter 后置");
    }
}



攔截器 (Interceptor)

攔截器它是鏈?zhǔn)秸{(diào)用,一個應(yīng)用中可以同時存在多個攔截器Interceptor, 一個請求也可以觸發(fā)多個攔截器 ,而每個攔截器的調(diào)用會依據(jù)它的聲明順序依次執(zhí)行。

首先編寫一個簡單的攔截器處理類,請求的攔截是通過HandlerInterceptor 來實現(xiàn),看到HandlerInterceptor 接口中也定義了三個方法。

  • preHandle() :這個方法將在請求處理之前進行調(diào)用?!缸⒁狻梗喝绻摲椒ǖ姆祷刂禐閒alse ,將視為當(dāng)前請求結(jié)束,不僅自身的攔截器會失效,還會導(dǎo)致其他的攔截器也不再執(zhí)行。
  • postHandle():只有在 preHandle() 方法返回值為true 時才會執(zhí)行。會在Controller 中的方法調(diào)用之后,DispatcherServlet 返回渲染視圖之前被調(diào)用。 「有意思的是」:postHandle() 方法被調(diào)用的順序跟 preHandle() 是相反的,先聲明的攔截器 preHandle() 方法先執(zhí)行,而postHandle()方法反而會后執(zhí)行。
  • afterCompletion():只有在 preHandle() 方法返回值為true 時才會執(zhí)行。在整個請求結(jié)束之后, DispatcherServlet 渲染了對應(yīng)的視圖之后執(zhí)行。

使用攔截器的代碼

@Component
public class MyInterceptor implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        System.out.println("Interceptor 前置");
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {

        System.out.println("Interceptor 處理中");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {

        System.out.println("Interceptor 后置");
    }
}

將自定義好的攔截器處理類進行注冊,并通過addPathPatterns、excludePathPatterns等屬性設(shè)置需要攔截或需要排除的 URL。

@Configuration
public class MyMvcConfig implements WebMvcConfigurer {

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
    
         //  /**表示攔截所有的路徑
        registry.addInterceptor(new MyInterceptor()).addPathPatterns("/**");
        registry.addInterceptor(new MyInterceptor1()).addPathPatterns("/**");
    }
}

主要區(qū)別

(1)攔截器是基于java的反射機制的,而過濾器是基于函數(shù)回調(diào)。

(2)攔截器不依賴于servlet容器,而過濾器依賴于servlet容器。

(3)攔截器只能對action請求起作用,而過濾器則可以對幾乎所有的請求起作用。

(4)攔截器可以訪問action上下文、值棧里的對象,而過濾器不能。

(5)在action的生命周期中,攔截器可以多次被調(diào)用,而過濾器只能在容器初始化時被調(diào)用一次。

攔截器和過濾器的區(qū)別

實現(xiàn)原理不同

過濾器和攔截器 底層實現(xiàn)方式大不相同,過濾器 是基于函數(shù)回調(diào)的,攔截器 則是基于Java的反射機制(動態(tài)代理)實現(xiàn)的。

這里重點說下過濾器!
在我們自定義的過濾器中都會實現(xiàn)一個 doFilter()方法,這個方法有一個FilterChain 參數(shù),而實際上它是一個回調(diào)接口。ApplicationFilterChain是它的實現(xiàn)類, 這個實現(xiàn)類內(nèi)部也有一個 doFilter() 方法就是回調(diào)方法。

public interface FilterChain {

    void doFilter(ServletRequest var1, ServletResponse var2) throws IOException, ServletException;
}

75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?,Spring最新全家桶面試1000道必考題,spring,java,后端,springmvc,ssm,spring boot,spring cloud
ApplicationFilterChain里面能拿到我們自定義的xxxFilter類,在其內(nèi)部回調(diào)方法doFilter()里調(diào)用各個自定義xxxFilter過濾器,并執(zhí)行 doFilter() 方法。

public final class ApplicationFilterChain implements FilterChain {
    @Override
    public void doFilter(ServletRequest request, ServletResponse response) {
            ...//省略
            internalDoFilter(request,response);
    }
 
    private void internalDoFilter(ServletRequest request, ServletResponse response){
    if (pos < n) {
            //獲取第pos個filter    
            ApplicationFilterConfig filterConfig = filters[pos++];        
            Filter filter = filterConfig.getFilter();
            ...
            filter.doFilter(request, response, this);
        }
    }
 
}


而每個xxxFilter 會先執(zhí)行自身的 doFilter() 過濾邏輯,最后在執(zhí)行結(jié)束前會執(zhí)行filterChain.doFilter(servletRequest, servletResponse),也就是回調(diào)ApplicationFilterChain的doFilter() 方法,以此循環(huán)執(zhí)行實現(xiàn)函數(shù)回調(diào)。

 @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {

        filterChain.doFilter(servletRequest, servletResponse);
    }

使用范圍不同

我們看到過濾器 實現(xiàn)的是 javax.servlet.Filter 接口,而這個接口是在Servlet規(guī)范中定義的,也就是說過濾器Filter 的使用要依賴于Tomcat等容器,導(dǎo)致它只能在web程序中使用。
75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?,Spring最新全家桶面試1000道必考題,spring,java,后端,springmvc,ssm,spring boot,spring cloud
而攔截器(Interceptor) 它是一個Spring組件,并由Spring容器管理,并不依賴Tomcat等容器,是可以單獨使用的。不僅能應(yīng)用在web程序中,也可以用于Application、Swing等程序中。
75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?,Spring最新全家桶面試1000道必考題,spring,java,后端,springmvc,ssm,spring boot,spring cloud

攔截的請求范圍不同

在上邊我們已經(jīng)同時配置了過濾器和攔截器,再建一個Controller接收請求測試一下。

@Controller
@RequestMapping()
public class TestController {

    @RequestMapping("/test1")
    @ResponseBody
    public String test1(String a) {
        System.out.println("我是controller");
        return null;
    }
}


項目啟動過程中發(fā)現(xiàn),過濾器的init()方法,隨著容器的啟動進行了初始化。
75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?,Spring最新全家桶面試1000道必考題,spring,java,后端,springmvc,ssm,spring boot,spring cloud
此時瀏覽器發(fā)送請求,F(xiàn)12 看到居然有兩個請求,一個是我們自定義的 Controller 請求,另一個是訪問靜態(tài)圖標(biāo)資源的請求。
75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?,Spring最新全家桶面試1000道必考題,spring,java,后端,springmvc,ssm,spring boot,spring cloud
看到控制臺的打印日志如下:

執(zhí)行順序 :Filter 處理中 -> Interceptor 前置 -> 我是controller -> Interceptor 處理中 -> Interceptor 處理后
Filter 處理中
Interceptor 前置
Interceptor 處理中
Interceptor 后置
Filter 處理中

過濾器Filter執(zhí)行了兩次,攔截器Interceptor只執(zhí)行了一次。這是因為過濾器幾乎可以對所有進入容器的請求起作用,而攔截器只會對Controller中請求或訪問static目錄下的資源請求起作用。

注入Bean情況不同

在實際的業(yè)務(wù)場景中,應(yīng)用到過濾器或攔截器,為處理業(yè)務(wù)邏輯難免會引入一些service服務(wù)。

下邊我們分別在過濾器和攔截器中都注入service,看看有什么不同?

@Component
public class TestServiceImpl implements TestService {

    @Override
    public void a() {
        System.out.println("我是方法A");
    }
}


過濾器中注入service,發(fā)起請求測試一下 ,日志正常打印出“我是方法A”。

@Autowired
private TestService testService;

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {

        System.out.println("Filter 處理中");
        testService.a();
        filterChain.doFilter(servletRequest, servletResponse);
 }

Filter 處理中
我是方法A
Interceptor 前置
我是controller
Interceptor 處理中
Interceptor 后置

在攔截器中注入service,發(fā)起請求測試一下 ,竟然報錯了,debug跟一下發(fā)現(xiàn)注入的service怎么是Null???
75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?,Spring最新全家桶面試1000道必考題,spring,java,后端,springmvc,ssm,spring boot,spring cloud
這是因為加載順序?qū)е碌膯栴},攔截器加載的時間點在springcontext之前,而Bean又是由spring進行管理。

解決方案也很簡單,我們在注冊攔截器之前,先將Interceptor 手動進行注入。「注意」:在registry.addInterceptor()注冊的是getMyInterceptor() 實例。

@Configuration
public class MyMvcConfig implements WebMvcConfigurer {

    @Bean
    public MyInterceptor getMyInterceptor(){
        System.out.println("注入了MyInterceptor");
        return new MyInterceptor();
    }
    
    @Override
    public void addInterceptors(InterceptorRegistry registry) {

        registry.addInterceptor(getMyInterceptor()).addPathPatterns("/**");
    }
}

控制執(zhí)行順序不同

實際開發(fā)過程中,會出現(xiàn)多個過濾器或攔截器同時存在的情況,不過,有時我們希望某個過濾器或攔截器能優(yōu)先執(zhí)行,就涉及到它們的執(zhí)行順序。

過濾器用@Order注解控制執(zhí)行順序,通過@Order控制過濾器的級別,值越小級別越高越先執(zhí)行。

@Order(Ordered.HIGHEST_PRECEDENCE)
@Component
public class MyFilter2 implements Filter {

攔截器默認(rèn)的執(zhí)行順序,就是它的注冊順序,也可以通過Order手動設(shè)置控制,值越小越先執(zhí)行。

 @Override
 public void addInterceptors(InterceptorRegistry registry) {
 
        registry.addInterceptor(new MyInterceptor2()).addPathPatterns("/**").order(2);
        registry.addInterceptor(new MyInterceptor1()).addPathPatterns("/**").order(1);
        registry.addInterceptor(new MyInterceptor()).addPathPatterns("/**").order(3);
        
 }

看到輸出結(jié)果發(fā)現(xiàn),先聲明的攔截器 preHandle() 方法先執(zhí)行,而postHandle()方法反而會后執(zhí)行。

postHandle() 方法被調(diào)用的順序跟 preHandle() 居然是相反的!如果實際開發(fā)中嚴(yán)格要求執(zhí)行順序,那就需要特別注意這一點。

Interceptor1 前置
Interceptor2 前置
Interceptor 前置
我是controller
Interceptor 處理中
Interceptor2 處理中
Interceptor1 處理中
Interceptor 后置
Interceptor2 處理后
Interceptor1 處理后

「那為什么會這樣呢?」 得到答案就只能看源碼了,我們要知道controller 中所有的請求都要經(jīng)過核心組件DispatcherServlet路由,都會執(zhí)行它的 doDispatch() 方法,而攔截器postHandle()、preHandle()方法便是在其中調(diào)用的。

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    
        try {
         ...........
            try {
           
                // 獲取可以執(zhí)行當(dāng)前Handler的適配器
                HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

                // Process last-modified header, if supported by the handler.
                String method = request.getMethod();
                boolean isGet = "GET".equals(method);
                if (isGet || "HEAD".equals(method)) {
                    long lastModified = ha.getLastModified(request, mappedHandler.getHandler());
                    if (logger.isDebugEnabled()) {
                        logger.debug("Last-Modified value for [" + getRequestUri(request) + "] is: " + lastModified);
                    }
                    if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
                        return;
                    }
                }
                // 注意: 執(zhí)行Interceptor中PreHandle()方法
                if (!mappedHandler.applyPreHandle(processedRequest, response)) {
                    return;
                }

                // 注意:執(zhí)行Handle【包括我們的業(yè)務(wù)邏輯,當(dāng)拋出異常時會被Try、catch到】
                mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

                if (asyncManager.isConcurrentHandlingStarted()) {
                    return;
                }
                applyDefaultViewName(processedRequest, mv);

                // 注意:執(zhí)行Interceptor中PostHandle 方法【拋出異常時無法執(zhí)行】
                mappedHandler.applyPostHandle(processedRequest, response, mv);
            }
        }
        ...........
    }


看看兩個方法applyPreHandle()、applyPostHandle()具體是如何被調(diào)用的,就明白為什么postHandle()、preHandle() 執(zhí)行順序是相反的了。

boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {
        HandlerInterceptor[] interceptors = this.getInterceptors();
        if(!ObjectUtils.isEmpty(interceptors)) {
            for(int i = 0; i < interceptors.length; this.interceptorIndex = i++) {
                HandlerInterceptor interceptor = interceptors[i];
                if(!interceptor.preHandle(request, response, this.handler)) {
                    this.triggerAfterCompletion(request, response, (Exception)null);
                    return false;
                }
            }
        }

        return true;
    }

void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv) throws Exception {
        HandlerInterceptor[] interceptors = this.getInterceptors();
        if(!ObjectUtils.isEmpty(interceptors)) {
            for(int i = interceptors.length - 1; i >= 0; --i) {
                HandlerInterceptor interceptor = interceptors[i];
                interceptor.postHandle(request, response, this.handler, mv);
            }
        }
    }

發(fā)現(xiàn)兩個方法中在調(diào)用攔截器數(shù)組 HandlerInterceptor[] 時,循環(huán)的順序竟然是相反的。。。,導(dǎo)致postHandle()、preHandle() 方法執(zhí)行的順序相反。文章來源地址http://www.zghlxwxcb.cn/news/detail-835057.html

到了這里,關(guān)于75.SpringMVC的攔截器和過濾器有什么區(qū)別?執(zhí)行順序?的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實不符,請點擊違法舉報進行投訴反饋,一經(jīng)查實,立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費用

相關(guān)文章

  • 【SpringBoot】過濾器,監(jiān)聽器,攔截器介紹

    【SpringBoot】過濾器,監(jiān)聽器,攔截器介紹

    通過兩幅圖我們可以理解攔截器和過濾器的特點 1、過濾器 過濾器是在請求進入tomcat容器后,但請求進入servlet之前進行預(yù)處理的。請求結(jié)束返回也是,是在servlet處理完后,返回給前端之前。 理解上面這句話我們就可以知道,進入servlet之前,主要是兩個參數(shù):ServletRequest,

    2024年02月04日
    瀏覽(27)
  • 過濾器,監(jiān)聽器與攔截器的區(qū)別

    過濾器,監(jiān)聽器與攔截器的區(qū)別

    ? 過濾器和監(jiān)聽器不是Spring MVC中的組件,而是Servlet的組件,由Servlet容器來管理。攔截器是Spring MVC中的組件,由Spring容器來管理 ? Servlet過濾器與Spring MVC 攔截器在Web應(yīng)用中所處的層次如下圖所示: 過濾器是Servlet的高級特性之一,是實現(xiàn)Filter接口的Java類。其基本功能就是對

    2024年02月14日
    瀏覽(53)
  • 登錄頁面jwt密鑰,過濾器,攔截器,異常處理

    需求: 用戶未登錄時,訪問其他也沒面,操作添加、刪除等操作時,強行跳轉(zhuǎn)至登錄頁面。 實現(xiàn)方法: 1.使用Cookie,登錄后后端添加一個cookie,每次頁面判斷是否有cookie, 2。使用session,原理同上,只不過session是存儲在服務(wù)器里的,cookie是在瀏覽器里。 3。使用jwt令牌,登

    2023年04月25日
    瀏覽(26)
  • spring boot 過濾器&攔截器與aop

    spring boot 過濾器&攔截器與aop

    在使用 Spring 框架時,可以通過在 web.xml 配置文件中注冊過濾器,使其在請求進入 Spring 前就能夠進行預(yù)處理。這樣可以在請求進入 Spring MVC 的 DispatcherServlet 之前,對請求進行攔截、修改或者過濾。 過濾器在 Spring 中的應(yīng)用場景包括但不限于: 字符編碼過濾:通過過濾器,在

    2024年02月01日
    瀏覽(22)
  • Springboot中使用攔截器、過濾器、監(jiān)聽器

    Springboot中使用攔截器、過濾器、監(jiān)聽器

    Javaweb三大組件:servlet、Filter(過濾器)、?Listener(監(jiān)聽器) SpringBoot特有組件:Interceptor(攔截器) 過濾器、攔截器、監(jiān)聽器、AOP(后續(xù)文章介紹)、全局異常處理器(后續(xù)文章介紹)是搭建系統(tǒng)框架時,經(jīng)常用到的部分,全局異常處理器的作用很明顯,就是處理接口執(zhí)行

    2024年02月03日
    瀏覽(25)
  • Spring Boot攔截器與過濾器的區(qū)別

    Spring Boot攔截器與過濾器的區(qū)別

    在使用Spring Boot開發(fā)Web應(yīng)用程序時,您可能需要在處理請求之前或之后執(zhí)行某些操作。這些操作可以包括身份驗證、日志記錄、性能監(jiān)測等。在這種情況下,您可以使用兩種不同的機制:攔截器和過濾器。本文將介紹這兩種機制及其區(qū)別,并提供一些示例代碼來演示如何在S

    2024年02月08日
    瀏覽(23)
  • SpringBoot2.0(過濾器,監(jiān)聽器,攔截器)

    SpringBoot2.0(過濾器,監(jiān)聽器,攔截器)

    使用Servlet3.0的注解進行配置 啟動類里面增加 @ServletComponentScan ,進行掃描 新建一個Filter類,implements Filter ,并實現(xiàn)對應(yīng)接口 @WebFilter 標(biāo)記一個類為Filter,被spring進行掃描 urlPatterns:攔截規(guī)則,支持正則 控制chain.doFilter的方法的調(diào)用,來實現(xiàn)是否通過放行, 不放行的話,web應(yīng)用

    2024年02月07日
    瀏覽(19)
  • springbboot攔截器,過濾器,監(jiān)聽器及執(zhí)行流程

    springbboot攔截器,過濾器,監(jiān)聽器及執(zhí)行流程

    過濾器是在請求進入容器后,但請求進入servlet之前進行預(yù)處理的。請求結(jié)束返回也是,是在servlet處理完后,返回給前端之前 請求按照上圖進入conteoller后執(zhí)行完再返回 過濾器是Servlet規(guī)范中定義的一種組件,可以用于在請求進入Web應(yīng)用程序之前或響應(yīng)離開Web應(yīng)用程序之前對請

    2024年02月13日
    瀏覽(19)
  • 過濾器(Filter)和攔截器(Interceptor)有什么不同?

    過濾器(Filter)和攔截器(Interceptor)有什么不同?

    過濾器(Filter)和攔截器(Interceptor)是用于處理請求和響應(yīng)的中間件組件,但它們在實現(xiàn)方式和應(yīng)用場景上有一些不同。 過濾器 是Servlet規(guī)范中定義的一種組件,通常以Java類的形式實現(xiàn)。過濾器通過在 web.xml 配置文件中聲明來注冊,并在Web應(yīng)用程序的請求和響應(yīng)鏈中攔截請

    2024年02月07日
    瀏覽(24)
  • Spring中的攔截器與過濾器:原理、區(qū)別與案例解析

    Spring中的攔截器與過濾器:原理、區(qū)別與案例解析

    前言 在Web應(yīng)用中,我們經(jīng)常需要對用戶的請求進行某種處理,比如權(quán)限驗證、日志記錄等。 Spring框架提供了兩種機制來實現(xiàn)這一需求:攔截器和過濾器。雖然它們的目標(biāo)相似,但在使用上存在一些差異。本篇文章我們將詳細探討這兩種機制的原理、區(qū)別,希望能給各位大佬

    2024年04月23日
    瀏覽(22)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包