详解使用Java原生代理实现AOP实例

发布时间 - 2026-01-10 22:23:17    点击率:

一说到AOP,大家一定会想到spring,因为这东西实在是太强大了.但是大家一定要清楚,AOP是一只编程思想,而Spring仅仅是AOP的一种实现罢了.

首先百度下:

在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

今天呢,咱们就一起用Java原生代理实现简单的AOP功能.

首先,你得需要了解基本的反射知识,否则可能会感到困惑.

不罗嗦了,直接开始撸码

首先,咱们先写一个简单的接口.名字叫AnimalInterface,用来声明规范动物的一些基本方法.

这些方法包括 设置名字,获取名字,叫声,属性(原谅我没文化,其实就是获得是陆栖还是水栖或者水陆两栖)

package proxy.imp;
public interface AnimalInterface {
  //设置名字
  void setName(String name);
  //获取名字
  String getName();
  //叫声
  void say();
  //获取栖性
  void getProperty();
}

然后咱们实现这个接口,创建一个名叫小黑的Dog

package proxy;

import proxy.imp.AnimalInterface;
public class DogImp implements AnimalInterface {
  private String name = "小黑";
  public DogImp() {
  }
  @Override
  public void setName(String name) {
    this.name = name;
  }
  @Override
  public String getName() {
    return this.name;
  }
  @Override
  public void say() {
    System.out.println("小狗:汪汪汪汪.....");
  }
  @Override
  public void getProperty() {
    System.out.println("小狗是陆地动物,但是会游泳哦");
  }
}

大家一定迫不及待了,怎么实现类似AOP的功能呢….

咱们先创建一个名为AOPHandle的类,让其实现InvocationHandler接口,不能使用invoke时使用proxy作为反射参数时,因为代理对象的接口,不同于对象,这种代理机制是面向接口,而不是面向类的,如果使用proxy,会造成无限递归.然后就是栈溢出,但是依旧能反射成功一次,这说明代理对象和对象的代理是不一样的,但是咱们可以通过proxy参数的proxy.getClass()获得class对象,然后获得被代理类的方法和参数,这也为注解注入,特定方法注入,属性注入提供了一种实现途径吧,关于这个,咱们后面再说..

package proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class AOPHandle implements InvocationHandler{
  //保存对象
  private Object o;
  public AOPHandle(Object o) {
    this.o=o;
  }
  /**
   * 这个方法会自动调用,Java动态代理机制
   * 会传入下面是个参数
   * @param Object proxy 代理对象的接口,不同于对象
   * @param Method method 被调用方法
   * @param Object[] args 方法参数
   * 不能使用invoke时使用proxy作为反射参数时,因为代理对象的接口,不同于对象
   * 这种代理机制是面向接口,而不是面向类的
   **/
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    //方法返回值
    Object ret=null;
    //打印方法名称
    System.err.println("执行方法:"+method.getName()+"n参数类型为:");
    //打印参数
    for(Class type:method.getParameterTypes())
      System.err.println(type.getName());
    //打印返回类型
    System.err.println("返回数据类型:"+method.getReturnType().getName());
    //反射调用方法
    ret=method.invoke(o, args);
    //声明结束
    System.err.println("方法执行结束");
    //返回反射调用方法的返回值
    return ret;
  }
}

动态代理已经搞定..然后就是咱们的AnimalFactory了..咱们继续

package proxy;
import java.lang.reflect.Proxy;
public class AnimalFactory {
  /***
   * 获取对象方法
   * @param obj
   * @return
   */
  private static Object getAnimalBase(Object obj){
    //获取代理对象
    return Proxy.newProxyInstance(obj.getClass().getClassLoader(),
        obj.getClass().getInterfaces(), new AOPHandle(obj));
  }
  /***
   * 获取对象方法
   * @param obj
   * @return
   */
  @SuppressWarnings("unchecked")
  public static T getAnimal(Object obj){
    return (T) getAnimalBase(obj);
  }
  /***
   * 获取对象方法
   * @param className
   * @return
   */
  @SuppressWarnings("unchecked")
  public static  T getAnimal(String className){
    Object obj=null;
    try {
      obj= getAnimalBase(Class.forName(className).newInstance());
    } catch (Exception e) {
      e.printStackTrace();
    }
    return (T)obj;
  }
  /***
   * 获取对象方法
   * @param clz
   * @return
   */
  @SuppressWarnings("unchecked")
  public static  T getAnimal(Class clz){
    Object obj=null;
    try {
      obj= getAnimalBase(clz.newInstance());
    } catch (Exception e) {
      e.printStackTrace();
    }
    return (T)obj;
  }
}

