关于JPA原生SQL查询

  1. 关于JPA写原生SQL一直很有争议,大部分说JPA不适合写SQL,或者对于SQL的支持很垃圾,但是在我使用JPA的过程中,发现JPA写SQL其实还可以,那么咱们就讨论下JPA原生SQL的写法.
    关于JPA原生SQL是采用entityManager类进行查询,当前查询可以返回Map<Object,Object>对象,当然也可以返回List<T>对象.

定义原生接口

  1. BaseDao.java
/**
 *  这里为了演示JPA原生sql,就先定义个空接口, 如果运用到项目中,这里可以定义 
 *  一些公共的查询方法,如简单的单表增删改查
 *  如: public T select(String id);
 *     public List<T> selectT(String id);
 *      .....   
**/
  public interface BaseDao<T> {

  }
  1. BaseDaoIml.java
@Repository
public class BaseDaoImpl<T> implements BaseDao<T> {

    @PersistenceContext
    private EntityManager entityManager;
    private Class<?> clz;

    public Class<?> getClz() {
        if (clz == null) {
            // 获取泛型的Class对象
            clz = ((Class<?>) (((ParameterizedType) (this.getClass().getGenericSuperclass())).getActualTypeArguments()[0]));
        }
        return clz;
    }

    public List<Object[]> listBySQL(String sql) {
        return null;
    }

    public Integer getCount(String sql, Map<String, Object> maps) {
        Query query = entityManager.createNativeQuery(sql);
        setAliasParameter(query, maps);
        BigInteger count = new BigInteger("0");
        try {
            count = (BigInteger) query.getSingleResult();
        } catch (NoResultException ex) {

        }


        return count.intValue();
    }


    public List<Object[]> executeObject(String queryStr) {
        Query query = entityManager.createNativeQuery(queryStr);
        List<Object[]> resultList = query.getResultList();
        return resultList;
    }


    public Page<T> executeNativeQuery(String queryStr, QueryConditionVo queryConditionVo) {
        return executeNativeQuery(queryStr, queryConditionVo, null);
    }

    public T executeNativeQueryObject(String querySQL) {
        return executeNativeQueryObject(querySQL, null);
    }

    public List<T> executeNativeQuery(String sql, Object... params) {
        Query query = entityManager.createNativeQuery(sql);
        setParameter(query, params);
        query.unwrap(SQLQuery.class).setResultTransformer(Transformers.aliasToBean(getClz()));
        List<T> resultList = query.getResultList();
        return resultList;
    }


    public List<T> executeNativeQuery(String sql, Map<String, Object> alias) {
        Query query = entityManager.createNativeQuery(sql);
        setAliasParameter(query, alias);
        query.unwrap(SQLQuery.class).setResultTransformer(Transformers.aliasToBean(getClz()));
        List<T> resultList = query.getResultList();
        return resultList;
    }


    public T executeNativeQueryObject(String querySQL, Object... param) {
        Query query = entityManager.createNativeQuery(querySQL);
        query.unwrap(SQLQuery.class).setResultTransformer(Transformers.aliasToBean(getClz()));
        setParameter(query, param);
        return (T) query.getSingleResult();
    }

    public <N extends Object> Page<T> executeNativeQuery(String queryStr, QueryConditionVo queryConditionVo, Map<String, Object> maps) {
        return executeNativeQuery(queryStr, queryConditionVo, maps, null);
    }


    public <N extends Object> Page<T> executeNativeQuery(String queryStr, QueryConditionVo queryConditionVo, Map<String, Object> alias, Object... params) {
        String cq = getCountHql(queryStr);
        queryStr = initSort(queryStr, queryConditionVo);
        Query query = entityManager.createNativeQuery(queryStr);
        Query cquery = entityManager.createNativeQuery(cq);
        query.unwrap(SQLQuery.class).setResultTransformer(Transformers.aliasToBean(getClz()));
        setAliasParameter(query, alias);
        setAliasParameter(cquery, alias);
        setParameter(query, params);
        setParameter(cquery, params);
        BigInteger total = new BigInteger("0");
        try {
            total = (BigInteger) cquery.getSingleResult();
        } catch (NoResultException ex) {

        }
        int page = queryConditionVo.getPagination().getPage();
        int size = queryConditionVo.getPagination().getSize();
        query.setFirstResult(page * size);
        query.setMaxResults(size);
        List resultList = query.getResultList();
        Pageable pageable = new PageRequest(page, size);
        return new PageImpl<T>(resultList, pageable, total.longValue());
    }

    private void setParameter(Query query, Object[] args) {
        if (args != null && args.length > 0) {
            int index = 1;
            for (Object arg : args) {
                if (arg == null) {
                    continue;
                }
                query.setParameter(index++, arg);
            }
        }
    }

    private void setAliasParameter(Query query, Map<String, Object> alias) {
        if (alias != null) {
            Set<String> keys = alias.keySet();
            for (String key : keys) {
                Object val = alias.get(key);
                if (val instanceof Collection) {
                    // 查询条件是列表
                    query.setParameter(key, (Collection) val);
                } else {
                    query.setParameter(key, val);
                }
            }
        }
    }

