前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >系统学习javaweb-09-javaweb基础增强

系统学习javaweb-09-javaweb基础增强

作者头像
csxiaoyao
发布2019-02-20 16:04:17
4650
发布2019-02-20 16:04:17
举报
文章被收录于专栏:csxiaoyaocsxiaoyao

名称:javaweb基础增强 内容:反射泛型、注解、日志以及利用反射泛型和注解开发自定义框架(模拟Struts)

代码结构

【package1】:com.csxiaoyao.study 使用注解对知识点1的BaseDao(所有dao的公用的方法)代码优化 【package2】:com.csxiaoyao.utils 注解工具类 【package其他】 自定义框架实现 【配置文件1】:log4j.properties 日志文件配置 【配置文件2】:mystruts.xml 自定义框架配置

【知识点】

1 反射泛型

BaseDao.java

/**
 * 所有dao的公用的方法,都在这里实现
 */
public class BaseDao<T>{
    // 保存当前运行类的参数化类型中的实际的类型
    private Class clazz;
    // 表名
    private String tableName;
    // 构造函数: 1. 获取当前运行类的参数化类型; 2. 获取参数化类型中实际类型的定义(class)
    public BaseDao(){
        //  this  表示当前运行类  (AccountDao/AdminDao)
        //  this.getClass()  当前运行类的字节码(AccountDao.class/AdminDao.class)
        //  this.getClass().getGenericSuperclass();  当前运行类的父类,即为BaseDao<Account>
        //                                           其实就是“参数化类型”, ParameterizedType   
        Type type = this.getClass().getGenericSuperclass();
        // 强制转换为“参数化类型”  【BaseDao<Account>】
        ParameterizedType pt = (ParameterizedType) type;
        // 获取参数化类型中,实际类型的定义  【new Type[]{Account.class}】
        Type types[] =  pt.getActualTypeArguments();
        // 获取数据的第一个元素:Accout.class
        clazz = (Class) types[0];
        // 表名  (与类名一样,只要获取类名就可以)
        tableName = clazz.getSimpleName();
    }
    /**
     * 主键查询
     * @param id    主键值
     * @return      返回封装后的对象
     */
    public T findById(int id){
        /*
         * 1. 知道封装的对象的类型
         * 2. 表名【表名与对象名称一样, 且主键都为id】
         * 
         * 即,
         *    ---》得到当前运行类继承的父类  BaseDao<Account>
         *   ----》 得到Account.class
         */
        String sql = "select * from " + tableName + " where id=? ";
        try {
            //自定义的连接池工具类
            return JdbcUtils.getQuerrRunner().query(sql, new BeanHandler<T>(clazz), id);
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
    }
    /**
     * 查询全部
     * @return
     */
    public List<T> getAll(){
        String sql = "select * from " + tableName ;
        try {
            return JdbcUtils.getQuerrRunner().query(sql, new BeanListHandler<T>(clazz));
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
    }
}

AccountDao.java

public class AccountDao extends BaseDao<Account> {
    // 只需要写父类没有实现的方法(个性化需求)
}

2 注解

2.1 常用的注解

// 重写父类的方法
@Override
public String toString() {
    return super.toString();
}
// 抑制编译器警告
@SuppressWarnings({"unused","unchecked"})
private void save() {
    List list = null;
}
// 标记方法以及过时
@Deprecated
private void save1() {
}

2.2 自定义注解

public @interface Author {
    /**
     * 注解属性
     *  1. 修饰为默认或public
     *    2. 不能有主体
     */
    String name();
    int age() default 30;   // 带默认值的注解;  使用的时候可以不写此属性值
}

使用

@Author(name = "sun", age = 30)
public void save() {

}

2.3 默认名称的注解

  1. 注解属性默认名称为value
public @interface Author {
    // 如果注解名称为value,使用时候可以省略名称,直接给值
    // (且注解只有一个属性时候才可以省略名称)
    String value();
}

使用

@Author("sun")
@Author(value = "sun")
  1. 注解属性类型为数组
public @interface Author {
    String[] value() default {"test1","test2"};
}

使用

@Author({"",""})
public void save() {
}

2.4 元注解

元注解,表示注解的注解

// 1. 指定注解的可用范围
@Target({
    TYPE,     类
    FIELD,     字段
    METHOD,  方法
    PARAMETER,   参数
    CONSTRUCTOR, 构造器
    LOCAL_VARIABLE  局部变量
})
// 2. 指定注解的声明周期
@Retention(RetentionPolicy.SOURCE)    注解只在源码级别有效
@Retention(RetentionPolicy.CLASS)      注解在字节码级别有默认值
@Retention(RetentionPolicy.RUNTIME)    注解在运行时期有效

2.5 注解反射

// 获取注解信息: name/age/remark
@Id
@Author(remark = "保存信息!!!", age = 19)
public void save() throws Exception {   
    // 1. 先获取代表方法的Method类型;
    Class clazz = App_1.class;
    Method m = clazz.getMethod("save");
    // 2. 再获取方法上的注解
    Author author = m.getAnnotation(Author.class);
    // 获取输出注解信息
    System.out.println(author.authorName());
    System.out.println(author.age());
    System.out.println(author.remark());
}

2.6 注解优化BaseDao代码

当表名与数据库名称不一致,字段与属性不一样,主键不叫id,上面的BaseDao失效,具体写法见代码

3 log4j

# 通过根元素指定日志输出的级别、目的地: 
# 日志输出优先级: debug < info < warn < error 
log4j.rootLogger=info,console,file
############# 日志输出到控制台 #############
# 日志输出到控制台使用的api类
log4j.appender.console=org.apache.log4j.ConsoleAppender
# 指定日志输出的格式: 灵活的格式
log4j.appender.console.layout=org.apache.log4j.PatternLayout
# 具体格式内容
log4j.appender.console.layout.ConversionPattern=%d %p %c.%M()-%m%n
############# 日志输出到文件 #############
log4j.appender.file=org.apache.log4j.RollingFileAppender
# 文件参数: 指定日志文件路径
log4j.appender.file.File=../logs/MyLog.log
# 文件参数: 指定日志文件最大大小
log4j.appender.file.MaxFileSize=5kb
# 文件参数: 指定产生日志文件的最大数目
log4j.appender.file.MaxBackupIndex=100
# 日志格式
log4j.appender.file.layout=org.apache.log4j.PatternLayout
log4j.appender.file.layout.ConversionPattern=%d %c.%M()-%m%n

4 应用:自定义框架

详见代码

5 Struts框架

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2017年05月09日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 代码结构
  • 【知识点】
  • 1 反射泛型
  • 2 注解
    • 2.1 常用的注解
      • 2.2 自定义注解
        • 2.3 默认名称的注解
          • 2.4 元注解
            • 2.5 注解反射
              • 2.6 注解优化BaseDao代码
              • 3 log4j
              • 4 应用:自定义框架
              • 5 Struts框架
              领券
              问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档