终于到最后了…还差什么呢,大家来这里看看效果吧…

哈哈…小二,上个菜..哦~不对,是个测试类..哈哈////

package proxy;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.BlockJUnit4ClassRunner;
import proxy.AnimalFactory;
import proxy.imp.AnimalInterface;

@RunWith(BlockJUnit4ClassRunner.class)
public class AOPTest {

  @Test
  public void Test1() {
    AnimalInterface dog=AnimalFactory.getAnimal(DogImp.class);
    dog.say();
    System.out.println("我的名字是"+dog.getName());
    dog.setName("二狗子");
    System.out.println("我的名字是"+dog.getName());
  }
}

啥?什么,,到了最后说,,这又卵用,这不是坑爹么?就捕获一个这个玩意,什么用啊…

什么AOP,我怎么一点AOP的影子都没有看到,怎么切入自定义方法,就一个syso输入,往这忽悠观众来了?…..

好吧,那咱们继续…看看如何实现注入自定义方法…

首先增加一个接口,咱们就称为AOP注入接口吧.取名AOPMethod哈

创建after和before方法,接收Object proxy, Method method, Object[] args参数

这样就能做更多的事情叻…比如执行方法前,记录类状态,写入log.监控xx变量,,,

开启你的脑洞吧.

package proxy.imp;
import java.lang.reflect.Method;

public interface AOPMethod{
  //实例方法执行前执行的方法
  void after(Object proxy, Method method, Object[] args);
  //实例方法执行后执行的方法
  void before(Object proxy, Method method, Object[] args);
}

然后修改AOPHandle类,增加AOPMethod属性.

在修改构造方法,让在类初始化时获得AOPMethod实例.

最后修改invoke方法….直接上代码哈

package proxy;

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

import proxy.imp.AOPMethod;

public class AOPHandle implements InvocationHandler{
  //保存对象
  private AOPMethod method;
  private Object o;
  public AOPHandle(Object o,AOPMethod method) {
    this.o=o;
    this.method=method;
  }
  /**
   * 这个方法会自动调用,Java动态代理机制
   * 会传入下面是个参数
   * @param Object proxy 代理对象的接口,不同于对象
   * @param Method method 被调用方法
   * @param Object[] args 方法参数
   * 不能使用invoke时使用proxy作为反射参数时,因为代理对象的接口,不同于对象
   * 这种代理机制是面向接口,而不是面向类的
   **/
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    Object ret=null;
    //修改的地方在这里哦
    this.method.before(proxy, method, args);
    ret=method.invoke(o, args);
    //修改的地方在这里哦
    this.method.after(proxy, method, args);
    return ret;
  }
}

呼呼,大功告成,,看起来一切都么问题,萌萌哒..

赶紧更新下测试类…

package proxy;

import java.lang.reflect.Method;

import org.junit.runner.RunWith;
import org.junit.runners.BlockJUnit4ClassRunner;
import proxy.imp.AOPMethod;
import proxy.imp.AnimalInterface;

@RunWith(BlockJUnit4ClassRunner.class)
public class AOPTest {

  public static void main(String[] args) {

    AnimalInterface dog = AnimalFactory.getAnimal(DogImp.class, new AOPMethod() {
      // 这里写方法执行前的AOP切入方法
      public void before(Object proxy, Method method, Object[] args) {
        System.err.println("我在" + method.getName() + "方法执行前执行");
      }

      // 这里系方法执行后的AOP切入方法
      public void after(Object proxy, Method method, Object[] args) {
        System.err.println("我在 " + method.getName() + "方法执行后执行");

      }
    });
    dog.say();
    String name1="我的名字是" + dog.getName();
    System.out.println(name1);
    dog.setName("二狗子");
    String name2="我的名字是"+dog.getName();
    System.out.println(name2);
  }
}

