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

簡要分析Java的Hibernate框架中的自定義類型

 更新時間:2016年01月03日 17:32:47   作者:cxshun  
這篇文章主要介紹了Java的Hibernate框架中的自定義類型,Hibernate是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下

最近看到hibernate的自定義類型,這個以前沒接觸過,在這里記錄一下,當(dāng)是對自己知識的鞏固,也讓沒有接觸過的朋友一起學(xué)習(xí)研究一番。
 1)自定義類型,顧名思義,當(dāng)然就是由于內(nèi)部的類型不滿足需求,而自己來進(jìn)行實現(xiàn)的類型。這種情況不多,但我們還是有必要學(xué)習(xí)一下,技多不壓身嘛。也學(xué)習(xí)一下,別人在做框架的時候是怎么去考慮的,怎么去思考擴(kuò)展性的。
 自定義類型有兩個方法來實現(xiàn),一種是實現(xiàn)UserType,另外一種實現(xiàn)CompositeUserType,另外可能還有一些方法,但我暫時沒用到,先不講了。
 我暫時只用到UserType,我們就先看一下UserType接口的定義:

public interface UserType { 
  /** 
   * Return the SQL type codes for the columns mapped by this type. The 
   * codes are defined on <tt>java.sql.Types</tt>. 
   */ 
  public int[] sqlTypes(); 
 
  /** 
   * The class returned by <tt>nullSafeGet()</tt>. 
   */ 
  public Class returnedClass(); 
 
  /** 
   * Compare two instances of the class mapped by this type for persistence "equality". 
   * Equality of the persistent state. 
   */ 
  public boolean equals(Object x, Object y) throws HibernateException; 
 
  /** 
   * Get a hashcode for the instance, consistent with persistence "equality" 
   */ 
  public int hashCode(Object x) throws HibernateException; 
 
  /** 
   * Retrieve an instance of the mapped class from a JDBC resultset. Implementors 
   * should handle possibility of null values. 
   */ 
  public Object nullSafeGet(ResultSet rs, String[] names, Object owner) throws HibernateException, SQLException; 
 
  /** 
   * Write an instance of the mapped class to a prepared statement. Implementors 
   * should handle possibility of null values. A multi-column type should be written 
   * to parameters starting from <tt>index</tt>. 
   */ 
  public void nullSafeSet(PreparedStatement st, Object value, int index) throws HibernateException, SQLException; 
 
  /** 
   * Return a deep copy of the persistent state, stopping at entities and at 
   * collections. It is not necessary to copy immutable objects, or null 
   * values, in which case it is safe to simply return the argument. 
   */ 
  public Object deepCopy(Object value) throws HibernateException; 
 
  /** 
   * Are objects of this type mutable? 
   * 
   * @return boolean 
   */ 
  public boolean isMutable(); 
 
  /** 
   * Transform the object into its cacheable representation. At the very least this 
   * method should perform a deep copy if the type is mutable. That may not be enough 
   * for some implementations, however; for example, associations must be cached as 
   * identifier values. (optional operation) 
   * 
   * @param value the object to be cached 
   * @return a cachable representation of the object 
   * @throws HibernateException 
   */ 
  public Serializable disassemble(Object value) throws HibernateException; 
 
  /** 
   * Reconstruct an object from the cacheable representation. At the very least this 
   * method should perform a deep copy if the type is mutable. (optional operation) 
   */ 
  public Object assemble(Serializable cached, Object owner) throws HibernateException; 
 
  /** 
   * During merge, replace the existing (target) value in the entity we are merging to 
   * with a new (original) value from the detached entity we are merging. For immutable 
   * objects, or null values, it is safe to simply return the first parameter. For 
   * mutable objects, it is safe to return a copy of the first parameter. For objects 
   * with component values, it might make sense to recursively replace component values. 
   */ 
  public Object replace(Object original, Object target, Object owner) throws HibernateException; 
} 

  其實大家看英文一般情況下都能理解,不再多做解釋了,這里我們最主要的就是實現(xiàn)nullSafeSet() 方法,這個方法主要用到把此類型的值保存到數(shù)據(jù)庫,這一次我們先學(xué)怎么用,以后我們再慢慢研究內(nèi)部是怎么來實現(xiàn)的。
 2)我學(xué)習(xí)時寫的例子是參照夏昕的例子,所以肯定和網(wǎng)上的大部分都一樣,我們只是大概分析一下:
 下面是User類

package org.hibernate.tutorial.domain; 
import java.io.Serializable; 
import java.util.List; 
public class User implements Serializable{ 
  public Long id; 
  private String name; 
  private List emails; 
    省略Get/Set方法 
} 

  下來是自定義的EmailList類:

package org.hibernate.tutorial.domain; 
import java.io.Serializable; 
import java.sql.PreparedStatement; 
import java.sql.ResultSet; 
import java.sql.SQLException; 
import java.sql.Types; 
import java.util.ArrayList; 
import java.util.List; 
import org.hibernate.Hibernate; 
import org.hibernate.HibernateException; 
import org.hibernate.usertype.UserType; 
 
