个人技术分享

1.日志:

日志的优势:

体系结构:

日志规范:

        ·日志规范大多是一些接口,提供给实现框架去设计的

        ·常见的规范是Commons Logging(JCL)、Simple Logging Facade for Java(slf4j)

常见日志框架:

        ·Log4j

        ·Logback(性能更好,广泛使用,重点)

Logback:

        使用Logback需要使用三个模块:

        ·slf4j-api:日志规范

        ·logback-core:基础模块

        ·logback-classic:功能模块,它完整实现了slf4j API

导包:

        需导入上述三个包slf4j-api、logback-core、logback-classic

        导包官方网站:Logback Home (qos.ch)SLF4J

代码演示:
测试类:
public class LogbackDemo1 {

    //获取日志对象
    public static final Logger LOGGER = LoggerFactory.getLogger("LogbackDemo1.class");

    public static void main(String[] args) {

        Scanner sc = new Scanner(System.in);
        System.out.println("请输入用户名");
        String username = sc.nextLine();
        System.out.println("请输入密码");
        String password = sc.nextLine();
        if("han".equals(username) && "123".equals(password)) {
            LOGGER.info("用户登录成功,用户名为:" + username + " 密码为:" + password);
            System.out.println("登录成功");
        } else {
            System.out.println("登录失败");
            LOGGER.info("用户登录失败");
        }

    }

}
xml配置文件:
<?xml version="1.0" encoding="UTF-8"?>
<configuration>
    <!--
        CONSOLE :表示当前的日志信息是可以输出到控制台的。
    -->
    <appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender">
        <!--输出流对象 默认 System.out 改为 System.err-->
        <target>System.out</target>
        <encoder>
            <!--格式化输出:%d表示日期,%thread表示线程名,%-5level:级别从左显示5个字符宽度
                %msg:日志消息,%n是换行符-->
            <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%-5level]  %c [%thread] : %msg%n</pattern>
        </encoder>
    </appender>

    <!-- File是输出的方向通向文件的 -->
    <appender name="FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <encoder>
            <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
            <charset>utf-8</charset>
        </encoder>
        <!--日志输出路径-->
        <file>D:/Java/JavaSEbase/logback/han-data.log</file>
        <!--指定日志文件拆分和压缩规则-->
        <rollingPolicy
                class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy">
            <!--通过指定压缩文件名称,来确定分割文件方式-->
            <fileNamePattern>D:/Java/JavaSEbase/logback/han-data-%d{yyyy-MMdd}.log%i.gz</fileNamePattern>
            <!--文件拆分大小-->
            <maxFileSize>1MB</maxFileSize>
        </rollingPolicy>
    </appender>

    <!--

    level:用来设置打印级别,大小写无关:TRACE, DEBUG, INFO, WARN, ERROR, ALL 和 OFF
   , 默认debug
    <root>可以包含零个或多个<appender-ref>元素,标识这个输出位置将会被本日志级别控制。
    -->
    <root level="info">
        <appender-ref ref="CONSOLE"/>
        <appender-ref ref="FILE" />
    </root>
</configuration>
运行结果:

xml配置文件:

日志级别:
作用:

        用于控制系统中哪些日志级别是可以输出的

级别介绍:

        级别从小到大依次是:TRACE<DEBUG<INFO<WARN<ERROR

        默认级别是debug(忽略大小写),只输出不低于当前级别的日志

        ALL和OFF分别是打开全部日志和关闭全部日志

        

2.类加载器:

功能:

        类加载器负责将.class文件(存储的物理文件)加载到内存中

类加载到内存的时机

        1.创建类的实例(对象)

        2.调用类的静态方法

        3.访问类或者接口的类变量(静态),或者为该类变量赋值

        4.使用反射方式来强制创建某个类或接口对应的java.lang.Class对象

        5.初始化某个类的子类

        6.直接使用java.exe命令来运行某个主类

        总结:用到就加载,不用不加载。

类加载的过程:

加载:

验证:

准备:

解析:

初始化:

类加载器的分类:

