欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Mybatis中TypeAliasRegistry的作用及使用方法

 更新時(shí)間:2023年05月08日 09:21:21   作者:假正經(jīng)的小柴  
Mybatis中的TypeAliasRegistry是一個(gè)類型別名注冊(cè)表,它的作用是為Java類型建立別名,使得在Mybatis配置文件中可以使用別名來代替完整的Java類型名。使用TypeAliasRegistry可以簡(jiǎn)化Mybatis配置文件的編寫,提高配置文件的可讀性和可維護(hù)性

一、引入類型別名

當(dāng)配置 XML 文件,需要指明Java類型時(shí),類型別名可替代Java類型的全名,一般會(huì)設(shè)置一個(gè)簡(jiǎn)單縮寫的類型別名去替代它,用于XML配置,以降低冗余的全限定類名書寫。(說白了其作用就是偷懶)

下面是使用全限定類名指定的配置:

    <select id="selectAll" resultType="com.ncpowernode.mybatis.bean.User">
        select * from t_user;
    </select>

當(dāng)在核心配置文件中配置完后:

<typeAliases>
<! --<package name="com.ncpowernode .mybatis.bean" />-->
<typeAlias type="com.ncpowernode .mybatis.bean. User"/>
</typeAliases>

即可寫成下面的形式:

<select id="selectAll" resultType="user">
        select * from t_user;
</select>

二、typeAlias的三種配置方式

type屬性和alias屬性雙搭

<typeAlias type="com.ncpowernode.mybatis.bean.User" alias="user"/>

直接寫type屬性指定全限定類名,底層會(huì)利用該類的simpleName去當(dāng)作這個(gè)alias。

<typeAlias type="com.ncpowernode.mybatis.bean.User"/>

直接配置包標(biāo)簽,使得指定包下的所有類都進(jìn)行別名配置。

<package name="com.ncpowernode.mybatis.bean"/>

三、TypeAliasRegistry源碼分析

三種配置方式源碼解析

Mybatis通過TypeAliasRegistry對(duì)象實(shí)現(xiàn)對(duì)別名的封裝,實(shí)現(xiàn)別名對(duì)應(yīng)Java類型的校驗(yàn)。TypeAliasRegistry類中是用一Map成員對(duì)象實(shí)現(xiàn)上面封裝效果的。

public class TypeAliasRegistry {
	// key對(duì)應(yīng)著別名,Class對(duì)應(yīng)著全限定類名轉(zhuǎn)換的Class對(duì)象
	private final Map<String, Class<?>> typeAliases = new HashMap<>();
}

解析核心配置文件時(shí)候支持上面提到的三種設(shè)置別名的方式,那自然底層實(shí)現(xiàn)也存在三種,但萬變不理其中,最后跳轉(zhuǎn)到的代碼(核心方法)如下所示:

  public void registerAlias(String alias, Class<?> value) {
  	//判斷別名是否為空
    if (alias == null) {
      throw new TypeException("The parameter alias cannot be null");
    }
		// 將別名都轉(zhuǎn)換為小寫存儲(chǔ),利于后續(xù)進(jìn)行校驗(yàn)
    String key = alias.toLowerCase(Locale.ENGLISH);
    // 從這段代碼可以知道別名可以支持多個(gè)對(duì)應(yīng)一個(gè)Class對(duì)象
    if (typeAliases.containsKey(key) && typeAliases.get(key) != null && !typeAliases.get(key).equals(value)) {
      throw new TypeException(
          "The alias '" + alias + "' is already mapped to the value '" + typeAliases.get(key).getName() + "'.");
    }
    // 封裝到Map中
    typeAliases.put(key, value);
  }

下面對(duì)三種方式源碼進(jìn)行解析

第一種:type和alias屬性雙用(直接跳轉(zhuǎn)到上面核心方法)

  public void registerAlias(String alias, String value) {
      registerAlias(alias, Resources.classForName(value));
  }