public class EmailList implements UserType { 
 
  private static final char SPLITTER = ';'; 
  private static final int[] TYPES = new int[] {Types.VARCHAR}; 
   
  private String assemble(List emailList) { 
    StringBuilder strBuf = new StringBuilder(); 
    for (int i = 0; i < emailList.size() - 1; i++){ 
      strBuf.append(emailList.get(i)).append(SPLITTER); 
    } 
    strBuf.append(emailList.get(emailList.size()-1)); 
    return strBuf.toString(); 
  } 
   
  private List parse(String value) { 
    String[] strs = org.hibernate.util.StringHelper.split(value,String.valueOf(SPLITTER)); 
    List emailList = new ArrayList(); 
    for (int i = 0;i < strs.length; i++) { 
      emailList.add(strs[i]); 
    } 
    return emailList; 
  } 
 
  public Object deepCopy(Object value) throws HibernateException { 
    List sourceList = (List)value; 
    List targetList = new ArrayList(); 
    targetList.add(sourceList); 
    return targetList; 
  } 
 
  public Serializable disassemble(Object value) throws HibernateException { 
    return null; 
  } 
 
  public boolean equals(Object x, Object y) throws HibernateException { 
    if (x == y) return true; 
     
    System.out.println("X:"+x+"Y:"+y); 
     
    if (x != null && y != null) { 
      List xList = (List)x; 
      List yList = (List)y; 
       
      if(xList.size() != yList.size()) return false; 
       
      for (int i = 0; i < xList.size(); i++) { 
        String str1 = (String)xList.get(i); 
        String str2 = (String)yList.get(i); 
         
        if (!str1.equals(str2)) return false; 
      } 
       
      return true; 
    } 
     
    return false; 
  } 
 
  public boolean isMutable() { 
    return false; 
  } 
 
  public Object nullSafeGet(ResultSet rs, String[] names, Object owner) 
      throws HibernateException, SQLException { 
    String value = (String)Hibernate.STRING.nullSafeGet(rs, names[0]); 
    if (value != null) { 
      return parse(value);//把List通過;分割 
    } else{ 
      return null; 
    } 
  } 
 
  public void nullSafeSet(PreparedStatement st, Object value, int index) 
      throws HibernateException, SQLException { 
    System.out.println("Set Method Executed!"); 
     
    System.out.println("value:" + value); 
     
    if (value != null){ 
      String str = assemble((List)value);//把字符串用;拼接 
       
      Hibernate.STRING.nullSafeSet(st, str, index); 
    } else { 
      Hibernate.STRING.nullSafeSet(st, value, index); 
    } 
  } 
 
  public Class returnedClass() { 
    return List.class; 
  } 
 
  public int[] sqlTypes() { 
    return TYPES; 
  } 
  //省略其他不需要修改的方法 
} 

  類中實現(xiàn)的方法是需要修改的方法,其他不需要修改暫時不用的方法則沒有寫出來,但還是需要實現(xiàn)的。
 3)接下來就是User類的映射文件:

<class name="User" table="USER"> 
    <id name="id" column="USER_ID" type="java.lang.Long"> 
      <generator class="native" /> 
    </id> 
    <property name="name" type="string" column="USER_NAME"/> 
    <property name="emails" type="org.hibernate.tutorial.domain.EmailList" column="emails"/> 
  </class> 

  相信大家都知道怎么進(jìn)行修改,這里也不進(jìn)行講解了,主要是修改emails的type,修改為我們剛才定義的EmailList類。
 4)最后我們來寫一個測試類:

import java.util.HashMap; 
import java.util.List; 
import java.util.Map; 
import java.util.ArrayList; 
import junit.framework.TestCase; 
 
import org.hibernate.EntityMode; 
import org.hibernate.Session; 
import org.hibernate.SessionFactory; 
import org.hibernate.Transaction; 
import org.hibernate.cfg.Configuration; 
import org.hibernate.tutorial.domain.User; 
 
public class HibernateTest extends TestCase{ 
 
  private Session session = null; 
   
  protected void setUp() throws Exception { 
     
    Configuration cfg = new Configuration().configure(); 
    SessionFactory sessionFactory = cfg.buildSessionFactory(); 
     
    session = sessionFactory.openSession(); 
 
  } 
   
  public void testInsert(){ 
    Transaction tran = null; 
    try{ 
      tran = session.beginTransaction(); 
       
      User user = new User(); 
       
      user.setName("shun"); 
       
      List list = new ArrayList(); 
       
      list.add("12312@sfsdf.com"); 
      list.add("123@123.com"); 
 
      user.setEmails(list); 
      session.save(user); 
       
      tran.commit(); 
    } catch (Exception ex) { 
      ex.printStackTrace(); 
      if (tran != null){ 
        tran.rollback(); 
      } 
    } 
  } 
   
