<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 閱讀(10117) 評論(2)  編輯  收藏

    Feedback

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

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


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


    網站導航:
     
    主站蜘蛛池模板: 久9久9精品免费观看| 永久久久免费浮力影院| 亚洲无mate20pro麻豆| 国产精品视_精品国产免费| av成人免费电影| 久久丫精品国产亚洲av| 日韩a级毛片免费观看| 91精品成人免费国产| 亚洲依依成人精品| 亚洲av无码不卡私人影院| 久久永久免费人妻精品下载| 亚洲欧洲无码一区二区三区| 亚洲国产成人片在线观看 | 久久久久国色AV免费看图片| 成人午夜免费视频| 亚洲伊人久久大香线蕉啊| 亚洲国产一成久久精品国产成人综合| 91香蕉国产线观看免费全集| MM1313亚洲国产精品| 亚洲综合色一区二区三区小说| 亚洲成a人无码av波多野按摩| 亚洲一区二区三区免费视频| 二个人看的www免费视频| 亚洲欧美成人av在线观看| 亚洲最新视频在线观看| 亚洲视频在线精品| 成年人视频在线观看免费| 嫩草成人永久免费观看| 特级一级毛片免费看| 亚洲色大网站WWW永久网站| 亚洲成人动漫在线| 久久影院亚洲一区| 国产成人无码a区在线观看视频免费 | 久久亚洲精品视频| 免费一级毛片一级毛片aa| 无码永久免费AV网站| 精品一区二区三区免费毛片爱| sss日本免费完整版在线观看| 亚洲av永久无码精品网址| 亚洲人成网男女大片在线播放| 亚洲色图国产精品|