<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    菜園子

    BlogJava 首頁 新隨筆 聯系 聚合 管理
      7 Posts :: 1 Stories :: 31 Comments :: 0 Trackbacks

    關于Hibernate的查詢從數據庫映射到JavaBean

     


     

    Hibernate除了HQL外,還支持SQL的查詢,API為createSQLQuery(sql),如果數據庫使用的是Oracle,
    由于數據庫表中的列都是大寫,所以在從resultset到javabean的時候,需要完全匹配。

    一般我們會用DTO或者作為DTO的Entity,無論是采用addEntity(Class class)還是

    setResultTransformer(new AliasToBeanResultTransformer (CatDTO.class))

    都會遇到數據庫字段到Java的大小寫映射的問題,如果數據庫字段是小寫的id, 數據庫里面是大寫的ID,
    則會遇到
    org.hibernate.PropertyNotFoundException: Could not find setter for ID on class com....的問題。

    通過源碼發現,要求java的屬性要和數據庫的字段名大小寫一樣,并且全匹配。

     

    這個可以通過模仿Spring的類似查詢解決。Spring的NamedJdbcTemplate有如下方法:namedJdbcTemplate.query(sql, params, new BeanPropertyRowMapper(clazz))

     

    也是通過執行sql并把返回的結果轉換成Java的方法,這個就可以忽視數據庫字段的大小寫問題,
    仿照這個BeanPropertyRowMapper寫一個適用于Hibernate的Transformer。

    代碼如下:

    public class BeanTransformerAdapter<Timplements ResultTransformer {

     

        /** Logger available to subclasses */

        protected final Log                     logger              = LogFactory.getLog(getClass());

     

        /** The class we are mapping to */

        private Class<T>                        mappedClass;

     

        /** Whether we're strictly validating */

        private boolean                         checkFullyPopulated             = false;

     

        /** Whether we're defaulting primitives when mapping a null value */

        private boolean                         primitivesDefaultedForNullValue = false;

     

        /** Map of the fields we provide mapping for */

        private Map<String, PropertyDescriptor> mappedFields;

     

        /** Set of bean properties we provide mapping for */

        private Set<String>                     mappedProperties;

     

        /**

         * Create a new BeanPropertyRowMapper for bean-style configuration.

         * @see #setMappedClass

         * @see #setCheckFullyPopulated

         */

        public BeanTransformerAdapter() {

        }

     

        /**

         * Create a new BeanPropertyRowMapper, accepting unpopulated properties

         * in the target bean.

         * <p>Consider using the {@link #newInstance} factory method instead,

         * which allows for specifying the mapped type once only.

         * @param mappedClass the class that each row should be mapped to

         */

        public BeanTransformerAdapter(Class<T> mappedClass) {

            initialize(mappedClass);

        }

     

        /**

         * Create a new BeanPropertyRowMapper.

         * @param mappedClass the class that each row should be mapped to

         * @param checkFullyPopulated whether we're strictly validating that

         * all bean properties have been mapped from corresponding database fields

         */

        public BeanTransformerAdapter(Class<T> mappedClassboolean checkFullyPopulated) {

            initialize(mappedClass);

            this.checkFullyPopulated = checkFullyPopulated;

        }

     

        /**

         * Set the class that each row should be mapped to.

         */

        public void setMappedClass(Class<T> mappedClass) {

            if (this.mappedClass == null) {

                initialize(mappedClass);

            } else {

                if (!this.mappedClass.equals(mappedClass)) {

                    throw new InvalidDataAccessApiUsageException("The mapped class can not be reassigned to map to " 
    mappedClass + " since it is already providing mapping for " + this.mappedClass);

                }

            }

        }

     

        /**

         * Initialize the mapping metadata for the given class.

         * @param mappedClass the mapped class.

         */

        protected void initialize(Class<T> mappedClass) {

            this.mappedClass = mappedClass;

            this.mappedFields = new HashMap<String, PropertyDescriptor>();

            this.mappedProperties = new HashSet<String>();

            PropertyDescriptor[] pds = BeanUtils.getPropertyDescriptors(mappedClass);

            for (PropertyDescriptor pd : pds) {

                if (pd.getWriteMethod() != null) {

                    this.mappedFields.put(pd.getName().toLowerCase(), pd);

                    String underscoredName = underscoreName(pd.getName());

                    if (!pd.getName().toLowerCase().equals(underscoredName)) {

                        this.mappedFields.put(underscoredNamepd);

                    }

                    this.mappedProperties.add(pd.getName());

                }

            }

        }

     

        /**

         * Convert a name in camelCase to an underscored name in lower case.

         * Any upper case letters are converted to lower case with a preceding underscore.

         * @param name the string containing original name

         * @return the converted name

         */

        private String underscoreName(String name) {

            if (!StringUtils.hasLength(name)) {

                return "";

            }

            StringBuilder result = new StringBuilder();

            result.append(name.substring(0, 1).toLowerCase());

            for (int i = 1; i < name.length(); i++) {

                String s = name.substring(ii + 1);

                String slc = s.toLowerCase();

                if (!s.equals(slc)) {

                    result.append("_").append(slc);

                } else {

                    result.append(s);

                }

            }

            return result.toString();

        }

     

        /**

         * Get the class that we are mapping to.

         */

        public final Class<T> getMappedClass() {

            return this.mappedClass;

        }

     

        /**

         * Set whether we're strictly validating that all bean properties have been

         * mapped from corresponding database fields.

         * <p>Default is {@code false}, accepting unpopulated properties in the

         * target bean.

         */

        public void setCheckFullyPopulated(boolean checkFullyPopulated) {

            this.checkFullyPopulated = checkFullyPopulated;

        }

     

        /**

         * Return whether we're strictly validating that all bean properties have been

         * mapped from corresponding database fields.

         */

        public boolean isCheckFullyPopulated() {

            return this.checkFullyPopulated;

        }

     

        /**

         * Set whether we're defaulting Java primitives in the case of mapping a null value

         * from corresponding database fields.

         * <p>Default is {@code false}, throwing an exception when nulls are mapped to Java primitives.

         */

        public void setPrimitivesDefaultedForNullValue(boolean primitivesDefaultedForNullValue) {

            this.primitivesDefaultedForNullValue = primitivesDefaultedForNullValue;

        }

     

        /**

         * Return whether we're defaulting Java primitives in the case of mapping a null value

         * from corresponding database fields.

         */

        public boolean isPrimitivesDefaultedForNullValue() {

            return primitivesDefaultedForNullValue;

        }

     

        /**

         * Initialize the given BeanWrapper to be used for row mapping.

         * To be called for each row.

         * <p>The default implementation is empty. Can be overridden in subclasses.

         * @param bw the BeanWrapper to initialize

         */

        protected void initBeanWrapper(BeanWrapper bw) {

        }

     

        /**

         * Retrieve a JDBC object value for the specified column.

         * <p>The default implementation calls

         * {@link JdbcUtils#getResultSetValue(java.sql.ResultSet, int, Class)}.

         * Subclasses may override this to check specific value types upfront,

         * or to post-process values return from {@code getResultSetValue}.

         * @param rs is the ResultSet holding the data

         * @param index is the column index

         * @param pd the bean property that each result object is expected to match

         * (or {@code null} if none specified)

         * @return the Object value

         * @throws SQLException in case of extraction failure

         * @see org.springframework.jdbc.support.JdbcUtils#getResultSetValue(java.sql.ResultSet, int, Class)

         */

        protected Object getColumnValue(ResultSet rsint index, PropertyDescriptor pdthrows SQLException {

            return JdbcUtils.getResultSetValue(rsindexpd.getPropertyType());

        }

     

        /**

         * Static factory method to create a new BeanPropertyRowMapper

         * (with the mapped class specified only once).

         * @param mappedClass the class that each row should be mapped to

         */

        public static <T> BeanPropertyRowMapper<T> newInstance(Class<T> mappedClass) {

            BeanPropertyRowMapper<T> newInstance = new BeanPropertyRowMapper<T>();

            newInstance.setMappedClass(mappedClass);

            return newInstance;

        }

     

        @Override

        public Object transformTuple(Object[] tuple, String[] aliases) {

            T mappedObject = BeanUtils.instantiate(this.mappedClass);

            BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(mappedObject);

            initBeanWrapper(bw);

     

            Set<String> populatedProperties = (isCheckFullyPopulated() ? new HashSet<String>() : null);

            for (int i = 0; i < aliases.lengthi++) {

                String column = aliases[i];

                PropertyDescriptor pd = this.mappedFields.get(column.replaceAll(" """).toLowerCase());

                if (pd != null) {

                    try {

                        Object value = tuple[i];

                        try {

                            bw.setPropertyValue(pd.getName(), value);

                        } catch (TypeMismatchException e) {

                            if (value == null && primitivesDefaultedForNullValue) {

                                logger.debug("Intercepted TypeMismatchException for column " + column + " and column '"
     + column + "' with value " + value + " when setting property '" + pd.getName() + "' of type " + pd.getPropertyType() 
    " on object: " + mappedObject);

                            } else {

                                throw e;

                            }

                        }

                        if (populatedProperties != null) {

                            populatedProperties.add(pd.getName());

                        }

                    } catch (NotWritablePropertyException ex) {

                        throw new DataRetrievalFailureException("Unable to map column " + column 
    " to property " + pd.getName(), ex);

                    }

                }

            }

     

            if (populatedProperties != null && !populatedProperties.equals(this.mappedProperties)) {

                throw new InvalidDataAccessApiUsageException("Given ResultSet does not contain all fields "
     + "necessary to populate object of class [" + this.mappedClass + "]: " + this.mappedProperties);

            }

     

            return mappedObject;

        }

     

        @Override

        public List transformList(List list) {

            return list;

        }

     

    使用方式如下:

    Query query = getSession().createSQLQuery(sql).setResultTransformer(new BeanTransformerAdapter(entityClass));

     

    就可以不用管Oracle字段的大寫問題了,會匹配到java的對應字段。

     



    QQ:24889356
    posted on 2014-08-27 15:08 GhostZhang 閱讀(10127) 評論(2)  編輯  收藏

    Feedback

    # re: 關于Hibernate的查詢從數據庫映射到JavaBean 2014-08-28 13:36 簡歷網
    謝謝博主的分享  回復  更多評論
      

    # re: 關于Hibernate的查詢從數據庫映射到JavaBean 2014-09-02 09:23 金利鎖業
    支持博主分享  回復  更多評論
      


    只有注冊用戶登錄后才能發表評論。


    網站導航:
     
    主站蜘蛛池模板: 亚洲AV无码无限在线观看不卡 | 亚洲欧美日韩久久精品| 免费a级毛片网站| 亚洲精品456人成在线| 日韩国产免费一区二区三区| 亚洲av福利无码无一区二区| 今天免费中文字幕视频| 亚洲精品蜜桃久久久久久| 少妇性饥渴无码A区免费 | 亚洲日本国产综合高清| 欧美最猛性xxxxx免费| 亚洲精品国产综合久久久久紧 | 成全高清视频免费观看| 在线a亚洲老鸭窝天堂av高清| 免费做爰猛烈吃奶摸视频在线观看| jlzzjlzz亚洲jzjzjz| a级毛片无码免费真人| 国产成人亚洲精品蜜芽影院| 免费在线观看污网站| 日韩a级无码免费视频| 精品亚洲成a人片在线观看少妇| 五月亭亭免费高清在线| yy6080久久亚洲精品| 亚洲视频在线免费| 最近中文字幕免费mv视频7| 久久亚洲精品高潮综合色a片| 2015日韩永久免费视频播放| 亚洲欧洲另类春色校园网站| 日本特黄特色aa大片免费| 一进一出60分钟免费视频| 亚洲国语精品自产拍在线观看| 99ee6热久久免费精品6| 亚洲Aⅴ在线无码播放毛片一线天 亚洲avav天堂av在线网毛片 | 久久精品国产亚洲AV麻豆网站| 免费可以在线看A∨网站| 国产亚洲精品仙踪林在线播放| 国产成A人亚洲精V品无码| 黄页网站免费在线观看| 一级A毛片免费观看久久精品| 久久亚洲AV成人无码| www国产亚洲精品久久久日本|