注解,反射的简单应用

java的注解和反射学了一直没有用到,主要是框架帮我们封装好了,那就好了,仿写框架就可以开拓新思路
仿mybatis写一个简易的orm框架

注解 (sql注解)

/**
 * @author
 * sql操作
*运行时检查
*注解标记的位置,在方法上
 * */
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Select {

    String value();

}

mapper接口

/**
 * @author cainiao
 * */
public interface UserMapper {

    @Select(value = "select * from user where id = #{id}")
    User select(int id);

    @Insert(value = "insert into user () values ()")
    void insert(User user);

    void update(User user);

    void delete(int id);
}

jdbc的工具类,为了简单,就不用数据库连接池

/**
 * @author 连接数据库
 */
public class JDBCConfig {

    private static String url = null;
    private static String driver = null;
    private static String username = null;
    private static String password = null;
    private static final String file = "db.properties";
    static {
        try {
            InputStream input = JDBCConfig.class.getClassLoader().getResourceAsStream(file);
            Properties properties = new Properties();
            properties.load(input);
            driver = properties.getProperty("driver");
            url = properties.getProperty("url");
            username = properties.getProperty("username");
            password = properties.getProperty("password");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static Connection getConnection() throws Exception{
        Class.forName(driver);
        return DriverManager.getConnection(url, username, password);
    }
    public static void getClose(Connection conn, PreparedStatement ps, ResultSet rs){
        if(rs!=null){
            try {
                rs.close();
            } catch (SQLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }finally{
                rs=null;
            }
        }
        if(ps!=null){
            try {
                ps.close();
            } catch (SQLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }finally{
                ps = null;
            }
        }
        if(conn!=null){
            try {
                conn.close();
            } catch (SQLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }finally{
                conn = null;
            }
        }
    }
}

包装数据的工具类

/**
 * @author cainiao
 * 封装返回数据
 */
public class MapperUtil {

    //增 删 改
    public static int update(String sql,Object...values) {
        Connection conn = null;
        PreparedStatement ps = null;
        ResultSet rs = null;
        int rows = -1 ;
        try {
            conn = JDBCConfig.getConnection();
            ps = conn.prepareStatement(sql);
            int len = values.length;
            for(int i = 0;i<len;i++){
                ps.setObject(i+1, values[i]);
            }
            rows = ps.executeUpdate();
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }finally{
            JDBCConfig.getClose(conn, ps,rs);
            return rows;
        }
    }
    /**
     * 查询
     * */
    public static <T> List<T> query(Class<T> clazz, String sql, Object[] values) {
        Connection conn = null;
        PreparedStatement ps = null;
        ResultSet rs = null;
        List<T> beans = new ArrayList<>();
        try {
            conn = JDBCConfig.getConnection();
            ps = conn.prepareStatement(sql);
            int valueCount = values.length;
            for (int i = 0; i < valueCount; i++) {
                ps.setObject(i + 1, values[i]);
            }
            rs = ps.executeQuery();
            ResultSetMetaData meta = rs.getMetaData();
            int colum = meta.getColumnCount();
            while (rs.next()) {
                T bean = clazz.newInstance();
                for (int i = 0; i < colum; i++) {
                    String label = meta.getColumnName(i + 1);
                    Field field = bean.getClass().getDeclaredField(label);
                    field.setAccessible(true);
                    Object value = rs.getObject(label);
                    field.set(bean, value);
                }
                beans.add(bean);
            }
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } finally {
            JDBCConfig.getClose(conn, ps, rs);
        }
        return beans;
    }
    public static void main(String[] args) {
        ArrayList list = (ArrayList) MapperUtil.query(User.class,"select * from user where id = ?",new Object[]{19});
        for (Object object:list){
            System.out.println((User)object);
        }
    }
}

简单的java对象

@Table(value = "user")
public class User {

    @Property(value = "id")
    private int id;
    private String username;
    private String password;
    private String salt;
    private int state;
    private String email;
    private String phone;
    private Date created;
    private Date modified;
}
<!--  setter和getter方法 -->

mapper接口没有实现类,要实现其功能需要用到jdk动态代理,这里代理不是做切面用,而是拿到接口的方法的各种参数
jdk动态代理

/**
 * @author jdk动态代理
 */
public class MapperProxy<T> implements InvocationHandler {

    private Class<T> target;
    /**
*存放截取过程中的sql片段
*/
    static List<String> list = new ArrayList<>();

    public MapperProxy(Class<T> target) {
        this.target = target;
    }

    /**
     * 拿到接口类
     * 代理类
     * 调用的方法
     * 方法的参数值
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //获取返回值类型
        Type type = method.getGenericReturnType();
        //获取参数类型
        Type[] types = method.getGenericParameterTypes();
        //变量名
        Parameter[] parameters = method.getParameters();
        //返回信息
        Object object = null;
        Annotation[] annotations = method.getDeclaredAnnotations();
        for (Annotation annotation : annotations) {
            String name = annotation.annotationType().getSimpleName();
            String sql = null;
            switch (name) {
                case "Select":
                    sql = method.getDeclaredAnnotation(Select.class).value();
                    object = select(sql, type, args, parameters);
                    break;
                case "Insert":
                    sql = method.getDeclaredAnnotation(Insert.class).value();
                    break;
                case "Update":
                    sql = method.getDeclaredAnnotation(Select.class).value();
                    break;
                case "Delete":
                    sql = method.getDeclaredAnnotation(Select.class).value();
                    break;
                default:
            }
        }
        return object;
    }

    /**
     * 查询方法
     */
    private Object select(String sql, Type type, Object[] args, Parameter[] parameters) throws ClassNotFoundException {
        Class<?> clazz = Class.forName("com.orm.pojo.User");
        StringBuilder sb = new StringBuilder();
        jiequ(sql);
        for (int i = 0; i < args.length; i++) {
            sb.append(list.get(i));
        }
        System.out.println(sb);
        //转为string ,不转会越界报错
        sql = sb + "";
        //执行sql
        List objects = MapperUtil.query(clazz, sql, args);
        if (objects.size() == 1) {
            try {
                return objects.get(0);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return objects;
    }

    private void insert(Type type, Type[] types, Object[] args) {

    }

    private void update(Type type, Type[] types, Object[] args) {

    }

    private void delete(Type type, Type[] types, Object[] args) {

    }

    /**
     * sql截取
     */
    public void jiequ(String sql) {
        if (sql.contains("#")) {
            String[] sqls = sql.split("and", 2);
            for (int i = 0; i < sqls.length; i++) {
                if (i == 0) {
                    String[] temps = sqls[i].split("\\#");
                    for (int j = 0; j < sqls.length; j++) {
                        if (j == 0) {
                            list.add(temps[j] + "?");
                        }
                    }
                } else {
                    jiequ(sqls[i]);
                }
            }
        }
    }

    public static void main(String[] args) {

    }

}

这里stringbuilder必须先转为String,不然prepareStatement会报越界错误
创建代理的类

/**
 * @author
 * */
public class MapperFactory {

    public static <T> T newInstance(Class<T> target){
        MapperProxy mapperProxy = new MapperProxy<T>(target);
        return (T) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
                new Class[]{target},mapperProxy);
    }
}

测试类

/**
 * @author
 * */
public class UserService {
    /**
     * 注入mapper
     * */
    private UserMapper userMapper = MapperFactory.newInstance(UserMapper.class);

    public void get(int id){
        User user = userMapper.select(id);
        System.out.println(user);
    }

    public static void main(String[] args) {
        UserService userService = new UserService();
        userService.get(19);
    }
}

测试


aaaa.png

还有不足,引用pojo类没有动态的获取

github:https://github.com/cailonghao/Imitation-writing

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