泛型,注解,反射配合优化BaseDao的猜想
package test; import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set; import org.apache.commons.dbutils.QueryRunner;
import org.junit.Test; import cn.itcast.tool.jdbc.TXQueryRunner; public class Demo2 {
@Test
public void fun1() throws Exception{
UserDao bd=new UserDao();
User user=new User();
user.setId("guodaxia");
user.setUsername("guozhen");
user.setPassword("961012gz");
user.setState(true);
bd.add(user);
} }
class UserDao extends BaseDAO<User>{ }
class BaseDAO<T>{
QueryRunner qr=new TXQueryRunner();
Class<T> beanClass;
String id; public BaseDAO(){
beanClass=(Class)((ParameterizedType)this.getClass().getGenericSuperclass()).getActualTypeArguments()[0];
} public int add(T bean) throws Exception{
/*
* 首先我们思考,表名如何得到呢?
* 我们假定将所有的bean类名都与表名相同,那样就变成了求类名
*
* values里面的?个数是多少呢?
* 我们又假定所有的bean的属性字段对应数据库表的字段,这样所有属性的个数就对应了?的个数
*
* 参数的值如何得到呢?
* 既然我们都得到了对应的bean属性,调用其中每个属性的的getXxx方法即可
*
* 刚才我们简单地推出了sql语句和简单地推出了参数值,这样就可以执行我们的操作了。
* 但是我们的类名和表名,属性名和数据库字段名就一定会对应吗?比如我得表名是“db_user”
* 我们可以使用配置文件达到效果,这里我们使用注解实现
*
*/
// String tableName=beanClass.getSimpleName();
String tableName=beanClass.getAnnotation(Table.class).value();
Field[] fields=beanClass.getDeclaredFields();
Map<String,Object> map=new LinkedHashMap<String, Object>();//这个是为了有序
for(Field f:fields){
Annotation[] anns=f.getAnnotations();
for(Annotation a:anns){
if(a.annotationType()==ID.class){
id=f.getAnnotation(ID.class).value();//这里只是针对简单的单个id的情况
map.put(id, f.getName());
break;
}else if(a.annotationType()==Column.class){
map.put(f.getAnnotation(Column.class).value(), f.getName());
break;
}
}
}
String sql="insert into "+tableName+" values (";
// for(int i=0;i<fields.length;i++){
// sql=sql+"?";
// if(i<fields.length-1){
// sql=sql+",";
// }
// }
for(int i=0;i<map.size();i++){
sql=sql+"?";
if(i<map.size()-1){
sql=sql+",";
}
}
sql=sql+")";
System.out.println(sql);
// String sql="insert into "+"标名"+" values ("+"几个?"+")";
// Object[] params={};
/*
* 当字段类型为boolean类型的时候,其getXxx或者isXxx都有可能
*
*
*/
// Object[] params=new Object[fields.length];
ArrayList<Object> params=new ArrayList<Object>();
// for(int i=0;i<fields.length;i++){
// Field f=fields[i];
//
// String methodName="get"+f.getName().substring(0, 1).toUpperCase()+f.getName().substring(1);
// Method method = null ;
// try {
// method = beanClass.getMethod(methodName);
// } catch (NoSuchMethodException e) {
// if(f.getType().getName().equals("boolean")){
// methodName="is"+f.getName().substring(0, 1).toUpperCase()+f.getName().substring(1);
// method=beanClass.getMethod(methodName);
// }
// }
//
// params[i]=method.invoke(bean);
// }
Set<String> keys=map.keySet();
for(String key:keys){
String filedName=(String) map.get(key);
Field f=beanClass.getDeclaredField(filedName);
String methodName="get"+filedName.substring(0, 1).toUpperCase()+filedName.substring(1);
Method method=null;
try {
method = beanClass.getMethod(methodName);
} catch (NoSuchMethodException e) {
if(f.getType().getName().equals("boolean")){
methodName="is"+filedName.substring(0, 1).toUpperCase()+filedName.substring(1);
method=beanClass.getMethod(methodName);
}else{
throw e;
}
}
params.add(method.invoke(bean));
} System.out.println(params.toString());
return 0;
// return qr.update(sql,params);
} public int delete(String uuid) throws SQLException{
String sql="";
Object[] params={};
return qr.update(sql,params);
} public int update(T bean) throws SQLException{
String sql="";
Object[] params={};
return qr.update(sql,params);
} public T load(String uuid) throws SQLException{
String sql="";
Object[] params={};
return null;
} public List<T> findAll(){
String sql="";
Object[] params={};
return null;
} } package test; import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface Column {
String value();
} package test; import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface ID {
String value();
} package test; import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Table {
String value();
} package test; @Table("tb_user")
public class User {
@ID("id")
private String id;
@Column("uname")
private String username;
@Column("pwd")
private String password;
@Column("state")
private boolean state;
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public boolean isState() {
return state;
}
public void setState(boolean state) {
this.state = state;
}
public User() {
super();
}
public User(String id, String username, String password, boolean state) {
super();
this.id = id;
this.username = username;
this.password = password;
this.state = state;
} }
泛型,注解,反射配合优化BaseDao的猜想的更多相关文章
- Android高效率编码-findViewById()的蜕变-注解,泛型,反射
Android高效率编码-findViewById()的蜕变-注解,泛型,反射 Android的老朋友findViewById()篇! 先看看他每天是在干什么 //好吧,很多重复的,只不过想表达项目里 ...
- 编写高质量代码:改善Java程序的151个建议(第7章:泛型和反射___建议106~109)
建议106:动态代理可以使代理模式更加灵活 Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目标类生成代理,避免重复开发.我们知道一个静态代理是通过主题角色(Prox ...
- C# 之 反射性能优化1
反射是一种很重要的技术,然而它与直接调用相比性能要慢很多,因此如何优化反射性能也就成为一个不得不面对的问题. 目前最常见的优化反射性能的方法就是采用委托:用委托的方式调用需要反射调用的方法(或者属性. ...
- C# 之 反射性能优化3
阅读目录 开始 用Delegate优化反射的缺点 用Delegate优化反射的优点 用CodeDOM优化反射的优点 如何用好CodeDOM? 用CodeDOM优化反射的缺点 能不能不使用委托? 根据反 ...
- C# 之 反射性能优化2
问题回顾 在上篇博客中,我介绍了优化反射的第一个步骤:用委托调用代替直接反射调用. 然而,那只是反射优化过程的开始,因为新的问题出现了:如何保存大量的委托? 如果我们将委托保存在字典集合中,会发现这种 ...
- Java学习:注解,反射,动态编译
狂神声明 : 文章均为自己的学习笔记 , 转载一定注明出处 ; 编辑不易 , 防君子不防小人~共勉 ! Java学习:注解,反射,动态编译 Annotation 注解 什么是注解 ? Annotat ...
- .NET反射的优化
写在前面 1)本篇博客并非原创,而是我针对.NET反射相关知识的总结.本篇内容来源汇总于3篇博客.在后面的介绍中会在开头给出对应的链接,方便读者自行学习.2)本篇博客主要针对表达式树代码进行详细讲解. ...
- JAVA的泛型与反射的联合应用
通过泛型与反射的结合,可以编写框架来使开发更容易,这里演示的是BaseDao部分的简单使用. BaseDao部分代码: public abstract class BaseDao<T>{ ...
- 枚举类&&注解&&反射
什么是枚举类? 枚举类是优化定义固定对象的一种特殊的类. 换句话说,在需要类的实例为一个或者多个并且相对固定的时候,使用枚举类.(枚举类可扩展) 类的实例相对来说固定的有日期,客观不变的一些数字等等. ...
随机推荐
- 浅析js绑定同一个事件依次触发问题系列(一)
算了 还是上代码吧 记得写过这篇文章,但是找不到了(对,就是找不到了,算了再写一遍吧) 也是在群中有人问这个绑定不同事件 或者同一个事件的依次触发问题 个人建议如果是一个事件的话那么最好写成函数, ...
- history命令使用方法详解
history是一条非常实用的shell命令,可以显示出之前在shell中运行的命令,配合last显示之前登录的用户,就可以追溯是哪个用户执行了某些命令.以下详细说明history使用中常见的命令或技 ...
- Linux进程间通信(一) - 管道
管道(pipe) 普通的Linux shell都允许重定向,而重定向使用的就是管道. 例如:ps | grep vsftpd .管道是单向的.先进先出的.无结构的.固定大小的字节流,它把一个进程的标准 ...
- SWD下载调试填坑,SWD连接丢失问题解决
野火SWD下载器,设置好以后,第一次下载成功,莫名其妙丢失连接,发现在复位状态可以连接(惊奇) 网络上搜索到把Boot0和Boot1置高,就可以把程序下载到RAM里, 能下载以后就好办了,把程序里SW ...
- 我自己曾经经历的CMMI3认证通过关于软件测试的访谈【转载】
转自<http://blog.csdn.net/jcy58/article/details/51908884> 因为当初我在公司里是负责软件测试工作的,所以CMMI3和测试相关的访谈,就是 ...
- WebApi 中使用 Session
1. 在 Global.asax.cs 文件中加入session支持 protected void Application_Start() { AreaRegistration.RegisterAll ...
- Nginx教程
Nginx教程 1.背景 介绍 Nginx是一个高性能的HTTP服务器,以及反向代理服务器 组成 Ngnix有内核和模块组成.微结构的内核根据配置文件将一个请求映射到一个location块中,该loc ...
- Arrays类--Arrays.asList()方法使用
java.util类 Arrays java.lang.Object——java.util.Arrays public class Arrays extends Object 此类包含用来操作数组(比 ...
- 在jsp中嵌入javascript代码执行对html的影响方式
1 javascript的作用范围 javascript操作的是html dom树. 它可以用来直接写入html标签:修改html的内容:响应事件:修改html中的图像:修改html的样式等等. 2 ...
- 【python】-- pymsql 外键
pymsql 外键 本片是以上一篇pymsql操作MySQL的补充,主要演示pymysql的外键操作使用 一.一对一外键关联 1.示意图 2.一对一外键关联示例 2.1.创建表结构,插入数据 from ...