摘要:并且,并且如果或者不為空不為且不為,將中斷處理直接返回不再渲染頁面對返回值的處理對返回值的處理是使用完成的對異步處理結(jié)果的處理使用示例文首說了,作為一個非公開,如果你要直接使用起來,還是稍微要費(fèi)點(diǎn)勁的。
每篇一句
想當(dāng)火影的人沒有近道可尋,當(dāng)上火影的人同樣無路可退前言
HandlerMethod它作為Spring MVC的非公開API,可能絕大多數(shù)小伙伴都對它比較陌生,但我相信你對它又不是那么的生疏,因?yàn)槟憧赡軟]用過但肯定見過。
比如Spring MVC的攔截器HandlerInterceptor的攔截方法的第三個入?yún)?b>Object handler,雖然它是Object類型,但其實(shí)絕大部分情況下我們都會當(dāng)作HandlerMethod來使用;又比如我之前的這篇講RequestMappingHandlerMapping的文章也大量的提到過HandlerMethod這個類。
經(jīng)由我這么“忽悠”,你是否覺得它還是相對比較重要的一個類了呢?不管你信不信,反正我是這么認(rèn)為的:HandlerMethod它是理解Spring MVC不可或缺的一個類,甚至可以說是你希望參與到Spring MVC的定制化里面來不可忽略的一個關(guān)鍵API。
HandlerMethodHandlerMethod它不是一個接口,也不是個抽象類,且還是public的。HandlerMethod封裝了很多屬性,在訪問請求方法的時候可以方便的訪問到方法、方法參數(shù)、方法上的注解、所屬類等并且對方法參數(shù)封裝處理,也可以方便的訪問到方法參數(shù)的注解等信息。
// @since 3.1 public class HandlerMethod { // Object類型,既可以是個Bean,也可以是個BeanName private final Object bean; // 如果是BeanName,拿就靠它拿出Bean實(shí)例了~ @Nullable private final BeanFactory beanFactory; private final Class> beanType; // 該方法所屬的類 private final Method method; // 該方法本身 private final Method bridgedMethod; // 被橋接的方法,如果method是原生的,它的值同method // 封裝方法參數(shù)的類實(shí)例,**一個MethodParameter就是一個入?yún)?* // MethodParameter也是Spring抽象出來的一個非常重要的概念 private final MethodParameter[] parameters; @Nullable private HttpStatus responseStatus; // http狀態(tài)碼(畢竟它要負(fù)責(zé)處理和返回) @Nullable private String responseStatusReason; // 如果狀態(tài)碼里還要復(fù)數(shù)原因,就是這個字段 可以為null // 通過createWithResolvedBean()解析此handlerMethod實(shí)例的handlerMethod。 @Nullable private HandlerMethod resolvedFromHandlerMethod; // 標(biāo)注在**接口入?yún)?*上的注解們(此處數(shù)據(jù)結(jié)構(gòu)復(fù)雜,List+二維數(shù)組) @Nullable private volatile ListinterfaceParameterAnnotations; // 它的構(gòu)造方法眾多 此處我只寫出關(guān)鍵的步驟 public HandlerMethod(Object bean, Method method) { ... this.beanType = ClassUtils.getUserClass(bean); this.bridgedMethod = BridgeMethodResolver.findBridgedMethod(method); this.parameters = initMethodParameters(); ... evaluateResponseStatus(); } // 這個構(gòu)造方法拋出了一個異常NoSuchMethodException public HandlerMethod(Object bean, String methodName, Class>... parameterTypes) throws NoSuchMethodException { ... this.method = bean.getClass().getMethod(methodName, parameterTypes); this.parameters = initMethodParameters(); ... evaluateResponseStatus(); } // 此處傳的是BeanName public HandlerMethod(String beanName, BeanFactory beanFactory, Method method) { ... // 這部判斷:這個BeanName是必須存在的 Class> beanType = beanFactory.getType(beanName); if (beanType == null) { throw new IllegalStateException("Cannot resolve bean type for bean with name "" + beanName + """); } this.parameters = initMethodParameters(); ... evaluateResponseStatus(); } // 供給子類copy使用的 protected HandlerMethod(HandlerMethod handlerMethod) { ... } // 所有構(gòu)造都執(zhí)行了兩個方法:initMethodParameters和evaluateResponseStatus // 初始化該方法所有的入?yún)?,此處使用的是?nèi)部類HandlerMethodParameter // 注意:處理了泛型的~~~ private MethodParameter[] initMethodParameters() { int count = this.bridgedMethod.getParameterCount(); MethodParameter[] result = new MethodParameter[count]; for (int i = 0; i < count; i++) { HandlerMethodParameter parameter = new HandlerMethodParameter(i); GenericTypeResolver.resolveParameterType(parameter, this.beanType); result[i] = parameter; } return result; } // 看看方法上是否有標(biāo)注了@ResponseStatus注解(接口上或者父類 組合注解上都行) // 若方法上沒有,還會去所在的類上去看看有沒有標(biāo)注此注解 // 主要只解析這個注解,把它的兩個屬性code和reason拿過來,最后就是返回它倆了~~~ // code狀態(tài)碼默認(rèn)是HttpStatus.INTERNAL_SERVER_ERROR-->(500, "Internal Server Error") private void evaluateResponseStatus() { ResponseStatus annotation = getMethodAnnotation(ResponseStatus.class); if (annotation == null) { annotation = AnnotatedElementUtils.findMergedAnnotation(getBeanType(), ResponseStatus.class); } if (annotation != null) { this.responseStatus = annotation.code(); this.responseStatusReason = annotation.reason(); } } ... // 省略所有屬性的get方法(無set方法) // 返回方法返回值的類型 此處也使用的MethodParameter public MethodParameter getReturnType() { return new HandlerMethodParameter(-1); } // 注意和上面的區(qū)別。舉個列子:比如方法返回的是Object,但實(shí)際return “fsx”字符串 // 那么上面返回永遠(yuǎn)是Object.class,下面你實(shí)際的值是什么類型就是什么類型 public MethodParameter getReturnValueType(@Nullable Object returnValue) { return new ReturnValueMethodParameter(returnValue); } // 該方法的返回值是否是void public boolean isVoid() { return Void.TYPE.equals(getReturnType().getParameterType()); } // 返回標(biāo)注在方法上的指定類型的注解 父方法也成 // 子類ServletInvocableHandlerMethod對下面兩個方法都有復(fù)寫~~~ @Nullable public A getMethodAnnotation(Class annotationType) { return AnnotatedElementUtils.findMergedAnnotation(this.method, annotationType); } public boolean hasMethodAnnotation(Class annotationType) { return AnnotatedElementUtils.hasAnnotation(this.method, annotationType); } // resolvedFromHandlerMethod雖然它只能被構(gòu)造進(jìn)來,但是它實(shí)際是銅鼓調(diào)用下面方法賦值 @Nullable public HandlerMethod getResolvedFromHandlerMethod() { return this.resolvedFromHandlerMethod; } // 根據(jù)string類型的BeanName把Bean拿出來,再new一個HandlerMethod出來~~~這才靠譜嘛 public HandlerMethod createWithResolvedBean() { Object handler = this.bean; if (this.bean instanceof String) { Assert.state(this.beanFactory != null, "Cannot resolve bean name without BeanFactory"); String beanName = (String) this.bean; handler = this.beanFactory.getBean(beanName); } return new HandlerMethod(this, handler); } public String getShortLogMessage() { return getBeanType().getName() + "#" + this.method.getName() + "[" + this.method.getParameterCount() + " args]"; } // 這個方法是提供給內(nèi)部類HandlerMethodParameter來使用的~~ 它使用的數(shù)據(jù)結(jié)構(gòu)還是蠻復(fù)雜的 private List getInterfaceParameterAnnotations() { List parameterAnnotations = this.interfaceParameterAnnotations; if (parameterAnnotations == null) { parameterAnnotations = new ArrayList<>(); // 遍歷該方法所在的類所有的實(shí)現(xiàn)的接口們(可以實(shí)現(xiàn)N個接口嘛) for (Class> ifc : this.method.getDeclaringClass().getInterfaces()) { // getMethods:拿到所有的public的方法,包括父接口的 接口里的私有方法可不會獲取來 for (Method candidate : ifc.getMethods()) { // 判斷這個接口方法是否正好是當(dāng)前method復(fù)寫的這個~~~ // 剛好是復(fù)寫的方法,那就添加進(jìn)來,標(biāo)記為接口上的注解們~~~ if (isOverrideFor(candidate)) { // getParameterAnnotations返回的是個二維數(shù)組~~~~ // 因?yàn)閰?shù)有多個,且每個參數(shù)前可以有多個注解 parameterAnnotations.add(candidate.getParameterAnnotations()); } } } this.interfaceParameterAnnotations = parameterAnnotations; } return parameterAnnotations; } // 看看內(nèi)部類的關(guān)鍵步驟 protected class HandlerMethodParameter extends SynthesizingMethodParameter { @Nullable private volatile Annotation[] combinedAnnotations; ... // 父類只會在本方法拿,這里支持到了接口級別~~~ @Override public Annotation[] getParameterAnnotations() { Annotation[] anns = this.combinedAnnotations; if (anns == null) { // 都只需要解析一次 anns = super.getParameterAnnotations(); int index = getParameterIndex(); if (index >= 0) { // 有入?yún)⒉判枰シ治雎? for (Annotation[][] ifcAnns : getInterfaceParameterAnnotations()) { if (index < ifcAnns.length) { Annotation[] paramAnns = ifcAnns[index]; if (paramAnns.length > 0) { List merged = new ArrayList<>(anns.length + paramAnns.length); merged.addAll(Arrays.asList(anns)); for (Annotation paramAnn : paramAnns) { boolean existingType = false; for (Annotation ann : anns) { if (ann.annotationType() == paramAnn.annotationType()) { existingType = true; break; } } if (!existingType) { merged.add(adaptAnnotation(paramAnn)); } } anns = merged.toArray(new Annotation[0]); } } } } this.combinedAnnotations = anns; } return anns; } } // 返回值的真正類型~~~ private class ReturnValueMethodParameter extends HandlerMethodParameter { @Nullable private final Object returnValue; public ReturnValueMethodParameter(@Nullable Object returnValue) { super(-1); // 此處傳的-1哦~~~~ 比0小是很有意義的 this.returnValue = returnValue; } ... // 返回值類型使用returnValue就行了~~~ @Override public Class> getParameterType() { return (this.returnValue != null ? this.returnValue.getClass() : super.getParameterType()); } } }
可以看到HandlerMethod它持有的屬性是非常多的,提供的能力也是很強(qiáng)的。
但是不知道小伙伴有沒有發(fā)現(xiàn),雖然它持有了目標(biāo)的Method,但是它并沒有提供invoke執(zhí)行它的能力,如果你要執(zhí)行它還得自己把Method拿去自己執(zhí)行。
所以總的來說它的職責(zé)還是很單一的:HandlerMethod它只負(fù)責(zé)準(zhǔn)備數(shù)據(jù),封裝數(shù)據(jù),而而不提供具體使用的方式方法~
看看它的繼承樹:
它主要有兩個子類:InvocableHandlerMethod和ServletInvocableHandlerMethod,從命名就知道他倆都是有invoke調(diào)用能力的~
它是對HandlerMethod的擴(kuò)展,增加了調(diào)用能力。這個能力在Spring MVC可是非常非常重要的,它能夠在調(diào)用的時候,把方法入?yún)⒌膮?shù)都封裝進(jìn)來(從HTTP request里,當(dāng)然借助的必然是HandlerMethodArgumentResolver)
// @since 3.1 public class InvocableHandlerMethod extends HandlerMethod { private static final Object[] EMPTY_ARGS = new Object[0]; // 它額外提供的幾個屬性,可以看到和數(shù)據(jù)綁定、數(shù)據(jù)校驗(yàn)就扯上關(guān)系了~~~ // 用于產(chǎn)生數(shù)據(jù)綁定器、校驗(yàn)器 @Nullable private WebDataBinderFactory dataBinderFactory; // HandlerMethodArgumentResolver用于入?yún)⒌慕馕? private HandlerMethodArgumentResolverComposite resolvers = new HandlerMethodArgumentResolverComposite(); // 用于獲取形參名 private ParameterNameDiscoverer parameterNameDiscoverer = new DefaultParameterNameDiscoverer(); ... // 省略構(gòu)造函數(shù) 全部使用super的 // 它自己的三大屬性都使用set方法設(shè)置進(jìn)來~~~并且沒有提供get方法 // 也就是說:它自己內(nèi)部使用就行了~~~ // 在給定請求的上下文中解析方法的參數(shù)值后調(diào)用該方法。 也就是說:方法入?yún)⒗锞湍軌蜃詣邮褂谜埱笥颍ò╬ath里的,requestParam里的、以及常規(guī)對象如HttpSession這種) // 解釋下providedArgs作用:調(diào)用者可以傳進(jìn)來,然后直接doInvoke()的時候原封不動的使用它 //(彌補(bǔ)了請求域沒有所有對象的不足,畢竟有些對象是用戶自定義的嘛~) @Nullable public Object invokeForRequest(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception { // 雖然它是最重要的方法,但是此處不講,因?yàn)楹诵脑瓉磉€是`HandlerMethodArgumentResolver` // 它只是把解析好的放到對應(yīng)位置里去~~~ // 說明:這里傳入了ParameterNameDiscoverer,它是能夠獲取到形參名的。 // 這就是為何注解里我們不寫value值,通過形參名字來匹配也是ok的核心原因~ Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs); if (logger.isTraceEnabled()) { // trace信息,否則日志也特多了~ logger.trace("Arguments: " + Arrays.toString(args)); } return doInvoke(args); } // doInvoke()方法就不說了,就是個普通的方法調(diào)用 // ReflectionUtils.makeAccessible(getBridgedMethod()); // return getBridgedMethod().invoke(getBean(), args); }
對于最后的invoke(),說明一點(diǎn):這里可是執(zhí)行的目標(biāo)方法getBean()哦~~~
這個子類主要提供的能力就是提供了invoke調(diào)用目標(biāo)Bean的目標(biāo)方法的能力,在這個調(diào)用過程中可大有文章可為,當(dāng)然最為核心的邏輯可是各種各樣的HandlerMethodArgumentResolver來完成的,詳見下文有分曉。
InvocableHandlerMethod這個子類雖然它提供了調(diào)用了能力,但是它卻依舊還沒有和Servlet的API綁定起來,畢竟使用的是Spring自己通用的的NativeWebRequest,so很容易想到它還有一個子類就是干這事的~
它是對InvocableHandlerMethod的擴(kuò)展,它增加了返回值和響應(yīng)狀態(tài)碼的處理,另外在ServletInvocableHandlerMethod有個內(nèi)部類ConcurrentResultHandlerMethod繼承于它,支持異常調(diào)用結(jié)果處理,Servlet容器下Controller在查找適配器時發(fā)起調(diào)用的最終就是ServletInvocableHandlerMethod。
public class ServletInvocableHandlerMethod extends InvocableHandlerMethod { private static final Method CALLABLE_METHOD = ClassUtils.getMethod(Callable.class, "call"); // 處理方法返回值 @Nullable private HandlerMethodReturnValueHandlerComposite returnValueHandlers; // 構(gòu)造函數(shù)略 // 設(shè)置處理返回值的HandlerMethodReturnValueHandler public void setHandlerMethodReturnValueHandlers(HandlerMethodReturnValueHandlerComposite returnValueHandlers) { this.returnValueHandlers = returnValueHandlers; } // 它不是復(fù)寫,但是是對invokeForRequest方法的進(jìn)一步增強(qiáng) 因?yàn)檎{(diào)用目標(biāo)方法還是靠invokeForRequest // 本處是把方法的返回值拿來進(jìn)一步處理~~~比如狀態(tài)碼之類的 public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception { Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs); // 設(shè)置HttpServletResponse返回狀態(tài)碼 這里面還是有點(diǎn)意思的 因?yàn)锧ResponseStatus#code()在父類已經(jīng)解析了 但是子類才用 setResponseStatus(webRequest); // 重點(diǎn)是這一句話:mavContainer.setRequestHandled(true); 表示該請求已經(jīng)被處理過了 if (returnValue == null) { // Request的NotModified為true 有@ResponseStatus注解標(biāo)注 RequestHandled=true 三個條件有一個成立,則設(shè)置請求處理完成并返回 if (isRequestNotModified(webRequest) || getResponseStatus() != null || mavContainer.isRequestHandled()) { mavContainer.setRequestHandled(true); return; } // 返回值不為null,@ResponseStatus存在reason 同樣設(shè)置請求處理完成并返回 } else if (StringUtils.hasText(getResponseStatusReason())) { mavContainer.setRequestHandled(true); return; } // 前邊都不成立,則設(shè)置RequestHandled=false即請求未完成 // 繼續(xù)交給HandlerMethodReturnValueHandlerComposite處理 // 可見@ResponseStatus的優(yōu)先級還是蠻高的~~~~~ mavContainer.setRequestHandled(false); Assert.state(this.returnValueHandlers != null, "No return value handlers"); try { // 關(guān)于對方法返回值的處理,參見:https://blog.csdn.net/f641385712/article/details/90370542 this.returnValueHandlers.handleReturnValue(returnValue, getReturnValueType(returnValue), mavContainer, webRequest); } catch (Exception ex) { if (logger.isTraceEnabled()) { logger.trace(formatErrorForReturnValue(returnValue), ex); } throw ex; } } // 設(shè)置返回的狀態(tài)碼到HttpServletResponse 里面去 private void setResponseStatus(ServletWebRequest webRequest) throws IOException { HttpStatus status = getResponseStatus(); if (status == null) { // 如果調(diào)用者沒有標(biāo)注ResponseStatus.code()此注解 此處就忽略它 return; } HttpServletResponse response = webRequest.getResponse(); if (response != null) { String reason = getResponseStatusReason(); // 此處務(wù)必注意:若有reason,那就是sendError 哪怕你是200哦~ if (StringUtils.hasText(reason)) { response.sendError(status.value(), reason); } else { response.setStatus(status.value()); } } // 設(shè)置到request的屬性,把響應(yīng)碼給過去。為了在redirect中使用 // To be picked up by RedirectView webRequest.getRequest().setAttribute(View.RESPONSE_STATUS_ATTRIBUTE, status); } private boolean isRequestNotModified(ServletWebRequest webRequest) { return webRequest.isNotModified(); } // 這個方法RequestMappingHandlerAdapter里有調(diào)用 ServletInvocableHandlerMethod wrapConcurrentResult(Object result) { return new ConcurrentResultHandlerMethod(result, new ConcurrentResultMethodParameter(result)); } // 內(nèi)部類們 private class ConcurrentResultMethodParameter extends HandlerMethodParameter { @Nullable private final Object returnValue; private final ResolvableType returnType; public ConcurrentResultMethodParameter(Object returnValue) { super(-1); this.returnValue = returnValue; // 主要是這個解析 兼容到了泛型類型 比如你的返回值是List它也能把你的類型拿出來 this.returnType = (returnValue instanceof ReactiveTypeHandler.CollectedValuesList ? ((ReactiveTypeHandler.CollectedValuesList) returnValue).getReturnType() : ResolvableType.forType(super.getGenericParameterType()).getGeneric()); } // 若返回的是List 這里就是List的類型哦 下面才是返回泛型類型 @Override public Class> getParameterType() { if (this.returnValue != null) { return this.returnValue.getClass(); } if (!ResolvableType.NONE.equals(this.returnType)) { return this.returnType.toClass(); } return super.getParameterType(); } // 返回泛型類型 @Override public Type getGenericParameterType() { return this.returnType.getType(); } // 即使實(shí)際返回類型為ResponseEntity >,也要確保對@ResponseBody-style處理從reactive 類型中收集值 // 是對reactive 的一種兼容 @Override public boolean hasMethodAnnotation(Class annotationType) { // Ensure @ResponseBody-style handling for values collected from a reactive type // even if actual return type is ResponseEntity > return (super.hasMethodAnnotation(annotationType) || (annotationType == ResponseBody.class && this.returnValue instanceof ReactiveTypeHandler.CollectedValuesList)); } } // 這個非常有意思 內(nèi)部類繼承了自己(外部類) 進(jìn)行增強(qiáng) private class ConcurrentResultHandlerMethod extends ServletInvocableHandlerMethod { // 返回值 private final MethodParameter returnType; // 此構(gòu)造最終傳入的handler是個Callable // result方法返回值 它支持支持異常調(diào)用結(jié)果處理 public ConcurrentResultHandlerMethod(final Object result, ConcurrentResultMethodParameter returnType) { super((Callable
HandlerMethod用于封裝Handler和處理請求的Method;InvocableHandlerMethod增加了方法參數(shù)解析和調(diào)用方法的能力;ServletInvocableHandlerMethod在此基礎(chǔ)上在增加了如下三個能力:
對@ResponseStatus注解的支持
1.當(dāng)一個方法注釋了`@ResponseStatus`后,**響應(yīng)碼就是注解上的響應(yīng)碼**。 **并且,并且如果returnValue=null或者reason不為空**(不為null且不為“”),將中斷處理直接返回(不再渲染頁面)
對返回值returnValue的處理
1. 對返回值的處理是使用`HandlerMethodReturnValueHandlerComposite`完成的
對異步處理結(jié)果的處理
使用示例文首說了,HandlerMethod作為一個非公開API,如果你要直接使用起來,還是稍微要費(fèi)點(diǎn)勁的。
但本文還是給出一個Demo,給出小伙伴們最為關(guān)心也是對你們最有用的一個需求:ModelFactory.getNameForParameter(parameter)這個靜態(tài)方法是給入?yún)⑸赡J(rèn)名稱的,當(dāng)然默認(rèn)處理方案最底層依賴的是它Conventions.getVariableNameForParameter(parameter),為了驗(yàn)證這塊對象、Object、List等等常用數(shù)據(jù)結(jié)構(gòu)的默認(rèn)處理,此處我借助HandlerMethod一次性全部打印出這個結(jié)論:
@Getter @Setter @ToString public class Person { @NotNull private String name; @NotNull @Positive private Integer age; public Object demoMethod(Person person, Object object, ListintList, List personList, Set intSet, Set personSet, Map myMap, String name, Integer age, int number, double money) { return "hello parameter"; } }
借助HandlerMethod完成此測試用例
public static void main(String[] args) { // 準(zhǔn)備一個HandlerMethod HandlerMethod handlerMethod = new HandlerMethod(new Person(), getPersonSpecfyMethod()); // 拿到該方法所有的參數(shù) MethodParameter[] methodParameters = handlerMethod.getMethodParameters(); for (MethodParameter parameter : methodParameters) { Class> parameterType = parameter.getParameterType(); String nameForParameter = ModelFactory.getNameForParameter(parameter); System.out.println("類型" + parameterType.getName() + "--->缺省的modelKey是:" + nameForParameter); } } private static Method getPersonSpecfyMethod() { for (Method method : Person.class.getMethods()) if (method.getName().equals("demoMethod")) return method; return null; }
運(yùn)行,打印結(jié)果如下:
類型com.fsx.bean.Person--->缺省的modelKey是:person 類型java.lang.Object--->缺省的modelKey是:object 類型java.util.List--->缺省的modelKey是:integerList 類型java.util.List--->缺省的modelKey是:personList 類型java.util.Set--->缺省的modelKey是:integerList // 可以看到即使是set 名稱也是同List的 類型java.util.Set--->缺省的modelKey是:personList 類型java.util.Map--->缺省的modelKey是:map 類型java.lang.String--->缺省的modelKey是:string 類型java.lang.Integer--->缺省的modelKey是:integer 類型int--->缺省的modelKey是:int 類型double--->缺省的modelKey是:double
這個結(jié)果是不同類型對應(yīng)的缺省的ModelKey,希望小伙伴們能夠記下來,這對理解和正確使用`
@SessionAttribute、@ModelAttribute`都是很重要的~
HandlerMethod雖然接觸少,但并不影響它的重要性。在理解Spring MVC的處理流程上它很重要,在與使用者關(guān)系較大的攔截器HandlerInterceptor定制化處理的時候,學(xué)會使用它一樣是非常有必要的。
在最后還提示大家一個你可能沒有關(guān)心到的小細(xì)節(jié):
HandlerMethod位于org.springframework.web.method包下,且是3.1后才有的
MethodParameter位于org.springframework.core核心包中。2.0就存在了
相關(guān)閱讀【小家Spring】Spring MVC容器的web九大組件之---HandlerAdapter源碼詳解---一篇文章帶你讀懂返回值處理器HandlerMethodReturnValueHandler
知識交流==The last:如果覺得本文對你有幫助,不妨點(diǎn)個贊唄。當(dāng)然分享到你的朋友圈讓更多小伙伴看到也是被作者本人許可的~==
**若對技術(shù)內(nèi)容感興趣可以加入wx群交流:Java高工、架構(gòu)師3群。
若群二維碼失效,請加wx號:fsx641385712(或者掃描下方wx二維碼)。并且備注:"java入群" 字樣,會手動邀請入群**
若文章格式混亂或者圖片裂開,請點(diǎn)擊`:原文鏈接-原文鏈接-原文鏈接
==若對Spring、SpringBoot、MyBatis等源碼分析感興趣,可加我wx:fsx641385712,手動邀請你入群一起飛==
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/77915.html
摘要:見名之意,它是處理器,也就是解析這個注解的核心。管理通過標(biāo)注了的特定會話屬性,存儲最終是委托了來實(shí)現(xiàn)。只會清楚注解放進(jìn)去的,并不清除放進(jìn)去的它的唯一實(shí)現(xiàn)類實(shí)現(xiàn)也簡單。在更新時,模型屬性與會話同步,如果缺少,還將添加屬性。 每篇一句 不是你當(dāng)上了火影大家就認(rèn)可你,而是大家都認(rèn)可你才能當(dāng)上火影 前言 該注解顧名思義,作用是將Model中的屬性同步到session會話當(dāng)中,方便在下一次請求中...
摘要:雖然它不是必須,但是它是個很好的輔助官方解釋首先看看官方的對它怎么說它將方法參數(shù)方法返回值綁定到的里面。解析注解標(biāo)注的方法參數(shù),并處理標(biāo)注的方法返回值。 每篇一句 我們應(yīng)該做一個:胸中有藍(lán)圖,腳底有計劃的人 前言 Spring MVC提供的基于注釋的編程模型,極大的簡化了web應(yīng)用的開發(fā),我們都是受益者。比如我們在@RestController標(biāo)注的Controller控制器組件上用@...
摘要:并且,并且如果或者不為空不為且不為,將中斷處理直接返回不再渲染頁面對返回值的處理對返回值的處理是使用完成的對異步處理結(jié)果的處理使用示例文首說了,作為一個非公開,如果你要直接使用起來,還是稍微要費(fèi)點(diǎn)勁的。 每篇一句 想當(dāng)火影的人沒有近道可尋,當(dāng)上火影的人同樣無路可退 前言 HandlerMethod它作為Spring MVC的非公開API,可能絕大多數(shù)小伙伴都對它比較陌生,但我相信你對它...
摘要:并且,并且如果或者不為空不為且不為,將中斷處理直接返回不再渲染頁面對返回值的處理對返回值的處理是使用完成的對異步處理結(jié)果的處理使用示例文首說了,作為一個非公開,如果你要直接使用起來,還是稍微要費(fèi)點(diǎn)勁的。 每篇一句 想當(dāng)火影的人沒有近道可尋,當(dāng)上火影的人同樣無路可退 前言 HandlerMethod它作為Spring MVC的非公開API,可能絕大多數(shù)小伙伴都對它比較陌生,但我相信你對它...
摘要:和一起使用參照博文從原理層面掌握的使用一起學(xué)。至于具體原因,可以移步這里輔助理解從原理層面掌握的使用核心原理篇一起學(xué)再看下面的變種例子重要訪問。 每篇一句 每個人都應(yīng)該想清楚這個問題:你是祖師爺賞飯吃的,還是靠老天爺賞飯吃的 前言 上篇文章 描繪了@ModelAttribute的核心原理,這篇聚焦在場景使用上,演示@ModelAttribute在不同場景下的使用,以及注意事項(xiàng)(當(dāng)然有些...
閱讀 494·2023-04-26 00:33
閱讀 3521·2021-11-24 09:39
閱讀 2834·2021-09-22 15:34
閱讀 2289·2019-08-23 18:07
閱讀 2897·2019-08-23 18:04
閱讀 3669·2019-08-23 16:06
閱讀 2882·2019-08-23 15:27
閱讀 1592·2019-08-23 14:32