第二種:僅用type屬性(熟知的簡(jiǎn)單類名,為什么是簡(jiǎn)單類名當(dāng)作別名下面的代碼很好的體現(xiàn)出來了,還有一種設(shè)置別名的方式是使用@Alias注解,但小編本人不喜歡用)

  public void registerAlias(Class<?> type) {
  	// 獲取類中的簡(jiǎn)單類名稱,充當(dāng)別名
    String alias = type.getSimpleName();
    // 如果在對(duì)應(yīng)類上用了@Alias注解的話,那對(duì)應(yīng)的是@Alias注解中的value屬性值
    // 但小編不建議用,出錯(cuò)了感覺不好排查
    Alias aliasAnnotation = type.getAnnotation(Alias.class);
    if (aliasAnnotation != null) {
      alias = aliasAnnotation.value();
    }
    // 進(jìn)行封裝
    registerAlias(alias, type);
  }

第三種:使用package標(biāo)簽(常用)

  public void registerAliases(String packageName) {
    registerAliases(packageName, Object.class);
  }
  public void registerAliases(String packageName, Class<?> superType) {
    ResolverUtil<Class<?>> resolverUtil = new ResolverUtil<>();
    // 獲取包下的類Class對(duì)象,僅該目錄下的,子目錄下的類對(duì)象不包括
    resolverUtil.find(new ResolverUtil.IsA(superType), packageName);
    Set<Class<? extends Class<?>>> typeSet = resolverUtil.getClasses();
    for (Class<?> type : typeSet) {
      // 對(duì)應(yīng)的類不能是匿名類/接口
      if (!type.isAnonymousClass() && !type.isInterface() && !type.isMemberClass()) {
      // 本質(zhì)是使用第二種方式
        registerAlias(type);
      }
    }
  }

校驗(yàn)過程

通過閱讀下面源碼可以知道:

  • 為什么使用別名的時(shí)候各個(gè)字符不區(qū)分大小寫;
  • 為什么使用別名也行,使用全限定類名也行。
  public <T> Class<T> resolveAlias(String string) {
    try {
     // 判斷傳來的別名(也可能是全限定類名)不為空
      if (string == null) {
        return null;
      }
      // 解析的時(shí)候都會(huì)轉(zhuǎn)換成小寫
      // 這就是為什么使用別名的時(shí)候別名字母小寫大寫無所謂
      String key = string.toLowerCase(Locale.ENGLISH);
      Class<T> value;
      if (typeAliases.containsKey(key)) {// 通過別名查找對(duì)應(yīng)的value(Class對(duì)象)
        value = (Class<T>) typeAliases.get(key);
      } else {// 無別名就當(dāng)作全限定類名處理,不存在就會(huì)拋出下面的異常
        value = (Class<T>) Resources.classForName(string);
      }
      // 返回結(jié)果Class對(duì)象
      return value;
    } catch (ClassNotFoundException e) {
      throw new TypeException("Could not resolve type alias '" + string + "'.  Cause: " + e, e);
    }
  }

Mybatis默認(rèn)的別名配置

Mybatis 在初始化Configuration對(duì)象的時(shí)候自身配置了一些Java類型的類型別名。如下所示:

這是Configuration類中TypeAliasRegistry成員初始化創(chuàng)建對(duì)象時(shí)候進(jìn)行的配置

