com.jredrain.dao.HibernateDao.java Source code

Java tutorial

Introduction

Here is the source code for com.jredrain.dao.HibernateDao.java

Source

/**
 * Copyright 2016 benjobs
 * <p>
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License. You may obtain a copy of the License at
 * <p>
 * http://www.apache.org/licenses/LICENSE-2.0
 * <p>
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied. See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */

package com.jredrain.dao;

import org.hibernate.*;
import org.hibernate.criterion.Restrictions;
import org.hibernate.jdbc.Work;
import org.hibernate.metadata.ClassMetadata;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Repository;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.ClassUtils;
import org.springframework.util.ObjectUtils;

import javax.annotation.Resource;
import java.io.InputStream;
import java.io.Reader;
import java.io.Serializable;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.net.URL;
import java.sql.*;
import java.util.*;
import java.util.Date;
import java.util.Map.Entry;

@Repository("hibernateDao")
@Transactional(readOnly = true)
@SuppressWarnings("unchecked")
public class HibernateDao {

    static final Logger logger = LoggerFactory.getLogger(HibernateDao.class);

    @Resource(name = "sessionFactory")
    protected SessionFactory sessionFactory;

    /**
     * ?SessionFactory
     *
     * @return
     */
    public SessionFactory getSessionFactory() {
        return sessionFactory;
    }

    /**
     * SessionFactory
     *
     * @param sessionFactory
     */
    public void setSessionFactory(SessionFactory sessionFactory) {
        this.sessionFactory = sessionFactory;
    }

    /**
     * ??session
     *
     * @return
     */
    public Session getSession() {
        return sessionFactory.getCurrentSession();
    }

    public <T, K extends Serializable> T get(Class<T> entityClass, K id) {
        entityClass = getEntityClass(entityClass);
        return (T) getSession().get(entityClass, id);
    }

    public <T> List<T> getAll(Class<T> entityClass) {
        entityClass = getEntityClass(entityClass);
        return createCriteria(entityClass).list();
    }

    public <T, K extends Serializable> List<T> getByIds(Class<T> entityClass, List<K> ids) {
        entityClass = getEntityClass(entityClass);
        String idName = getClassMetadata(entityClass).getIdentifierPropertyName();
        return createCriteria(entityClass).add(Restrictions.in(idName, ids)).list();
    }

    public <T> List<T> getByProperties(Class<T> entityClass, Map<String, Object> properties) {
        entityClass = getEntityClass(entityClass);
        return createCriteria(entityClass).add(Restrictions.allEq(properties)).list();
    }

    @Transactional(propagation = Propagation.REQUIRED, readOnly = false)
    protected void delete(Object entity) {
        getSession().delete(entity);
    }

    @Transactional(propagation = Propagation.REQUIRED, readOnly = false)
    protected <K extends Serializable> void delete(Class<?> entityClass, K id) {
        delete(getClassMetadata(entityClass).instantiate(id, null));
    }

    @Transactional(propagation = Propagation.REQUIRED, readOnly = false)
    protected <T> T save(T entity) {
        return (T) getSession().merge(entity);
    }

    @Transactional(propagation = Propagation.REQUIRED, readOnly = false)
    protected <T> void save(List<T> entities) {
        StatelessSession session = sessionFactory.openStatelessSession();
        for (T entity : entities) {
            session.update(entity);
        }
        session.close();
    }

    /**
     * @see #createSQLQuery(String, Object...)
     */

    public List<Map<String, ?>> sqlQuery(String sql, Object... parameters) {
        return createSQLQuery(sql, parameters).setResultTransformer(new MapResultTransFormer()).list();
    }

    /**
     * @see #createSQLQuery(String, Object...)
     */

    public <T> T sqlUniqueQuery(Class<T> beanClass, String sql, Object... parameters) {
        return (T) setQueryType(createSQLQuery(sql, parameters), beanClass).uniqueResult();
    }