    private String initSort(String hql, QueryConditionVo vo) {

        if (vo != null && null != vo.getSort() && vo.getSort().size() > 0) {
            Map<String, String> map = vo.getSort();
            for (String key : map.keySet()) {
                hql += " order by " + key + " " + map.get(key);
            }
        }
        return hql;
    }

    protected String getCountHql(String hql) {
        String e = hql.substring(hql.toLowerCase().indexOf("from"));
        String c = "select ifnull(count(*),0) " + e;

        return c;
    }


    public Integer findBySql(String sql, Object... args) {
        Query query = entityManager.createNativeQuery(sql);
        int i = 0;
        for (Object arg : args) {
            query.setParameter(++i, arg);
        }
        return query.getResultList().size();
    }

    public Integer findBySql(String sql, Map<String, Object> args) {
        Query query = entityManager.createNativeQuery(sql);
        setAliasParameter(query, args);
        return query.getResultList().size();
    }
}

以上就是原生SQL的具体实现.关于调用,请看下面的例子

public interface IDataSourceCustomRepository extends BaseDao<DataSourceModel> {


    Page<DataSourceModel> findDataSourceByAdmin(QueryConditionVo queryConditionVo, User user);

    Page<DataSourceModel> findDataSourceByGuest(QueryConditionVo queryConditionVo, User user);

}

 @Repository
public class DataSourceCustomRepository extends BaseDaoImpl<DataSourceModel> implements IDataSourceCustomRepository {

    @Override
    public Page<DataSourceModel> findDataSourceByAdmin(QueryConditionVo queryConditionVo, User user) {
        StringBuilder sql = new StringBuilder("SELECT T.* FROM ( SELECT " +
                "   mds.id,   " +
                "   DATE_FORMAT(mds.gmt_create,'%Y-%m-%d %H:%i:%s') AS gmtCreate,   " +
                "   mds.back_ground_color AS backGroundColor,   " +
                "   mds.data_list_name AS dataListName,   " +
                "   mds.data_list_path AS dataListPath,   " +
                "   mds.data_source_content AS dataSourceContent,   " +
                "   mds.data_source_name AS dataSourceName,   " +
                "   if(mds.data_source_state = 'PRIVATE','PUBLIC','PUBLIC')AS dataSourceState,  " +
                "   IF( 1, 'ADMIN' ,'ADMIN') AS userState," +
                "   mds.font_path AS fontPath,   " +
                "   mds.layers_num AS layersNum,   " +
                "   mds.map_thumbnail_name AS mapThumbnailName,   " +
                "   mds.map_thumbnail_path AS mapThumbnailPath,   " +
                "   mds.remark AS remark,   " +
                "   mds.province AS province,   " +
                "   mds.city AS city,   " +
                "   mds.area AS area,   " +
                "   mds.create_by_id AS createBy ," +
                "   mds.is_deleted AS isDeleted, " +
                "   mds.address AS address," +
                "  mds.resource_state AS resourceState," +
                "   (select COUNT(*) from m_project_datasource where mds.id=datasource_id) projectCount " +
                " FROM   " +
                "   m_data_source AS mds  ) AS T WHERE 1=1  and T.isDeleted='N' ");
        SearchDataSourceModel searchDataSourceModel = JSON.parseObject(queryConditionVo.getParams(), SearchDataSourceModel.class);
        Map<String, Object> map = Maps.newHashMap();
        return getDataSourceModels(queryConditionVo, searchDataSourceModel, sql, map);

    }

    @Transactional
    public Page<DataSourceModel> findDataSourceByGuest(QueryConditionVo queryConditionVo, User user) {
        SearchDataSourceModel searchDataSourceModel = JSON.parseObject(queryConditionVo.getParams(), SearchDataSourceModel.class);
        StringBuilder sql = new StringBuilder("SELECT T.* FROM ( SELECT " +
                "  mds.id, " +
                "   DATE_FORMAT(mds.gmt_create,'%Y-%m-%d %H:%i:%s') AS gmtCreate,   " +
                "  mds.back_ground_color AS backGroundColor, " +
                "  mds.data_list_name AS dataListName, " +
                "  mds.data_list_path AS dataListPath, " +
                "  mds.data_source_content AS dataSourceContent, " +
                "  mds.data_source_name AS dataSourceName, " +
                "IF " +
                "  ( " +
                "    mdsp.application_enum IS NULL, " +
                "    mds.data_source_state, " +
                "    mdsp.application_enum  " +
                "  ) AS dataSourceState, " +
                "  mds.font_path AS fontPath, " +
                "  mds.layers_num AS layersNum, " +
                "  mds.map_thumbnail_name AS mapThumbnailName, " +
                "  mds.map_thumbnail_path AS mapThumbnailPath, " +
                "  mds.remark AS remark, " +
                "  mds.province AS province, " +
                "  mds.city AS city, " +
                "  mds.area AS area,   " +
                "  mds.create_by_id AS createBy ," +
                "  mds.is_deleted AS isDeleted, " +
                "  mds.address AS address," +
                "  mds.resource_state AS resourceState," +
                "   (select COUNT(*) from m_project_datasource where mds.id=datasource_id) projectCount " +
                " FROM " +
                "  m_data_source AS mds " +
                "  LEFT JOIN ( SELECT * FROM m_data_source_application WHERE application_user =:userId AND apply_current_enum='CURRENT_APPLY') AS mdsp ON mds.id = mdsp.data_source_id " +
                " WHERE mds.is_deleted = 'N'  ) AS T WHERE 1=1 and T.isDeleted='N'  ");

        Map<String, Object> map = Maps.newHashMap();
        map.put("userId", user.getId());
        return getDataSourceModels(queryConditionVo, searchDataSourceModel, sql, map);
    }