  protected void tearDown() throws Exception { 
    session.close(); 
  } 
} 

  這里可能會出現(xiàn)問題,當(dāng)我們只保存一個email時,它會出現(xiàn)異常,在數(shù)據(jù)庫里面是email字段是空的,而當(dāng)我們?nèi)缟厦娲a一樣,有兩個時,并不會出現(xiàn)問題,數(shù)據(jù)庫中結(jié)果如圖:

201613173332381.png (535×20)

而當(dāng)我們只保存一個時,異常如下:

java.lang.ClassCastException: java.util.ArrayList cannot be cast to java.lang.String 

 它發(fā)生在EmailList的equals方法中的String str1 = (String)xList.get(i);這句代碼中,經(jīng)檢查是在插入數(shù)據(jù)傳到EmailList的nullSafeSet方法時變成了List的List,即
value:[[12312@sfsdf.com, 123@123.com]]這樣的形式,這樣在比較的時候就會出問題,它永遠(yuǎn)都只有一個值,而在比較的時候卻是不同的,

if(xList.size() != yList.size()) return false; 

 所以在強(qiáng)制轉(zhuǎn)換時會出問題。
 而經(jīng)過檢查,equals方法里:
 

X:[[12312@sfsdf.com, 123@123.com]]Y:[12312@sfsdf.com, 123@123.com] 

 這樣的結(jié)果卻是很奇怪的。網(wǎng)上并沒有講到為什么會出現(xiàn)這種情況。這里提出一下:我用的hibernate版本是Hibernate 3.3.2.GA。不知道是版本問題還是其他問題,我們明天再研究一下。如果有哪位兄弟知道為什么的,希望也不吝告訴我一下。

相關(guān)文章

  • SpringSecurity整合springBoot、redis實現(xiàn)登錄互踢功能

    SpringSecurity整合springBoot、redis實現(xiàn)登錄互踢功能

    這篇文章主要介紹了SpringSecurity整合springBoot、redis實現(xiàn)登錄互踢,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-05-05
  • Java notify和notifyAll的區(qū)別和相同

    Java notify和notifyAll的區(qū)別和相同

    本文主要介紹Java notify和notifyAll的知識,這里整理詳細(xì)的資料來說明notify 和NotifAll的區(qū)別,有需要的小伙伴可以參考下
    2016-09-09
  • spring級聯(lián)屬性賦值的兩種方式解析

    spring級聯(lián)屬性賦值的兩種方式解析

    這篇文章主要介紹了spring級聯(lián)屬性賦值的兩種方式解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-01-01
  • freemarker簡介_動力節(jié)點Java學(xué)院整理

    freemarker簡介_動力節(jié)點Java學(xué)院整理

    FreeMarker是一個模板引擎,一個基于模板生成文本輸出的通用工具,使用純Java編寫,有興趣的可以了解一下
    2017-08-08
  • SpringSecurity導(dǎo)致SpringBoot跨域失效的問題解決

    SpringSecurity導(dǎo)致SpringBoot跨域失效的問題解決

    本文主要介紹了SpringSecurity導(dǎo)致SpringBoot跨域失效的問題解決,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-01-01
  • MyBatis-Plus Generator配置詳解

    MyBatis-Plus Generator配置詳解

    這篇文章主要介紹了MyBatis-Plus Generator配置詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-08-08
  • Eclipse不自動編譯java文件的終極解決方法

    Eclipse不自動編譯java文件的終極解決方法

    這篇文章主要介紹了Eclipse不自動編譯java文件的終極解決方法,需要的朋友可以參考下
    2015-12-12
  • Java開發(fā)崗位面試被問到泛型怎么辦

    Java開發(fā)崗位面試被問到泛型怎么辦

    泛型在java中有很重要的地位,在面向?qū)ο缶幊碳案鞣N設(shè)計模式中有非常廣泛的應(yīng)用。java泛型知識點也是Java開發(fā)崗位必問的一個話題,今天小編就給大家普及下Java泛型常見面試題,感興趣的朋友一起看看吧
    2021-07-07
  • SpringBoot實現(xiàn)ImportBeanDefinitionRegistrar動態(tài)注入

    SpringBoot實現(xiàn)ImportBeanDefinitionRegistrar動態(tài)注入

    在閱讀Spring Boot源碼時,看到Spring Boot中大量使用ImportBeanDefinitionRegistrar來實現(xiàn)Bean的動態(tài)注入,它是Spring中一個強(qiáng)大的擴(kuò)展接口,本文就來詳細(xì)的介紹一下如何使用,感興趣的可以了解一下
    2024-02-02
  • Maven中jar包沖突原理與解決辦法

    Maven中jar包沖突原理與解決辦法

    這篇文章主要介紹了Maven中jar包沖突原理與解決辦法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-09-09

最新評論