    /**
     * @see #createSQLQuery(String, Object...)
     */

    public <T> List<T> sqlQuery(Class<T> beanClass, String sql, Object... parameters) {
        return setQueryType(createSQLQuery(sql, parameters), beanClass).list();
    }

    /**
     * sql
     * @param <T>
     * @param beanClass
     * @param sql
     * @param isNorm ??falsebeanClass?
     * @param parameters
     * @return
     */
    public <T> List<T> sqlQuery(Class<T> beanClass, String sql, Boolean isNorm, Object... parameters) {
        if (!isNorm) {
            return createSQLQuery(sql, parameters).setResultTransformer(BeanResultTransFormer.get(beanClass))
                    .list();
        } else {
            return sqlQuery(beanClass, sql, parameters);
        }
    }

    /**
     * @see #createSQLQuery(String, Object...)
     */

    public List<Map<String, ?>> sqlPageQuery(String sql, int pageNo, int pageSize, Object... parameters) {
        Query query = createSQLQuery(sql, parameters).setResultTransformer(new MapResultTransFormer());
        return pageQuery(query, pageNo, pageSize);
    }

    /**
     * @see #createSQLQuery(String, Object...)
     */

    public <T> List<T> sqlPageQuery(Class<T> beanClass, String sql, int pageNo, int pageSize,
            Object... parameters) {
        return pageQuery(setQueryType(createSQLQuery(sql, parameters), beanClass), pageNo, pageSize);
    }

    /**
     * HQL,?<br/>
     *
     * @param hql
     * @param parameters
     * @return
     * @see #createQuery(String, Object...)
     */
    public <T> T hqlUniqueQuery(String hql, Object... parameters) {
        return (T) createQuery(hql, parameters).uniqueResult();
    }

    /**
     * HQL<br/>
     *
     * @param hql
     * @param parameters
     * @return
     * @see #createQuery(String, Object...)
     */
    public List hqlQuery(String hql, Object... parameters) {
        return createQuery(hql, parameters).list();
    }

    /**
     * HQL<br/>
     *
     * @param hql
     * @param pageNo
     * @param pageSize
     * @param parameters
     * @return
     * @see #createQuery(String, Object...)
     */
    public List hqlPageQuery(String hql, int pageNo, int pageSize, Object... parameters) {
        return pageQuery(createQuery(hql, parameters), pageNo, pageSize);
    }

    /**
     * 
     *
     * @param query
     * @param pageNo
     *            ?
     * @param pageSize
     *            ???
     * @return
     */
    public static List pageQuery(Query query, int pageNo, int pageSize) {
        return query.setFirstResult((pageNo - 1) * pageSize).setMaxResults(pageSize).list();
    }

    /**
     * SQLQuery,?<br/>
     * ???<br/>
     * ??,???,:
     * <ul>
     * <li>?,?String,Date,Number...,??JPA? ?n </li>
     * <li>Map,?keyvalue????</li>
     * <li>POJO,???????</li>
     * ?,??in?
     * </ul>
     * :
     *
     * <pre>
     * &lt;code&gt;Map&lt;String, Object&gt; map = new HashMap&lt;String, Object&gt;();
     * map.put(&quot;idList&quot;, new Long[]{1L, 2L, 3L});
     * map.put(&quot;title&quot;, &quot;%2%&quot;);
     * EventAlias event = new EventAlias();
     * event.setTitle(&quot;%%&quot;);
     * sql = &quot;select e1.event_id ID,e1.event_date Date,e1.title,e1.type from event e1 where e1.title like :title or e1.title like ?1 and e1.event_id in :idList&quot;;
     * list = hibernateDao.sqlQuery(EventAlias.class, sql,event, &quot;%%&quot;, map);&lt;/code&gt;
     * </pre>
     *
     * @param sql
     * @param parameters
     * @return
     */
    public SQLQuery createSQLQuery(String sql, Object... parameters) {
        return setParameters(getSession().createSQLQuery(sql), parameters);
    }

