通过反射封装JDBC
具体上代码我的BaseDao:
public class BaseDao<T> {
private Class clazz;
private Properties pro=null;
public BaseDao(){
ParameterizedType pt=(ParameterizedType) this.getClass().getGenericSuperclass(); //按英译意思是获取这个类的泛型父类
clazz=(Class) pt.getActualTypeArguments()[0]; //得到真实的类型内容
pro=MainKey.getMainKeyType(clazz); //静态方法获取的是每个实体类的类名(相当于数据库里的表明)和数据库里的主键自动增长列
}
//返回数据库连接
private Connection getConnection() throws IOException{
try {
Properties pro=new Properties();
pro.load(BaseDao.class.getClassLoader().getResourceAsStream("config.Properties")); //我的连接数据库配置文件
Class.forName(pro.getProperty("driver"));
return DriverManager.getConnection(pro.getProperty("url"), pro.getProperty("user"), pro.getProperty("pwd"));
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
/**
* 添加
* @param t
* 要保存的类
* @return 成功返回true/失败返回false
*/
public boolean save(T t){
Connection con=null;
try {
con = getConnection();
Field[] filds=clazz.getDeclaredFields(); //获取公开的字段包括私有
StringBuffer sb=new StringBuffer();
sb.append("INSERT INTO ["+t.getClass().getSimpleName()+ "](");
for (Field field : filds) {
if(field.getName().equals(pro.getProperty("idname"))){
continue;
}
sb.append(field.getName()+",");
}
sb.deleteCharAt(sb.length()-1);
sb.append(")VALUES(");
List<Object> prams=new ArrayList<Object>();
for (Field field : filds) {
if(field.getName().equals(pro.getProperty("idname"))){
continue;
}
PropertyDescriptor pd=new PropertyDescriptor(field.getName(),clazz);
Method mt=pd.getReadMethod();
Object obj=mt.invoke(t);
prams.add(obj);
sb.append("?,");
}
sb.deleteCharAt(sb.length()-1);
sb.append(")");
PreparedStatement ps=con.prepareStatement(sb.toString(),Statement.RETURN_GENERATED_KEYS); //Statement.RETURN_GENERATED_KEYS可以获
//得自增列的id号
for (int i = 0; i < prams.size(); i++) {
ps.setObject(i+1,prams.get(i));
}
int count=ps.executeUpdate();
ResultSet rs=ps.getGeneratedKeys(); //在这里可以把自增列的ID号读出来
if(rs.next()){
System.out.println("key:"+rs.getInt(1));
}
if(count>0){
return true;
}
} catch (Exception e) {
e.printStackTrace();
}finally{
try {
con.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
return false;
}
/**
* 删除
* @param ser 参数
* @return 成功返回true/失败返回false
*/
public boolean delete(Serializable ser){
Connection con=null;
try {
con=getConnection();
String sql="DELETE FROM"+pro.getProperty("tableName")+" where "+pro.getProperty("idname")+"=?";
PreparedStatement ps=con.prepareStatement(sql);
ps.setObject(1,ser);
int count=ps.executeUpdate();
if(count>0){
return true;
}
} catch (Exception e) {
e.printStackTrace();
}finally{
try {
con.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
return false;
}
/**
* 更新
* @param t要更新的类
* @return 成功返回true/失败返回false
*/
public boolean update(T t){
Connection con=null;
try {
con=getConnection();
StringBuffer sb=new StringBuffer();
sb.append("UPDATE"+pro.getProperty("tableName")+"set");
Field[] fields=clazz.getDeclaredFields();
List<Object> prams=new ArrayList<Object>();
for (Field field : fields) {
if(pro.getProperty("idname").equals(field.getName())){
continue;
}
sb.append(" "+field.getName()+"=?,");
PropertyDescriptor pd=new PropertyDescriptor(field.getName(), clazz);
Method mt=pd.getReadMethod();
Object obj=mt.invoke(t);
prams.add(obj);
}
sb.deleteCharAt(sb.length()-1);
sb.append(" where "+pro.getProperty("idname")+"=?");
PropertyDescriptor pd=new PropertyDescriptor(pro.getProperty("idname"), clazz);
Method mt=pd.getReadMethod();
Object obj=mt.invoke(t);
prams.add(obj);
PreparedStatement ps=con.prepareStatement(sb.toString());
for (int i = 0; i < prams.size(); i++) {
ps.setObject(i+1,prams.get(i));
}
int count=ps.executeUpdate();
if(count>0){
return true;
}
} catch (Exception e) {
e.printStackTrace();
}finally{
try {
con.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
return false;
}
/**
* 查询所有
* @return 实体类集合
*/
public List<T> findAll(){
Connection con=null;
PreparedStatement ps=null;
ResultSet rs=null;
try {
con=getConnection();
String sql="select * from"+pro.getProperty("tableName");
ps=con.prepareStatement(sql);
rs=ps.executeQuery();
List<T> list=new ArrayList<T>();
while(rs.next()){
T t=(T) clazz.newInstance();
Map<String,Field> map=new HashMap<String, Field>();
for (Field field : clazz.getDeclaredFields()) {
map.put(field.getName(),field);
}
int count=rs.getMetaData().getColumnCount();//每行多少列
for (int i = 0; i < count; i++) {
try {
Field field=map.get(rs.getMetaData().getColumnName(i+1).toLowerCase());
field.setAccessible(true);
if (field.getType().getName().equals(int.class.getName())|| field.getType().getName().equals(Integer.class.getName())) {
field.set(t, Integer.valueOf(rs.getObject(i + 1).toString()));
} else if (field.getType().getName().equals(Date.class.getName())) {
field.set(t, new Date(rs.getDate(i + 1).getTime()));
} else if (field.getType().getName().equals(Double.class.getName())|| field.getType().getName().equals(double.class.getName())) {
field.set(t, rs.getDouble(i + 1));
} else if (field.getType().getName().equals(
String.class.getName())) {
field.set(t, rs.getObject(i + 1).toString());
} else {
field.set(t, rs.getObject(i));
}
} catch (Exception e) {
e.printStackTrace();
}
}
list.add(t);
}
return list;
} catch (Exception e) {
e.printStackTrace();
}finally{
try {
rs.close();
ps.close();
con.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
return null;
}
****************************************************************************************************************************
每个实体类都配置了个Properties:用来存放每个实体类的类名(相当于数据库里的表明)和数据库里的主键自动增长列
tableName=[user]
MainKeyType=identity
idname=id
***********************************************
我的UserDao只要继承BaseDao就行:
public class UserDao extends BaseDao<User>
**********************************************
我的工具类用来返回 读取每个实体类的类名(相当于数据库里的表明)和数据库里的主键自动增长列的Properties:
public class MainKey {
public static Properties getMainKeyType(Class clazz){
Properties pro=new Properties();
String path=clazz.getName().replace(".", "/")+".Properties";
System.out.println(path);
try {
pro.load(MainKey.class.getClassLoader().getResourceAsStream(path));
} catch (IOException e) {
e.printStackTrace();
}
return pro;
*******************************************************
Main方法测试:
public static void main(String[] args) throws IOException {
UserDao ud=new UserDao();
User user=new User();
user.setId(2);
user.setAge(22);
user.setName("sp");
user.setPhone(138438);
ud.save(user);
// ud.update(user);
// System.out.println(ud.delete(1));
//MainKey.getMainKeyType(User.class);
}
}
}
通过反射封装JDBC的更多相关文章
- jdbc操作mysql(四):利用反射封装
前言 有了前面利用注解拼接sql语句,下面来看一下利用反射获取类的属性和方法 不过好像有一个问题,数据库中的表名和字段中带有下划线该如何解决呢 实践操作 工具类:获取connection对象 publ ...
- MySQL数据库学习笔记(十一)----DAO设计模式实现数据库的增删改查(进一步封装JDBC工具类)
[声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...
- MySQL数据库学习笔记(十)----JDBC事务处理、封装JDBC工具类
[声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...
- JDBC学习笔记(5)——利用反射及JDBC元数据编写通用的查询方法
JDBC元数据 1)DatabaseMetaData /** * 了解即可:DatabaseMetaData是描述数据库的元数据对象 * 可以由Connection得到 */ 具体的应用代码: @Te ...
- MySQL JDBC事务处理、封装JDBC工具类
MySQL数据库学习笔记(十)----JDBC事务处理.封装JDBC工具类 一.JDBC事务处理: 我们已经知道,事务的概念即:所有的操作要么同时成功,要么同时失败.在MySQL中提供了Commit. ...
- DAO设计模式实现数据库的增删改查(进一步封装JDBC工具类)
DAO设计模式实现数据库的增删改查(进一步封装JDBC工具类) 一.DAO模式简介 DAO即Data Access Object,数据访问接口.数据访问:故名思义就是与数据库打交道.夹在业务逻辑与数据 ...
- 【转】JDBC学习笔记(5)——利用反射及JDBC元数据编写通用的查询方法
转自:http://www.cnblogs.com/ysw-go/ JDBC元数据 1)DatabaseMetaData /** * 了解即可:DatabaseMetaData是描述数据库的元数据对象 ...
- Java -- JDBC_利用反射及 JDBC 元数据编写通用的查询方法
先利用 SQL 进行查询,得到结果集: 利用反射创建实体类的对象:创建对象: 获取结果集的列的别名: 再获取结果集的每一列的值, 结合 3 得到一个 Map,键:列的别名,值:列的值: 再利用反射为 ...
- 一、JDBC的概述 二、通过JDBC实现对数据的CRUD操作 三、封装JDBC访问数据的工具类 四、通过JDBC实现登陆和注册 五、防止SQL注入
一.JDBC的概述###<1>概念 JDBC:java database connection ,java数据库连接技术 是java内部提供的一套操作数据库的接口(面向接口编程),实现对数 ...
随机推荐
- VS附加到进程调试的方法及应用场景
应用场景:.Net做网站时,代码量很大的时候,每次调试一个网页都编译整个网站是不显示的,而且有时候整个网站是存在错误的,通不过编译.这时你又要调试某部分网页,就可以通过附加到进程调试.方法如下: (1 ...
- Python 汉字简体和繁体的相互转换
其实利用python实现汉字的简体和繁体相互转早有人做过,并发布到github上了,地址:https://github.com/skydark/nstools/tree/master/zhtools ...
- 20145211 《Java程序设计》第6周学习总结——三笑徒然当一痴
教材学习内容总结 I/O--InputStream与OutStream Java中I/O操作主要是指使用Java进行输入,输出操作.这与c++中的iostream并无太大区别. Java所有的I/O机 ...
- 转:自定义ASP.NET MVC Html辅助方法
在ASP.NET MVC中,Html辅助方法给我们程序员带来很多方便,其重要性也就不言自明.有时候,我们不想重复地写一些HTML代码,或者MS没有提供我们想要的那个HTML标签的Html辅助方法,那么 ...
- OO之美
㈠ 设计的分寸 对于设计,还有很多看似"惯常"的法则与经验广泛存在于软件系统中,例如除了经典的23种设计设计模式.还有很多模式之外的模式,按照粒度的大小,系统的特点,规模的大小,而 ...
- synchronized原理
http://www.cnblogs.com/YDDMAX/p/5658607.html http://www.cnblogs.com/YDDMAX/p/5658668.html synzhroniz ...
- 详解Spring事件驱动模型
转载自:http://jinnianshilongnian.iteye.com/blog/1902886#comments 事件驱动模型简介 事件驱动模型也就是我们常说的观察者,或者发布-订阅模型:理 ...
- 让DIV中的内容水平和垂直居中
让一个层水平垂直居中是一个非常常见的布局方式,但在html中水平居中使用margin:0px auto;可以实现,但垂直居中使用外边距是无法达到效果的.(页面设置height:100%;是无效的),这 ...
- CAS实现无锁模式
用多线程实现一个数字的自增长到1000000,分别用无锁模式和锁模式来实现代码. 1.使用ReentrantLock. package test; import java.util.concurren ...
- JavaScript : DOM文档解析详解
JavaScript DOM 文档解析 1.节点(node):来源于网络理论,代表网络中的一个连接点.网络是由节点构成的集合 <p title=“a gentle reminder”> ...