利用泛型抽取Dao层,加事务注解问题(java.lang.Class cannot be cast to java.lang.reflect.ParameterizedType)

利用泛型抽取Dao层,加事务注解问题(java.lang.Class cannot be cast to java.lang.reflect.ParameterizedType)

大家好,又见面了,我是全栈君。

想利用泛型抽取BaseDao层,简化操作时出现故障:

@Transactional这个注解是能够继承的。于是就想写在抽取的BaseDao层上,让实现的类能够不用写@Transactional,就可开启事务。

问题描写叙述:

因为偷懒。没给BaseDao抽接口。代码例如以下:

package com.liang.ssh2.base;

import java.lang.reflect.ParameterizedType;
import java.util.Collections;
import java.util.List;

import javax.annotation.Resource;

import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.springframework.transaction.annotation.Transactional;

import com.liang.ssh2.config.Configuration;
import com.liang.ssh2.entity.Page;
import com.liang.ssh2.util.QueryHelper;
@Transactional
@SuppressWarnings("unchecked")
public  class BaseDao<T>{
	@Resource
	private SessionFactory sessionFactory;
	Class <T> clazz;
	/**
	 * 通过反射获取參数类型
	 */
	public BaseDaoImpl(){
		System.out.println(this);
		ParameterizedType pt = (ParameterizedType) this.getClass().getGenericSuperclass();
		clazz=(Class<T>) pt.getActualTypeArguments()[0];
	}
	/**
	 * 获取当前可用的Session
	 * @return
	 */
	protected Session getSession() {
		return sessionFactory.getCurrentSession();
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#save(T)
	 */
	public void save(T entity){
		getSession().save(entity);
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#getById(java.lang.Long)
	 */
	
	public T getById(Long id){
		if(id==null) return null;
		return (T) getSession().get(clazz, id);
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#update(T)
	 */
	public void update(T entity){
		getSession().update(entity);
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#delete(java.lang.Long)
	 */
	public void delete(Long id){
		if(id!=null){
			Object entity=getById(id);
			if(entity!=null){
				getSession().delete(entity);
			}
		}
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#getByIds(java.lang.Long[])
	 */
	public List<T> getByIds(Long[] ids){
		if(ids==null||ids.length==0){
			return Collections.EMPTY_LIST;
		}
		return getSession().createQuery(//
				"from "+clazz.getSimpleName()+"where id in(:ids)")//
				.setParameterList("ids", ids)//
				.list();
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#findAll()
	 */
	public List<T> findAll(){
		return getSession().createQuery(//
				"from "+clazz.getSimpleName())//
				.list();
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#getPage(long, java.lang.Long, com.liang.ssh2.util.QueryHelper)
	 */
	public Page getPage(long currentPage,Long pageSize,QueryHelper queryHelper){
		
		//假设用户没有动态指定pageSize。则读取配置文件
		if(pageSize==null){
			pageSize=Configuration.getPageSize();
		}
		//获得參数
		List<Object> parameters = queryHelper.getParameters();
		
		Query query = getSession().createQuery(queryHelper.getQueryListHql());
		
		if(parameters!=null&¶meters.size()>0){
			for (int i = 0; i < parameters.size(); i++) {
				query.setParameter(i, parameters.get(i));
			}
		}
		query.setFirstResult((int) ((currentPage-1)*pageSize));
		query.setMaxResults(pageSize.intValue());
		List recordList = query.list();
		// 查询总记录数
		query = getSession().createQuery(queryHelper.getQueryCountHql()); // 注意空格!

if (parameters != null && parameters.size() > 0) { // 设置參数 for (int i = 0; i < parameters.size(); i++) { query.setParameter(i, parameters.get(i)); } } Long recordCount = (Long) query.uniqueResult(); // 查询 return new Page(currentPage, pageSize, recordCount, recordList); }}

使用例如以下:

package com.liang.ssh2.service.impl;

import org.springframework.stereotype.Service;

import com.liang.ssh2.base.BaseDao;
import com.liang.ssh2.entity.User;
@Service
public class UserServiceImpl extends BaseDao<User>{

}


注意我是把@Transactional放在BaseDao上了。因为@Transactional可继承,所以UserServiceImpl是不用放的

当启动server时会报错:Caused by: java.lang.ClassCastException: java.lang.Class cannot be cast to java.lang.reflect.ParameterizedType

为什么会这样呢?

測试了半天发现,出错就在BaseDao上注解:@Transactional上,原因不是非常清楚!

。!

两种解决方式:

一、直接将BaseDao上的@Transactional注解去掉,在实现类上加@Transactional,对于本例,在UserServiceImpl加上@Transactional就可以开启事务。也不会报错!

二、别偷懒了。乖乖的给BaseDao抽个接口吧,其它什么都不用改,@Transactional还是照样能够继承。改动代码例如以下:

package com.liang.ssh2.base;

import java.util.List;

import com.liang.ssh2.entity.Page;
import com.liang.ssh2.util.QueryHelper;

public interface BaseDao<T> {

	/**
	 * 保存实体
	 * @param entity
	 */
	public abstract void save(T entity);

	/**
	 * 依据id获取实体
	 * @param id
	 * @return
	 */

	public abstract T getById(Long id);

	public abstract void update(T entity);

	public abstract void delete(Long id);

	public abstract List<T> getByIds(Long[] ids);

	public abstract List<T> findAll();

	/**
	 *  获取page
	 * @param currentPage
	 * @param pageSize //假设用户没有动态指定pageSize(null),则读取配置文件
	 * @param queryHelper
	 * @return
	 */
	public abstract Page getPage(long currentPage, Long pageSize,
			QueryHelper queryHelper);

}

package com.liang.ssh2.base;

import java.lang.reflect.ParameterizedType;
import java.util.Collections;
import java.util.List;

import javax.annotation.Resource;

import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.springframework.transaction.annotation.Transactional;

import com.liang.ssh2.config.Configuration;
import com.liang.ssh2.entity.Page;
import com.liang.ssh2.util.QueryHelper;
@Transactional
@SuppressWarnings("unchecked")
public  class BaseDaoImpl<T> implements BaseDao<T>{
	@Resource
	private SessionFactory sessionFactory;
	Class <T> clazz;
	/**
	 * 通过反射获取參数类型
	 */
	public BaseDaoImpl(){
		System.out.println(this);
		ParameterizedType pt = (ParameterizedType) this.getClass().getGenericSuperclass();
		clazz=(Class<T>) pt.getActualTypeArguments()[0];
	}
	/**
	 * 获取当前可用的Session
	 * @return
	 */
	protected Session getSession() {
		return sessionFactory.getCurrentSession();
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#save(T)
	 */
	public void save(T entity){
		getSession().save(entity);
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#getById(java.lang.Long)
	 */
	
	public T getById(Long id){
		if(id==null) return null;
		return (T) getSession().get(clazz, id);
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#update(T)
	 */
	public void update(T entity){
		getSession().update(entity);
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#delete(java.lang.Long)
	 */
	public void delete(Long id){
		if(id!=null){
			Object entity=getById(id);
			if(entity!=null){
				getSession().delete(entity);
			}
		}
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#getByIds(java.lang.Long[])
	 */
	public List<T> getByIds(Long[] ids){
		if(ids==null||ids.length==0){
			return Collections.EMPTY_LIST;
		}
		return getSession().createQuery(//
				"from "+clazz.getSimpleName()+"where id in(:ids)")//
				.setParameterList("ids", ids)//
				.list();
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#findAll()
	 */
	public List<T> findAll(){
		return getSession().createQuery(//
				"from "+clazz.getSimpleName())//
				.list();
	}
	/* (non-Javadoc)
	 * @see com.liang.ssh2.base.BaseDao#getPage(long, java.lang.Long, com.liang.ssh2.util.QueryHelper)
	 */
	public Page getPage(long currentPage,Long pageSize,QueryHelper queryHelper){
		
		//假设用户没有动态指定pageSize,则读取配置文件
		if(pageSize==null){
			pageSize=Configuration.getPageSize();
		}
		//获得參数
		List<Object> parameters = queryHelper.getParameters();
		
		Query query = getSession().createQuery(queryHelper.getQueryListHql());
		
		if(parameters!=null&¶meters.size()>0){
			for (int i = 0; i < parameters.size(); i++) {
				query.setParameter(i, parameters.get(i));
			}
		}
		query.setFirstResult((int) ((currentPage-1)*pageSize));
		query.setMaxResults(pageSize.intValue());
		List recordList = query.list();
		// 查询总记录数
		query = getSession().createQuery(queryHelper.getQueryCountHql()); // 注意空格。
		if (parameters != null && parameters.size() > 0) { // 设置參数
			for (int i = 0; i < parameters.size(); i++) {
				query.setParameter(i, parameters.get(i));
			}
		}
		Long recordCount = (Long) query.uniqueResult(); // 查询
		
		return new Page(currentPage, pageSize, recordCount, recordList);
	}
}

</pre><pre name="code" class="java">package com.liang.ssh2.service.impl;

import org.springframework.stereotype.Service;

import com.liang.ssh2.base.BaseDaoImpl;
import com.liang.ssh2.entity.User;
@Service
public class UserServiceImpl extends BaseDaoImpl<User>{

}

想偷个懒。少写个接口。也不easy啊!!!!!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/116530.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 关于linux文件系统软连接_centos7删除目录命令

    关于linux文件系统软连接_centos7删除目录命令前言经常使用centos系统的同学都知道,在全局安装命令指令时,即使是执行了npminstall-g但是安装之后的命令仍然说找不到那么我们改如何操作呢?软连接简介centos下的ln命令就相当于window下的建立快捷方式,链接文件甚至可以链接不存在的文件,这就产生一般称之为”断链”的现象,链接文件甚至可以循环链接自己。类似于编程语言中的递归。软链接文件只是其源文件的一个标记,当删除了源…

    2022年9月1日
    1
  • js中正则表达式的用法_iperf使用方法

    js中正则表达式的用法_iperf使用方法首先必须说明的是,这类文章(js正则表达式)在c站或者整个it类论坛是很多人写过的,而我认为我这篇的不同之处在于更加“小白”化,这也与我一贯的风格有关吧。关于JavaScript正则表达式,其他的文章大多一上来就太过激进,不利于初学者学习(我当粗就是这么被劝退的),这也是我为什么要坚持写这篇文章,希望小白在看了这篇文章后,不管能不能完全掌握JavaScript正则表达式,但至少对JavaScript正则表达式能有一个比较深刻的印象吧。

    2022年9月20日
    0
  • html设置背景图片透明度代码,css设置图片背景透明度[通俗易懂]

    html设置背景图片透明度代码,css设置图片背景透明度[通俗易懂]我们经常会使用到背景图片,但有时候需要把背景图片设置一个半透明的效果,那么就需要以下的CSS了。这是一个CSS滤镜达到的效果,应用十分广泛。.modbr{filter:alpha(opacity=80);-moz-opacity:0.8;width:auto!important;width:100%}注:代码应用的式CSS中的Alpha滤镜,这个滤镜可以设置目标元素的透明度。还可以通过指定坐标,从…

    2022年5月12日
    50
  • pytorch-DataLoader(数据迭代器)

    pytorch-DataLoader(数据迭代器)目录1.1dataset1.1.1Map-styledatasets实现方法一(简单直白法)实现方法二(借助TensorDataset直接将数据包装成dataset类)实现方法三(地址读取法)1.1.1Iterable-styledatasets我们一般使用一个for循环(或多层的)来训练神经网络,每一次迭代,加载一个batch的数据,神经网络前向反向传播各一次并更新一次参数。而这个过程中加载一个batch的数据这一步需要使用一个torch.utils.data.DataLoader对象,并且

    2022年6月11日
    76
  • 2021 大数据应用开发Java 1+x初级实操题答案汇总–含2篇

    2021 大数据应用开发Java 1+x初级实操题答案汇总–含2篇2021大数据应用开发Java1+x初级实操题答案汇总–含2篇

    2022年5月4日
    233
  • Loadrunner11激活成功教程详解[通俗易懂]

    Loadrunner11激活成功教程详解[通俗易懂]使用说明:要以管理员的身份运行1、正常安装完LR11后。然后双击deletelicense.exe2、然后解压替换其中的2个DLL文件拷贝到”C:\ProgramFiles\HP\LoadRunner\bin\”下替换原有文件3、进入LR,输入以上的序列号即可global-100:AEACFSJI-YASEKJJKEAHJD-BCLBRweb-10000

    2022年7月22日
    48

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号