// TypeAliasRegistry類構(gòu)造器初始化
  public TypeAliasRegistry() {
    registerAlias("string", String.class);
    registerAlias("byte", Byte.class);
    registerAlias("char", Character.class);
    registerAlias("character", Character.class);
    registerAlias("long", Long.class);
    registerAlias("short", Short.class);
    registerAlias("int", Integer.class);
    registerAlias("integer", Integer.class);
    registerAlias("double", Double.class);
    registerAlias("float", Float.class);
    registerAlias("boolean", Boolean.class);
    registerAlias("byte[]", Byte[].class);
    registerAlias("char[]", Character[].class);
    registerAlias("character[]", Character[].class);
    registerAlias("long[]", Long[].class);
    registerAlias("short[]", Short[].class);
    registerAlias("int[]", Integer[].class);
    registerAlias("integer[]", Integer[].class);
    registerAlias("double[]", Double[].class);
    registerAlias("float[]", Float[].class);
    registerAlias("boolean[]", Boolean[].class);
    registerAlias("_byte", byte.class);
    registerAlias("_char", char.class);
    registerAlias("_character", char.class);
    registerAlias("_long", long.class);
    registerAlias("_short", short.class);
    registerAlias("_int", int.class);
    registerAlias("_integer", int.class);
    registerAlias("_double", double.class);
    registerAlias("_float", float.class);
    registerAlias("_boolean", boolean.class);
    registerAlias("_byte[]", byte[].class);
    registerAlias("_char[]", char[].class);
    registerAlias("_character[]", char[].class);
    registerAlias("_long[]", long[].class);
    registerAlias("_short[]", short[].class);
    registerAlias("_int[]", int[].class);
    registerAlias("_integer[]", int[].class);
    registerAlias("_double[]", double[].class);
    registerAlias("_float[]", float[].class);
    registerAlias("_boolean[]", boolean[].class);
    registerAlias("date", Date.class);
    registerAlias("decimal", BigDecimal.class);
    registerAlias("bigdecimal", BigDecimal.class);
    registerAlias("biginteger", BigInteger.class);
    registerAlias("object", Object.class);
    registerAlias("date[]", Date[].class);
    registerAlias("decimal[]", BigDecimal[].class);
    registerAlias("bigdecimal[]", BigDecimal[].class);
    registerAlias("biginteger[]", BigInteger[].class);
    registerAlias("object[]", Object[].class);
    registerAlias("map", Map.class);
    registerAlias("hashmap", HashMap.class);
    registerAlias("list", List.class);
    registerAlias("arraylist", ArrayList.class);
    registerAlias("collection", Collection.class);
    registerAlias("iterator", Iterator.class);
    registerAlias("ResultSet", ResultSet.class);
  }

這是Configuration對(duì)象創(chuàng)建時(shí)候的別名配置

  public Configuration() {
    typeAliasRegistry.registerAlias("JDBC", JdbcTransactionFactory.class);
    typeAliasRegistry.registerAlias("MANAGED", ManagedTransactionFactory.class);
    typeAliasRegistry.registerAlias("JNDI", JndiDataSourceFactory.class);
    typeAliasRegistry.registerAlias("POOLED", PooledDataSourceFactory.class);
    typeAliasRegistry.registerAlias("UNPOOLED", UnpooledDataSourceFactory.class);
    typeAliasRegistry.registerAlias("PERPETUAL", PerpetualCache.class);
    typeAliasRegistry.registerAlias("FIFO", FifoCache.class);
    typeAliasRegistry.registerAlias("LRU", LruCache.class);
    typeAliasRegistry.registerAlias("SOFT", SoftCache.class);
    typeAliasRegistry.registerAlias("WEAK", WeakCache.class);
    typeAliasRegistry.registerAlias("DB_VENDOR", VendorDatabaseIdProvider.class);
    typeAliasRegistry.registerAlias("XML", XMLLanguageDriver.class);
    typeAliasRegistry.registerAlias("RAW", RawLanguageDriver.class);
    typeAliasRegistry.registerAlias("SLF4J", Slf4jImpl.class);
    typeAliasRegistry.registerAlias("COMMONS_LOGGING", JakartaCommonsLoggingImpl.class);
    typeAliasRegistry.registerAlias("LOG4J", Log4jImpl.class);
    typeAliasRegistry.registerAlias("LOG4J2", Log4j2Impl.class);
    typeAliasRegistry.registerAlias("JDK_LOGGING", Jdk14LoggingImpl.class);
    typeAliasRegistry.registerAlias("STDOUT_LOGGING", StdOutImpl.class);
    typeAliasRegistry.registerAlias("NO_LOGGING", NoLoggingImpl.class);
    typeAliasRegistry.registerAlias("CGLIB", CglibProxyFactory.class);
    typeAliasRegistry.registerAlias("JAVASSIST", JavassistProxyFactory.class);
  }

這些都是Mybatis框架會(huì)配置好的,你可以直接使用。

四、總結(jié)

TypeAliasRegistry 類完成的別名機(jī)制,總的來說源碼還是比較簡(jiǎn)單的,一個(gè)Map對(duì)象封裝起來的就完成了。解析核心配置文件擴(kuò)大別名使用,有懶就偷。

