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

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

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

    菜園子

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

    關于Hibernate的查詢從數(shù)據(jù)庫映射到JavaBean

     


     

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

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

    setResultTransformer(new AliasToBeanResultTransformer (CatDTO.class))

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

    通過源碼發(fā)現(xiàn),要求java的屬性要和數(shù)據(jù)庫的字段名大小寫一樣,并且全匹配。

     

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

     

    也是通過執(zhí)行sql并把返回的結果轉換成Java的方法,這個就可以忽視數(shù)據(jù)庫字段的大小寫問題,
    仿照這個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的查詢從數(shù)據(jù)庫映射到JavaBean 2014-08-28 13:36 簡歷網(wǎng)
    謝謝博主的分享  回復  更多評論
      

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


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


    網(wǎng)站導航:
     
    主站蜘蛛池模板: 黑人粗长大战亚洲女2021国产精品成人免费视频 | 精品免费国产一区二区三区| 大妹子影视剧在线观看全集免费| 亚洲午夜理论片在线观看| 久久精品7亚洲午夜a| 精品国产香蕉伊思人在线在线亚洲一区二区| 国产92成人精品视频免费| a级毛片高清免费视频| 极品美女一级毛片免费| 国产人成亚洲第一网站在线播放| 亚洲国产成人私人影院| 亚洲中文字幕无码日韩| 国产精品无码免费视频二三区| 久久久久久曰本AV免费免费| 成人影片一区免费观看| CAOPORN国产精品免费视频| 色天使色婷婷在线影院亚洲| 亚洲AV成人无码天堂| 亚洲欧洲日韩在线电影| 2022年亚洲午夜一区二区福利| 亚洲gv白嫩小受在线观看| 狠狠综合久久综合88亚洲| 国产成人精品久久亚洲高清不卡 | 成人免费无码精品国产电影| 成人毛片免费观看视频| 99久久免费国产精品特黄| 成人免费激情视频| 免费在线视频你懂的| 久草免费在线观看视频| 国产精品免费网站| 一个人在线观看视频免费| 91免费播放人人爽人人快乐| 全免费毛片在线播放| 久久受www免费人成_看片中文| 国产91色综合久久免费| 亚洲人成网站免费播放| 女人被男人桶得好爽免费视频 | 亚洲AV无码乱码在线观看裸奔 | 黄色网址免费在线观看| 国产JIZZ中国JIZZ免费看| sss在线观看免费高清|