双亲委派模型:

        每个加载器得到任务就会往上一层委托其他加载器,当最上层加载器无法完成,则又会一层一层往下返回。

        自定义类加载器一般不会使用。

常用方法:

3.单元测试:

        单元测试就是针对最小的功能单元编写测试代码,Java程序最小的功能单元是方法,因此,单元测试就是针对Java方法的测试,进而检查方法的正确性。

Junit单元测试框架

Junit优点:

注意点:

        该测试方法必须是公共的无参数无返回值的非静态方法

使用步骤:

        在实际开发中,如果想要测试一个方法是否正确,并不是直接在当前方法的上面写@Test的,而是自己独立编写一个测试类(不要写main方法),在这个类中,编写一些方法,在方法里面调用要被测试的方法即可。

常用注解:

        @Test:测试方法

        @Before:用来修饰实例方法,该方法会在每一个测试方法执行之前执行一次

        @After:用来修饰实例方法,该方法会在每一个测试方法执行之后执行一次

练习:

        需求:测试File类中的demete方法是否书写正确

        先在当前模块下创建一个文件,存储以下元素

        

代码演示:
public class JunitDemo2 {

    //注意不能污染数据
    //测试完文件不能少,也不能多出来其他的文件

    //1.用Before初始化数据
    //2.用Test测试方法
    //3.用After还原数据

    @Before
    public void beforeMethod() throws IOException {
        //拷贝文件
        File src = new File("C:\\Users\\Han\\IdeaProjects\\myjunit\\src\\a.txt");
        File dest = new File("C:\\Users\\Han\\IdeaProjects\\myjunit\\src\\copy.txt");

        FileInputStream fis = new FileInputStream(src);
        FileOutputStream fos = new FileOutputStream(dest);
        int b;
        while((b = fis.read()) != -1) {
            fos.write(b);
        }

        fos.close();
        fis.close();

    }

    @Test
    public void method() {
        File file = new File("C:\\Users\\Han\\IdeaProjects\\myjunit\\src\\a.txt");
        boolean delete = file.delete();

        boolean exists = file.exists();

        //断言
        //参数一:出现错误后提示的信息
        //参数二:实际的解果
        //参数三:预期的结果
        Assert.assertEquals("delute方法出错了",delete,true);
        Assert.assertEquals("delute方法出错了",exists,false);

    }

    @After
    public void afterMethod() throws IOException {
        //恢复文件
        File src = new File("C:\\Users\\Han\\IdeaProjects\\myjunit\\src\\copy.txt");
        File dest = new File("C:\\Users\\Han\\IdeaProjects\\myjunit\\src\\a.txt");

        FileInputStream fis = new FileInputStream(src);
        FileOutputStream fos = new FileOutputStream(dest);
        int b;
        while((b = fis.read()) != -1) {
            fos.write(b);
        }

        fos.close();
        fis.close();
    }


}
运行结果:

扩展:

        默认情况下,相对路径是相对项目的,而在单元测试中,相对路径是相对当前模块的。

注解:

        Annotation表示注解,注解是给编译器或JVM看的,它们可以根据注解来完成对应的功能

注释和注解的区别:

共同点:都可以对程序进行解释说明。

不同点:注释,是给程序员看的。只在Java中有效。在class文件中不存在注释的。

当编译之后,会进行注释擦除。

注解,是给虚拟机看的。当虚拟机看到注解之后,就知道要做什么事情了。

作用:

        对我们的程序进行标注,通过注解可以给类增加额外的信息

Java已提供的注解:

        @Override:表示重写的方法

        @Deprecated:表示已过时的方法

        @SuppressWarnings("all"):压制所有警告信息(黄色波浪线)

自定义注解方法:

特殊属性:

        value:

        当注解中只有"一个属性",并且属性名是"value",使用注解时,可以省略value属性名

代码演示:

//注解的定义
public @interface Anno2 {
    public String value();

    public int age() default 23;
}

//注解的使用
@Anno2("123")
public class AnnoDemo2 {

    @Anno2("123")
    public void method(){

    }
}

元注解:

        可以写在注解上面的注解

        @Target :指定注解能在哪里使用

        @Retention :可以理解为保留时间(生命周期)