    private Page<DataSourceModel> getDataSourceModels(QueryConditionVo queryConditionVo, SearchDataSourceModel searchDataSourceModel, StringBuilder sql, Map<String, Object> map) {
        if (StringUtils.isNotEmpty(searchDataSourceModel.getDataSourceName())) {
            sql.append(" and  T.dataSourceName LIKE :dataSourceName");
            map.put("dataSourceName", "%" + searchDataSourceModel.getDataSourceName() + "%");
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getGmtCreateStart())) {
            sql.append(" and DATE_FORMAT(T.gmtCreate,'%Y-%m-%d') >= :gmtCreate");
            map.put("gmtCreate", searchDataSourceModel.getGmtCreateStart());
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getGmtCreateEnd())) {
            sql.append(" and  DATE_FORMAT(T.gmtCreate,'%Y-%m-%d') <= :gmtCreate");
            map.put("gmtCreate", searchDataSourceModel.getGmtCreateEnd());
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getSourceState())) {
            sql.append(" and  T.dataSourceState IN(:dataSourceState)");
            map.put("dataSourceState", searchDataSourceModel.getSourceState());
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getProvince())) {
            sql.append(" and  T.province =:province");
            map.put("province", searchDataSourceModel.getProvince());
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getCity())) {
            sql.append(" and  T.city = :city");
            map.put("city", searchDataSourceModel.getCity());
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getArea())) {
            sql.append(" and  T.area = :area");
            map.put("area", searchDataSourceModel.getArea());
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getResourceState())) {
            sql.append(" and  T.resourceState = :resourceState");
            map.put("resourceState", searchDataSourceModel.getResourceState());
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getUseState()) && searchDataSourceModel.getUseState().equals(UseState.USE)) {
            sql.append(" and  T.projectCount > 0");
        }
        if (!ObjectUtils.isEmpty(searchDataSourceModel.getUseState()) && searchDataSourceModel.getUseState().equals(UseState.NOT_USE)) {
            sql.append(" and  T.projectCount = 0");
        }
        return super.executeNativeQuery(sql.toString(), queryConditionVo, map);
    }

}

返回的实体类对象

@Data
public class DataSourceModel {

    private String id;
    /**
     * 数据源名称
     */

    private String dataSourceName;

    /**
     * gis数据源内容,这个就是URL地址
     */
    private String dataSourceContent;
    /**
     * 备注信息,简介
     */
    private String remark;
    /**
     * 创建人
     */
    private String createBy;
    /**
     * 更新人
     @OneToOne private User updateBy;*/
    /**
     * 字体地址
     */
    private String fontPath;

    /**
     * 雪碧图地址
     */
    private String spritePath;
    /**
     * 数据源状态 private:私有
     * <p>
     * public : 公开
     */
    private String dataSourceState;

    /**
     * 数据清单,文件路径
     */
    private String dataListPath;

    /**
     * 地图缩略图,文件路径
     */
    private String mapThumbnailPath;

    /**
     * 数据清单,文件名称
     */
    private String dataListName;


    /**
     * 地图缩略图,图片名称 申请
     */
    private String mapThumbnailName;

    /**
     * 图层总数
     */
    private String layersNum;

    /**
     * 背景色,用户前端展示
     */
    private String backGroundColor;


    /**
     * 区域地址
     */
    private String province;


    private String city;


    private String area;

    private String gmtCreate;

    private String userState = UserState.GUEST.getName();


    private String isDeleted;

    private BigInteger projectCount;

    private String address;

    private String resourceState;

}

以上就是原生SQL返回实体类的调用方式,是不是很简单呢. 注意:JPA返回对象属性时,会有较高的要求,一般int类型时,jpa会返回biginteger,等等,有些属性需要注意,不过报错时,jpa会给出友好的提示,修改属性就好了

对了,关于BaseDaoImpl.java的QueryConditionVo类,这里没有给出,请翻阅我之前的文章JPA表达式(一)文章,里面有具体写法

欢迎小伙伴留言.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容