呼呼,亲们,是不是有注入的感觉了?是不是感觉把自己的方法切进去了???哈哈….

看起来一切都已经完美了,但是总觉得差了点什么?哦,对,缺少了类似于Spring那么样的配置文件..

其实那些已经很简单了,交给你们去做吧,设计好XML格式化就妥了,等等,你说什么,还不能拦截自定义方法?

不能像Spring那样拦截自定义方法?oh~~NO,其实已经可以了在before(Object proxy, Method method, Object[] args)中利用method和的给methodName就能做判断了.

当然,本例的并没有什么实用意义,更不能个各种完善的AOP框架相比,本文仅仅为您提供一种思路,但是一定要记住,再牛的东西也是一点点积累出来的

实例下载:http://xiazai./201701/yuanma/JavaAOP_jb51.rar

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# java  # aop实现  # aop动态代理  # aop实现原理  # java基于spring注解AOP的异常处理的方法  # Java之Spring AOP 实现用户权限验证  # Java AOP知识详细介绍  # Java实现AOP面向切面编程的实例教程  # 实例讲解Java的Spring框架中的AOP实现  # Java的Spring框架中AOP项目的一般配置和部署教程  # 举例讲解Java的Spring框架中AOP程序设计方式的使用  # Java的Spring框架下的AOP编程模式示例  # Java动态代理实现AOP  # 详解Java反射实现Aop代理  # 自定义  # 是个  # 在这里  # 我在  # 一切都  # 而不是  # 能做  # 递归  # 创建一个  # 自己的  # 返回值  # 法会  # 有什么  # 来了  # 你说  # 说到  # 这也  # 可以通过  # 我没  # 很简单 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 零基础网站服务器架设实战:轻量应用与域名解析配置指南  郑州企业网站制作公司,郑州招聘网站有哪些?  如何用VPS主机快速搭建个人网站?  Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  Python进程池调度策略_任务分发说明【指导】  佛山企业网站制作公司有哪些,沟通100网上服务官网?  Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案  Laravel安装步骤详细教程_Laravel环境搭建指南  Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧  Laravel如何保护应用免受CSRF攻击?(原理和示例)  Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言  Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】  如何快速选择适合个人网站的云服务器配置?  如何快速搭建个人网站并优化SEO?  免费视频制作网站,更新又快又好的免费电影网站?  历史网站制作软件,华为如何找回被删除的网站?  iOS正则表达式验证手机号、邮箱、身份证号等  无锡营销型网站制作公司,无锡网选车牌流程?  如何在万网利用已有域名快速建站?  网页制作模板网站推荐,网页设计海报之类的素材哪里好?  ChatGPT 4.0官网入口地址 ChatGPT在线体验官网  怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?  Laravel怎么定时执行任务_Laravel任务调度器Schedule配置与Cron设置【教程】  如何在阿里云部署织梦网站?  如何续费美橙建站之星域名及服务?  高端建站如何打造兼具美学与转化的品牌官网?  在线制作视频的网站有哪些,电脑如何制作视频短片?  JavaScript如何实现类型判断_typeof和instanceof有什么区别  千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】  Win11搜索不到蓝牙耳机怎么办 Win11蓝牙驱动更新修复【详解】  javascript事件捕获机制【深入分析IE和DOM中的事件模型】  Laravel如何生成URL和重定向?(路由助手函数)  米侠浏览器网页图片不显示怎么办 米侠图片加载修复  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  大学网站设计制作软件有哪些,如何将网站制作成自己app?  如何在搬瓦工VPS快速搭建网站?  如何正确下载安装西数主机建站助手?  Laravel Artisan命令怎么自定义_创建自己的Laravel命令行工具完全指南  美食网站链接制作教程视频,哪个教做美食的网站比较专业点?  如何用AI帮你把自己的生活经历写成一个有趣的故事?  湖南网站制作公司,湖南上善若水科技有限公司做什么的?  如何用y主机助手快速搭建网站?  PHP正则匹配日期和时间(时间戳转换)的实例代码  敲碗10年!Mac系列传将迎来「触控与联网」双革新  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  Laravel集合Collection怎么用_Laravel集合常用函数详解  Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全  香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南  如何快速搭建自助建站会员专属系统?