亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

SpringMVC中@InitBinder注解怎么使用

發布時間:2022-07-02 13:55:18 來源:億速云 閱讀:303 作者:iii 欄目:開發技術

這篇文章主要講解了“SpringMVC中@InitBinder注解怎么使用”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“SpringMVC中@InitBinder注解怎么使用”吧!

簡介

@Controller或@ControllerAdvice類可以有@InitBinder方法來初始化WebDataBinder的實例,這些方法可以:

  • 將請求參數(即表單或查詢數據)綁定到模型對象。

  • 將基于字符串的請求值(如請求參數、路徑變量、頭、cookie等)轉換為控制器方法參數的目標類型。

  • 渲染HTML表單時,將模型對象的值格式化為字符串值。

@InitBinder方法可以注冊控制器特定的java.bean.PropertyEditor或Spring Converter和 Formatter組件。另外,你可以使用MVC配置在全局共享的FormattingConversionService中注冊Converter和Formatter類型。

@InitBinder方法支持許多與@RequestMapping方法相同的參數,除了@ModelAttribute(命令對象)參數。通常,它們是用WebDataBinder參數(用于注冊)和一個void返回值聲明的。

應用示例

@RestController
@RequestMapping("/demos")
public class DemoController {
  @InitBinder // 1
  public void bind(WebDataBinder binder) { // 2
    binder.registerCustomEditor(Long.class, new PropertyEditorSupport() { // 3
      @Override
      public void setAsText(String text) throws IllegalArgumentException {
        setValue(Long.valueOf(text) + 666L) ;
      }
    }) ;
  }
  @GetMapping("/index")
  public Object index(Long id) {
    return "index - " + id ;
  }
}

注意以下幾點:

  • 使用 @InitBinder 注解。

  • 接收 WebDataBinder 參數。

  • 注冊自定義的轉換器。

  • 方法返回值必須是 void。

在上面的示例中注冊了一個類型轉換器從字符串轉換為Long類型 并且在原來值基礎上增加了666L。

原理解讀

HandlerAdapter 執行。

