Java之动态代理类实现日志简单实例

开心一笑

上课呢,同桌撕了一片纸放嘴里了咬了咬,又吐回了手里。
揉了揉,揉成个球状,然后又把这东西给放嘴里,咽下去了。
我问:你干嘛呢?
这二货幽幽地说:这两天有点感冒,捏个药丸吃。

视频教程

大家好,我录制的视频《Java之优雅编程之道》已经在CSDN学院发布了,有兴趣的同学可以购买观看,相信大家一定会收获到很多知识的。谢谢大家的支持……

视频地址:http://edu.csdn.net/lecturer/994

提出问题

Java动态代理类如何实现简单的日志???
简书地址:http://www.jianshu.com/users/d38a3668be58/latest_articles

解决问题

假如你已经有一定的java基础
假如你已经知道什么是动态代理。

1.假如下面是真实的业务类及其实现类

package com.hwy.test;

/**
 * Created by Ay on 2016/7/1.
 */
public interface BusinessClassService {

    public void doSomeThing();
}

实现类:

package com.hwy.test;

/**
 * 业务类
 * Created by Ay on 2016/7/1.
 */
public class BusinessClassServiceImpl implements BusinessClassService{

    /** 执行某事 **/
    public void doSomeThing(){
        System.out.println("do something ......");
    }
}

2.下面是日志类和实现类:

package com.hwy.test;

import java.lang.reflect.Method;

/**
 * 日志类接口
 * Created by Ay on 2016/6/30.
 */
public interface MyLogger {

    /** 纪录进入方法时间 **/
    public void saveIntoMethodTime(Method method);

    /** 纪录退出方法时间**/
    public void saveOutMethodTime(Method method);
}

实现类:

package com.hwy.test;

import java.lang.reflect.Method;

/**
 * 日志类实现
 * Created by Ay on 2016/6/30.
 */
public class MyLoggerImpl implements MyLogger {

    @Override
    public void saveIntoMethodTime(Method method) {
        System.out.println("进入" + method.getName()  +"方法时间为: " + System.currentTimeMillis());
    }

    @Override
    public void saveOutMethodTime(Method method) {
        System.out.println("退出" + method.getName() + "方法时间为:" + System.currentTimeMillis());

    }
}

3.下面是日志类的handler实现:

package com.hwy.test;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

/**
 * Created by Ay on 2016/6/30.
 */
public class MyLoggerHandler implements InvocationHandler {

    /** 原始对象 **/
    private Object objOriginal;

    /** 这里很关键 **/
    private MyLogger myLogger = new MyLoggerImpl();


    public MyLoggerHandler(Object obj){
        super();
        this.objOriginal = obj;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        Object result = null;
        /** 日志类的方法 **/
        myLogger.saveIntoMethodTime(method);
        /** 调用代理类方法 **/
        result = method.invoke(this.objOriginal ,args);
        /** 日志类方法**/
        myLogger.saveOutMethodTime(method);

        return result;
    }



}

4.最后是测试类:

package com.hwy.test;

import java.lang.reflect.Proxy;

/**
 * Created by A on 2016/6/30.
 */
public class MyLoggerTest {


    public static void main(String[] args) {
        /** 实例化真实项目中业务类 **/
        BusinessClassService businessClassService = new BusinessClassServiceImpl();
        /** 日志类的handler **/
        MyLoggerHandler myLoggerHandler = new MyLoggerHandler(businessClassService);
        /** 获得代理类对象 **/
        BusinessClassService businessClass = (BusinessClassService)Proxy.newProxyInstance(businessClassService.getClass().getClassLoader(), businessClassService.getClass().getInterfaces(),myLoggerHandler);
        /** 执行代理类方法 **/
        businessClass.doSomeThing();

    }
}

5.运行MyLoggerTest类

进入doSomeThing方法时间为: 1467326179684
do something ......
退出doSomeThing方法时间为:1467326179685

6.唠叨几句:

1)上面类中,类的初始化,基本可以利用spring配置到配置文件中
2)上面只是一个简单的例子而已

用一张图大致表达下:

这里写图片描述

读书感悟

来自《老男孩》

  • 笑,全世界都陪你笑;哭,只有你一个人流泪。
  • 回答错了问题并不可怕,可怕的是问错了问题,那就永远得不到正确的答案
  • 只剩下麻木的我没有了当年的热血。
  • 不管是沙还是石头,都会沉下去。
anylog 是一个可以在代码的任意区域无入侵地加入日志的工具,适用于线上问题排查。 anylog 为开发人员提供一个易于使用的平台,帮助开发人员在正在运行的系统随时加入自己想要的日志,而免于修改代码和重启。 使用场景举例     1、一些同学在写代码时,把异常吃掉了,使得问题难以查找,可以使用这个工具,动态打印出被吃掉         的异常,而不用停机。     2、一些项目依赖第三方jar包,如果发生问题,但第三方包日志打印,以往可能需要重新编译第         三方包,加上日志,重启服务,然后排查问题。但使用这个工具,就可以直接动态加入日志,而不用         修改第三方jar包,也不用重启。 已有功能     1、让系统打印某个exception的堆栈,无论此exception是否已经被吃掉都可打印     2、在某个指定的某个方法的某一行,输出日志。     3、在某个指定的某个方法的开始,输出日志。     4、在某个指定的某个方法的结束,输出日志。       5、打印方法耗时,支持方法嵌套。     如果需要扩展新的功能(例如输出jvm的cpu占用,内存大小等),只需要实现spi的     com.github.jobop.anylog.spi.TransformDescriptor      和com.github.jobop.anylog.spi.TransformHandler接口,     然后把实现的jar包放到providers目录即可识别。 使用方法     1、获取运行程序:         1)可以到以下地址获取正式发行版:https://github.com/jobop/release/tree/master/anylog         2)你也可以clone下源码后,执行如下命令,生成运行程序,生成的运行程序将在dist目录下             生成windows版本:  mvn install             生成linux版本:  mvn install -Plinux     2、直接执行startup.bat或者startup.sh即可运行起来     3、访问 http://127.0.0.1:52808 即可使用 功能扩展     anylog利用spi机制实现其扩展,如果你想要对anylog增加新的功能(例如添加返回值打印的功能)可以按照如下步骤操作:     1、使用如下命令,生成一个spi实现工程,并导入eclipse     mvn archetype:generate -DarchetypeGroupId=com.github.jobop -DarchetypeArtifactId=anylogspi-archetype -DarchetypeVersion=1.0.4     2、参照该工程已有的两个例子(一个是在方法开始插入日志,一个是在方法结束插入日志),实现TransformDescriptor和TransformHandler接口     3、把两个接口实现的全路径,分别加到以下两个文件         src/main/resources/META-INF/services/com.github.jobop.anylog.spi.TransformDescriptor         src/main/resources/META-INF/services/com.github.jobop.anylog.spi.TransformHandler     4、执行mvn install打包,在dist下会生成你的扩展实现jar。     5、把扩展实现jar拷贝到anylog的providers目录下,重启即可生效。     tips:在实现spi时,我们提供了SpiDesc注解,该注解作用在你实现的TransformDescriptor上,可以用来生成功能描述文字。          如果要深入了解spi机制,请自行google:java spi 标签:anylog
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

阿_毅

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值