泛型,注解,反射配合优化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>{ ...
- 枚举类&&注解&&反射
什么是枚举类? 枚举类是优化定义固定对象的一种特殊的类. 换句话说,在需要类的实例为一个或者多个并且相对固定的时候,使用枚举类.(枚举类可扩展) 类的实例相对来说固定的有日期,客观不变的一些数字等等. ...
随机推荐
- Template Method模式
模版方法模式,实际上就是指子类做方法实现,父类做算法实现. 通常情况下,子类继承父类,我们是站在子类的视角上来看父类的,目的不外乎下面三个 1,子类继承父类的方法 2,通过子类来增加方法,实现新的功能 ...
- laravel学习之路5缓存
redis需要先安装 需要通过 Composer 安装 predis/predis 扩展包 (~1.0) 或者使用 PECL 安装 PhpRedis PHP 拓展. composer require ...
- 文本信息检索——布尔模型和TF-IDF模型
文本信息检索--布尔模型和TF-IDF模型 1. 布尔模型 如要检索"布尔检索"或"概率检索"但不包括"向量检索"方面的文档,其相应的查 ...
- 《转》PyQt4 精彩实例分析* 实例2 标准对话框的使用
和大多数操作系统一样,Windows及Linux都提供了一系列的标准对话框,如文件选择,字体选择,颜色选择等,这些标准对话框为应用程序提供了一致的观感.Qt对这些标准对话框都定义了相关的类.这些类让使 ...
- jQuery Easy UI Draggable(拖动)组件
上文已经提到过了 jQuery EasyUI插件引用一般我们经常使用的有两种方式(排除easyload载入方式),所以本篇要总结的Draggable组件相同有两种方式载入: (1).使用class载入 ...
- 驱动程序分层分离概念_总线驱动设备模型_P
分层概念: 驱动程序向上注册的原理: 比如:输入子程序一个input.c作为一层,下层为Dev.c和Dir.c,分别编写Dev.c和Dir.c向上Input.c注册:如图所示 分离概念: 分离概念主要 ...
- 【BZOJ2422】Times 树状数组
[BZOJ2422]Times Description 小y作为一名资深的dotaer,对视野的控制有着深刻的研究.每个单位在一段特定的时间内会出现在小y的视野内,除此之外的时间都在小y看不到的地方. ...
- 【BZOJ2780】[Spoj]8093 Sevenk Love Oimaster 广义后缀自动机
[BZOJ2780][Spoj]8093 Sevenk Love Oimaster Description Oimaster and sevenk love each other. But r ...
- vue路由vue-route
首先先引入插件 <script src="Vue.js"></script> //vue.js在前面 <script src="vue-ro ...
- RedisTemplate访问Redis数据结构(介绍和常用命令)
Redis 数据结构简介 Redis 可以存储键与5种不同数据结构类型之间的映射,这5种数据结构类型分别为String(字符串).List(列表).Set(集合).Hash(散列)和 Zset(有序集 ...