public class RequestMappingHandlerAdapter extends AbstractHandlerMethodAdapter implements BeanFactoryAware, InitializingBean {
  protected ModelAndView invokeHandlerMethod(HttpServletRequest request, HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {
    // ...
    // 這里會查找當前執行的Controller中定義的所有@InitBinder注解的方法
    WebDataBinderFactory binderFactory = getDataBinderFactory(handlerMethod);
    ServletInvocableHandlerMethod invocableMethod = createInvocableHandlerMethod(handlerMethod);
    invocableMethod.invokeAndHandle(webRequest, mavContainer);
    // ...
  }
}

ServletInvocableHandlerMethod 執行。

public class ServletInvocableHandlerMethod extends InvocableHandlerMethod {
  public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {
    // 調用父類方法
    Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs);
    // ...
  }
}
// 執行父類方法調用
public class InvocableHandlerMethod extends HandlerMethod {
  public Object invokeForRequest(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {
    Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs);
    return doInvoke(args);
  }
  protected Object[] getMethodArgumentValues(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {
    Object[] args = new Object[parameters.length];
    for (int i = 0; i < parameters.length; i++) {
      // 解析參數
      args[i] = this.resolvers.resolveArgument(parameter, mavContainer, request, this.dataBinderFactory);
    }
  }
}

參數解析

在上面的Controller示例中,參數的解析器是RequestParamMethodArgumentResolver。

調用父類的resolveArgument方法。

public abstract class AbstractNamedValueMethodArgumentResolver {
  public final Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {
    // 封裝方法參數的名稱這里為:id
    NamedValueInfo namedValueInfo = getNamedValueInfo(parameter);
    // resolvedName = id
    Object resolvedName = resolveEmbeddedValuesAndExpressions(namedValueInfo.name);
    // ...
    // 獲取參數名對應的請求參數值:/demos/index?id=100 , 這就返回100
    Object arg = resolveName(resolvedName.toString(), nestedParameter, webRequest);
    // ...
    if (binderFactory != null) {
      // 根據當前的Request對象及請求參數名創建WebDataBinder對象
      // 內部創建的ExtendedServletRequestDataBinder對象
      WebDataBinder binder = binderFactory.createBinder(webRequest, null, namedValueInfo.name);
      try {
        // 執行類型轉換
        arg = binder.convertIfNecessary(arg, parameter.getParameterType(), parameter);
      }
    }
  }
}
// 創建WebDataBinder對象
public class DefaultDataBinderFactory implements WebDataBinderFactory {
  public final WebDataBinder createBinder(NativeWebRequest webRequest, @Nullable Object target, String objectName) throws Exception {
    WebDataBinder dataBinder = createBinderInstance(target, objectName, webRequest);
    if (this.initializer != null) {
      // 初始化WebDataBinder對象,這里最主要的就是為其設置類型轉換器
      this.initializer.initBinder(dataBinder, webRequest);
    }
    // 初始化執行@InitBinder注解的方法
    initBinder(dataBinder, webRequest);
    return dataBinder;
  }
}
public class InitBinderDataBinderFactory extends DefaultDataBinderFactory {
  public void initBinder(WebDataBinder dataBinder, NativeWebRequest request) throws Exception {
    // 遍歷所有@InitBinder注解的方法
    for (InvocableHandlerMethod binderMethod : this.binderMethods) {
      if (isBinderMethodApplicable(binderMethod, dataBinder)) {
        // 這里就是執行@InitBinder注解的方法
       Object returnValue = binderMethod.invokeForRequest(request, null, dataBinder);
        // 如果@InitBinder注解的方法有返回值則拋出異常
        if (returnValue != null) {
          throw new IllegalStateException("@InitBinder methods must not return a value (should be void): " + binderMethod);
        }
      }
    }
  }
}
// 解析@InitBinder注解方法的參數及方法執行
public class InvocableHandlerMethod extends HandlerMethod {
  public Object invokeForRequest(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {
    // 解析獲取@InitBinder注解方法的參數
    Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs);
    // 執行調用
    return doInvoke(args);
  }
}

執行類型轉換

在上面執行流程中,我們知道獲取了一個WebDataBinder對象和由@InitBinder 注解的方法的調用執行。接下來就是進行類型的轉換。

public abstract class AbstractNamedValueMethodArgumentResolver {
  public final Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {
    if (binderFactory != null) {
      // 根據當前的Request對象及請求參數名創建WebDataBinder對象
      // 內部創建的ExtendedServletRequestDataBinder對象
      WebDataBinder binder = binderFactory.createBinder(webRequest, null, namedValueInfo.name);
      try {
        // 執行類型轉換
        arg = binder.convertIfNecessary(arg, parameter.getParameterType(), parameter);
      }
    }
  }
}
// 最終通過該類調用類型轉換
class TypeConverterDelegate {
  public <T> T convertIfNecessary(@Nullable String propertyName, @Nullable Object oldValue, @Nullable Object newValue, @Nullable Class<T> requiredType, @Nullable TypeDescriptor typeDescriptor) throws IllegalArgumentException {
    // Custom editor for this type?
    // 獲取自定義的類型轉換器(首先獲取的就是我們上面自定義的)
    PropertyEditor editor = this.propertyEditorRegistry.findCustomEditor(requiredType, propertyName);
    // ...
    Object convertedValue = newValue;
    // ...
    convertedValue = doConvertValue(oldValue, convertedValue, requiredType, editor);
  }
  private Object doConvertValue(@Nullable Object oldValue, @Nullable Object newValue, @Nullable Class<?> requiredType, @Nullable PropertyEditor editor) { 
    // ...
    if (convertedValue instanceof String) {
      if (editor != null) {
        String newTextValue = (String) convertedValue;
        // 最終的調用
        return doConvertTextValue(oldValue, newTextValue, editor);
      } else if (String.class == requiredType) {
        returnValue = convertedValue;
      }
    }
    return returnValue;
  }
  // 最終得到了我們想要的值
  private Object doConvertTextValue(@Nullable Object oldValue, String newTextValue, PropertyEditor editor) {
    try {
      editor.setValue(oldValue);
    }
    // ...
    editor.setAsText(newTextValue);
    return editor.getValue();
  }
}

感謝各位的閱讀,以上就是“SpringMVC中@InitBinder注解怎么使用”的內容了,經過本文的學習后,相信大家對SpringMVC中@InitBinder注解怎么使用這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

依安县| 岑巩县| 和林格尔县| 南投市| 哈密市| 淮北市| 治多县| 泰宁县| 日土县| 古交市| 江永县| 奎屯市| 万荣县| 鄱阳县| 泸水县| 运城市| 扎鲁特旗| 炉霍县| 武威市| 肇州县| 延寿县| 兴城市| 集贤县| 孟村| 石柱| 金川县| 灯塔市| 新宾| 南陵县| 芦溪县| 二手房| 定安县| 新丰县| 丽水市| 华阴市| 宜春市| 英吉沙县| 武宣县| 通州市| 灵川县| 资讯|