到此這篇關(guān)于Mybatis中TypeAliasRegistry的作用及使用方法的文章就介紹到這了,更多相關(guān)Mybatis TypeAliasRegistry內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java調(diào)用微信客服消息實(shí)現(xiàn)發(fā)貨通知的方法詳解

    Java調(diào)用微信客服消息實(shí)現(xiàn)發(fā)貨通知的方法詳解

    這篇文章主要介紹了Java調(diào)用微信客服消息實(shí)現(xiàn)發(fā)貨通知的方法,結(jié)合實(shí)例形式詳細(xì)分析了java針對(duì)微信接口調(diào)用的原理、調(diào)用方法與相關(guān)注意事項(xiàng),需要的朋友可以參考下
    2017-08-08
  • Java修改Integer變量值遇到的問題及解決

    Java修改Integer變量值遇到的問題及解決

    這篇文章主要介紹了Java修改Integer變量值遇到的問題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • Java遺傳算法之沖出迷宮

    Java遺傳算法之沖出迷宮

    這篇文章首先詳細(xì)介紹了什么是遺傳算法,然后通過遺傳算法的思想用實(shí)例解析使用遺傳算法解決迷宮問題,需要的朋友可以參考下
    2017-09-09
  • WMTS中TileMatrix與ScaleDenominator淺析

    WMTS中TileMatrix與ScaleDenominator淺析

    這篇文章主要為大家介紹了WMTS中TileMatrix與ScaleDenominator淺析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03
  • 利用棧使用簡(jiǎn)易計(jì)算器(Java實(shí)現(xiàn))

    利用棧使用簡(jiǎn)易計(jì)算器(Java實(shí)現(xiàn))

    這篇文章主要為大家詳細(xì)介紹了Java利用棧實(shí)現(xiàn)簡(jiǎn)易計(jì)算器,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-09-09
  • Java面試必備之ArrayList陷阱解析

    Java面試必備之ArrayList陷阱解析

    昨天小楓接到了一個(gè)公司的面試電話,其中一道面試題覺得有點(diǎn)意思,在這里和大家一起分享下。面試題是ArrayList如何刪除指定元素。乍聽很簡(jiǎn)單的問題,但是如果沒有實(shí)際踩過坑很容易掉進(jìn)面試官的陷阱中,我們一起來分析下吧
    2022-02-02
  • 使Java的JButton文字隱藏功能的實(shí)現(xiàn)(不隱藏按鈕的前提)

    使Java的JButton文字隱藏功能的實(shí)現(xiàn)(不隱藏按鈕的前提)

    這篇文章主要介紹了使Java的JButton文字隱藏功能的實(shí)現(xiàn)(不隱藏按鈕的前提),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-01-01
  • Java多線程之線程安全問題詳情

    Java多線程之線程安全問題詳情

    這篇文章主要介紹了Java多線程之線程安全問題詳情,線程安全問題是指因多線程搶占式執(zhí)行而導(dǎo)致程序出現(xiàn)bug的問題。內(nèi)容介紹詳細(xì)內(nèi)容需要的小伙伴可以參考下面文章內(nèi)容
    2022-06-06
  • 詳解Java中格式化日期的DateFormat與SimpleDateFormat類

    詳解Java中格式化日期的DateFormat與SimpleDateFormat類

    DateFormat其本身是一個(gè)抽象類,SimpleDateFormat 類是DateFormat類的子類,一般情況下來講DateFormat類很少會(huì)直接使用,而都使用SimpleDateFormat類完成,下面我們具體來看一下兩個(gè)類的用法:
    2016-05-05
  • 詳細(xì)分析JVM類加載機(jī)制

    詳細(xì)分析JVM類加載機(jī)制

    JVM將class文件字節(jié)碼文件加載到內(nèi)存中,?并將這些靜態(tài)數(shù)據(jù)轉(zhuǎn)換成方法區(qū)中的運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu),在堆(并不一定在堆中,HotSpot在方法區(qū)中)中生成一個(gè)代表這個(gè)類的java.lang.Class?對(duì)象,作為方法區(qū)類數(shù)據(jù)的訪問入口,接下來將詳細(xì)講解JVM類加載機(jī)制
    2022-04-04

最新評(píng)論