    /**
     * Query,?<br/>
     * ???<br/>
     * ??,???,:
     * <ul>
     * <li>?,?String,Date,Number...,??JPA? ?n </li>
     * <li>Map,?keyvalue????</li>
     * <li>POJO,???????</li>
     * ?,??in?<br/>
     * ,SQL(HQL)???JDBC??,??,??in?
     * </ul>
     * :
     *
     * <pre>
     * &lt;code&gt;Map&lt;String, Object&gt; map = new HashMap&lt;String, Object&gt;();
     * map.put(&quot;idList&quot;, new Long[]{1L, 2L, 3L});
     * map.put(&quot;title&quot;, &quot;%2%&quot;);
     * EventAlias event = new EventAlias();
     * event.setTitle(&quot;%%&quot;);
     * sql = &quot;select e1.event_id ID,e1.event_date Date,e1.title,e1.type from event e1 where e1.title like :title or e1.title like ?1 and e1.event_id in :idList&quot;;
     * list = hibernateDao.sqlQuery(EventAlias.class, sql,event, &quot;%%&quot;, map);&lt;/code&gt;
     * </pre>
     *
     * @param hql
     * @param parameters
     * @return
     */
    public Query createQuery(String hql, Object... parameters) {
        return setParameters(getSession().createQuery(hql), parameters);
    }

    /**
     * ?Criteria
     *
     * @param entityClass
     * @param <T>
     * @return
     */
    public <T> Criteria createCriteria(Class<T> entityClass) {
        return getSession().createCriteria(getEntityClass(entityClass));
    }

    @Transactional(readOnly = false)
    public void executeBatch(final String[] sqlList) {
        getSession().doWork(new Work() {

            public void execute(Connection connection) throws SQLException {
                connection.setAutoCommit(false);
                Statement stmt = connection.createStatement();
                for (String sql : sqlList) {
                    stmt.addBatch(sql);
                }
                stmt.executeBatch();
                connection.commit();
            }
        });
    }

    @Transactional(readOnly = false)
    public void executeBatch(final String sql, final Object[]... parameters) {
        getSession().doWork(new Work() {

            public void execute(Connection connection) throws SQLException {
                connection.setAutoCommit(false);
                PreparedStatement stmt = connection.prepareStatement(sql);
                for (Object[] arr : parameters) {
                    int i = 1;
                    for (Object p : arr) {
                        stmt.setObject(i++, p);
                    }
                    stmt.addBatch();
                }
                stmt.executeBatch();
                connection.commit();
            }
        });
    }

    /**
     * Query? ???<br/>
     * ??,???,:
     * <ul>
     * <li>?,?String,Date,Number...,??JPA? ?n </li>
     * <li>Map,?keyvalue????</li>
     * <li>POJO,???????</li>
     * ?,??in?
     * </ul>
     * :
     *
     * <pre>
     * &lt;code&gt;Map&lt;String, Object&gt; map = new HashMap&lt;String, Object&gt;();
     * map.put(&quot;idList&quot;, new Long[]{1L, 2L, 3L});
     * map.put(&quot;title&quot;, &quot;%2%&quot;);
     * EventAlias event = new EventAlias();
     * event.setTitle(&quot;%%&quot;);
     * sql = &quot;select e1.event_id ID,e1.event_date Date,e1.title,e1.type from event e1 where e1.title like :title or e1.title like ?1 and e1.event_id in :idList&quot;;
     * list = hibernateDao.sqlQuery(EventAlias.class, sql,event, &quot;%%&quot;, map);&lt;/code&gt;
     * </pre>
     *
     * ?,SQL(HQL)???JDBC???,??,??in?
     *
     * @param query
     * @param parameters
     * @param <T>
     * @return
     */
    private static <T extends Query> T setParameters(T query, Object... parameters) {
        if (query.getNamedParameters().length == 0) {
            for (int i = 0; i < parameters.length; i++) {
                query.setParameter(i, parameters[i]);
            }
            return query;
        }
        Map<String, Object> jpaMap = new HashMap<String, Object>();
        for (int i = 0; i < parameters.length; i++) {
            Object param = parameters[i];
            if (param == null || param instanceof Collection || isSimple(param) || param.getClass().isArray()) {
                jpaMap.put(String.valueOf(i), handle(param));
            } else if (param instanceof Map) {
                query.setProperties(namedParameters((Map) param));
            } else {
                query.setProperties(param);
            }
        }
        query.setProperties(jpaMap);
        return query;
    }

    /**
     * Query
     *
     * @param query
     * @param beanClass
     * @return
     */
    private SQLQuery setQueryType(SQLQuery query, Class beanClass) {
        if (getClassMetadata(beanClass) != null) {
            query.addEntity(beanClass);
        } else {
            query.setResultTransformer(BeanResultTransFormer.get(beanClass));
        }
        return query;
    }

    /**
     * key,value,Map?JPA?
     *
     * @param parameters
     * @return
     */
    public static Map jpaParameters(Object... parameters) {
        Map<String, Object> map = new HashMap<String, Object>();
        int i = 0;
        for (Object parameter : parameters) {
            map.put(String.valueOf(i++), handle(parameter));
        }
        return map;
    }

    /**
     * Mapprimitive?,?In?
     *
     * @param map
     * @return
     */
    public static Map namedParameters(Map map) {
        for (Object o : map.entrySet()) {
            Entry en = (Entry) o;
            en.setValue(handle(en.getValue()));
        }
        return map;
    }

    /**
     * ?,??Spring?
     *
     * @param entity
     * @return
     */
    private static Class getEntityClass(Object entity) {
        if (entity instanceof Class) {
            return ClassUtils.getUserClass((Class) entity);
        }
        return ClassUtils.getUserClass(entity);
    }

    /**
     * @param entityClass
     *            
     * @return
     */
    private ClassMetadata getClassMetadata(Class entityClass) {
        return getSessionFactory().getClassMetadata(entityClass);
    }

    private static Object handle(Object value) {
        if (value != null && value.getClass().isArray()) {
            return ObjectUtils.toObjectArray(value);
        }
        return value;
    }

    private static boolean isSimple(Object value) {
        Class<? extends Object> type = value.getClass();
        if (type.isArray() || simpleTypes.contains(type))
            return true;
        for (Class clazz : simpleTypes) {
            if (clazz.isInstance(value))
                return true;
        }
        return false;
    }

    static final Set<Class> simpleTypes = new HashSet<Class>();

    {
        // primitive
        simpleTypes.add(boolean.class);
        simpleTypes.add(byte.class);
        simpleTypes.add(byte[].class);
        simpleTypes.add(double.class);
        simpleTypes.add(float.class);
        simpleTypes.add(int.class);
        simpleTypes.add(long.class);
        simpleTypes.add(short.class);
        simpleTypes.add(Boolean.class);
        simpleTypes.add(Byte.class);
        simpleTypes.add(Double.class);
        simpleTypes.add(Float.class);
        simpleTypes.add(Integer.class);
        simpleTypes.add(Long.class);
        simpleTypes.add(Short.class);

        // ?
        simpleTypes.add(String.class);
        simpleTypes.add(BigDecimal.class);
        simpleTypes.add(BigInteger.class);
        simpleTypes.add(Number.class);
        simpleTypes.add(Date.class);
        simpleTypes.add(Time.class);
        simpleTypes.add(Timestamp.class);

        // ?
        simpleTypes.add(Blob.class);
        simpleTypes.add(Clob.class);
        simpleTypes.add(InputStream.class);
        simpleTypes.add(Reader.class);
        simpleTypes.add(Ref.class);
        simpleTypes.add(SQLXML.class);
        simpleTypes.add(URL.class);

        // 
        simpleTypes.add(Class.class);
    }

}