代理模式深度解析

目录

一 静态代理

1.1 优点

1.2 缺点

1.3 适用场景

 二 JDK动态代理

1 JDK动态代理的工作原理

1.1 创建代理类

1.2 加载代理类

1.3 实现方法调用

2. Proxy.newProxyInstance() 的核心工作流程

方法签名

工作步骤

3. 代理类的生成与加载

3.1 代理类生成的关键方法

Proxy.getProxyClass()

ProxyGenerator.generateProxyClass()

3.2 代理类的加载

代理类是如何加载的?

代理类加载的时机

三 CGLIB动态代理

1. 代理方式

2. 核心组件

二、字节码生成

1. 动态生成原理

2. 性能优化

3. 示例:生成的字节码文件(伪代码)

三、类加载与代理对象创建

1. 类加载流程

2. 突破限制

四、与 JDK 动态代理的关键区别

五、典型限制与解决方案

1. 无法代理 final 方法/类

2. 构造函数调用问题

六、应用场景


一 静态代理

静态代理就是手动编写代理类,在编译期就确定代理关系,并让代理类和目标类实现相同的接口。代理类通过调用目标类的方法来完成任务,同时可以在调用前后添加一些额外的操作。

1.1 优点

  1. 简单直观:代码结构清晰,易于理解和实现。
  2. 无侵入性:无需修改目标类代码,通过代理类实现功能增强。
  3. 性能好:代理逻辑在编译期确定,没有动态生成的额外开销。

1.2 缺点

  1. 冗余代码:每个目标类都需要手动编写对应的代理类,代码量大。
  2. 灵活性差:代理关系在编译期固定,无法动态切换代理逻辑。
  3. 接口依赖:若目标类没有实现接口,则无法使用静态代理(需通过继承实现,类似CGLIB的思路,但需要手动编写)。

1.3 适用场景

  • 需要代理的类数量较少。
  • 代理逻辑简单且无需频繁变更。
  • 对性能要求高,避免动态代理的开销。

public interface BaseSimpleService {public void save(Object obj);}
@Slf4j
public class BaseSimpleServiceImpl implements BaseSimpleService {@Overridepublic void save(Object obj) {System.out.println("对象保存成功");}}
/*** 对 BaseSimpleService类做一个增强,在不侵入原业务代码的基础上,实现日志记录*/
@Slf4j
public class SimpleServiceProxy {private BaseSimpleService baseSimpleService;// 注入原对象public SimpleServiceProxy(BaseSimpleService baseSimpleService){this.baseSimpleService=baseSimpleService;}public void save(Object obj) {log.info("save obj:{}",obj);baseSimpleService.save(obj);log.info("保存: {}对象成功",obj);}
}
public class TestProxy {public static void main(String[] args) {// 基础实现BaseSimpleService baseService=new BaseSimpleServiceImpl();Object obj = new Object();baseService.save(obj);// 静态代理实现SimpleServiceProxy simpleServiceProxy=new SimpleServiceProxy(baseService);simpleServiceProxy.save(obj);// 原对象不受代理影响baseService.save(obj);}
}

 二 JDK动态代理

JDK动态代理的实现基于反射和类加载器, 通过 Proxy.newProxyInstance() 方法在运行时动态生成代理类,并将其加载到JVM中。

1 JDK动态代理的工作原理

Proxy.newProxyInstance()是JDK动态代理的核心方法,它会动态生成一个代理类,并返回该代理类的实例(即代理对象)。代理类的生成和加载涉及以下核心步骤

1.1 创建代理类

  • 当调用 Proxy.newProxyInstance() 时,JVM 会动态生成一个代理类,该代理类继承自 java.lang.reflect.Proxy 并实现目标对象的接口。
  • 代理类的名称是动态生成的,形如 com.sun.proxy.$Proxy0$Proxy0 是 JDK 动态代理生成的代理类的默认命名规则。

1.2 加载代理类

  • 生成的代理类会通过指定的类加载器(ClassLoader)加载到 JVM 中。
  • 代理类的字节码在内存中生成,并不会保存为 .class 文件。

1.3 实现方法调用

  • 代理类会实现目标接口的所有方法,但这些方法的逻辑会被重定向到 InvocationHandlerinvoke() 方法。
  • 代理类中,方法的实际调用行为是通过反射来完成的。

2. Proxy.newProxyInstance() 的核心工作流程

方法签名

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)

工作步骤

1 校验参数

    • 检查 ClassLoader 是否为 null
    • 检查 interfaces 是否为空,并确保所有接口都是有效的(如不能是 final 接口)。
    • 检查 InvocationHandler 是否为 null

生成代理类

    • 调用 Proxy 类的私有方法 getProxyClass0(),动态生成代理类的字节码。
    • 代理类的字节码是通过 sun.misc.ProxyGenerator 工具生成的。

加载代理类

    • 使用指定的 ClassLoader 将代理类的字节码加载到 JVM 中。

实例化代理类对象

    • 调用代理类的构造方法,传入 InvocationHandler 对象。
    • 返回动态生成的代理类的实例(即代理对象),它继承自 java.lang.reflect.Proxy 并实现了 Service 接口。

3. 代理类的生成与加载

3.1 代理类生成的关键方法

Proxy.getProxyClass()
  • 该方法通过目标接口数组生成代理类。
  • 代理类的字节码由 sun.misc.ProxyGenerator 工具生成。
  • 代理类继承自 java.lang.reflect.Proxy,并实现了目标接口。
Class<?> proxyClass = Proxy.getProxyClass(loader, interfaces);
ProxyGenerator.generateProxyClass()
  • 生成代理类的字节码。
  • 代理类的每个方法都会调用 InvocationHandler.invoke()
  • 字节码存储在内存中,并不会生成 .class 文件。

3.2 代理类的加载

代理类是如何加载的?
  • 代理类的字节码通过 ClassLoader 加载到 JVM 中。
  • 加载时会为代理类分配内存,并将其方法表注册到 JVM 的方法区中。
代理类加载的时机
  • 代理类是在 Proxy.newProxyInstance() 调用时动态生成并加载的。
  • 每次调用 newProxyInstance() 都会检查是否已存在对应接口的代理类。如果已存在,则直接加载;否则,重新生成代理类。

import lombok.extern.slf4j.Slf4j;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;// 目标接口
interface BaseSimpleService {public void save(Object obj);}// 接口实现类
class BaseSimpleServiceImpl implements BaseSimpleService {@Overridepublic void save(Object obj) {System.out.println("对象保存成功");}}// 调用处理器 不改变目标对象方法的基础上 记录日志
@Slf4j
class JdkLoggingInvocationHandler implements InvocationHandler {private Object target;public JdkLoggingInvocationHandler(Object target) {this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {log.info("save obj:{}",args);Object result = method.invoke(target, args);log.info("保存: {}对象成功",args);return result;}}
import java.lang.reflect.Proxy;public class TestProxy {public static void main(String[] args) {BaseSimpleService baseSimpleService=new BaseSimpleServiceImpl();JdkLoggingInvocationHandler handler=new JdkLoggingInvocationHandler(baseSimpleService);// 该实例具有代理类的指定调用处理程序,该代理类由指定的类加载器定义并实现指定的接口// simpleServiceProxy代理类由指定的类加载器加载 并实现了BaseSimpleService的接口BaseSimpleService simpleServiceProxy = (BaseSimpleService) Proxy.newProxyInstance(baseSimpleService.getClass().getClassLoader(), baseSimpleService.getClass().getInterfaces(),handler);simpleServiceProxy.save(new Object());}
}

三 CGLIB动态代理

CGLIB(Code Generation Library)是一个基于字节码生成的第三方库,用于在运行时动态生成Java类的子类,从而实现对目标类的代理。它主要用于代理没有实现接口的类,解决了JDK动态代理只能基于接口的局限性。其核心原理如下:

1. 代理方式
  • 继承式代理:生成目标类的子类(如 UserService$$EnhancerByCGLIB$$123456),通过重写父类非 final 方法实现代理。
  • 无需接口:直接代理普通类,弥补 JDK 动态代理的局限性。
  • 代理类会重写目标方法,并在方法中插入拦截逻辑
2. 核心组件
  • Enhancer
    负责生成代理类,配置代理策略(如回调方法、类加载器)。
        // 创建 CGLIB 代理的 Enhancer对象Enhancer enhancer=new Enhancer();// 设置代理的目标类// CGLIB通过继承这个目标类来生成代理类enhancer.setSuperclass(BaseService.class);// 设置方法拦截器,代理会调用这个拦截器enhancer.setCallback(new CglibLoggingMethodInterceptor(baseService));// 创建代理对象 当前的代理对象不是原始对象了 而是通过继承目标类BaseService得到的新类BaseService baseServiceProxy = (BaseService) enhancer.create();baseServiceProxy.save(new Object());
  • MethodInterceptor
    代理类的方法调用会被转发到MethodInterceptor接口的intercept方法中,实现增强逻辑。
/**
* obj: 代理对象本身
* method: 被拦截的方法(目标方法的反射对象)
* args: 方法参数
* proxy: 方法代理对象,用于调用父类(目标类)的原始方法
*/
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {// 前置增强(如日志)Object result = proxy.invokeSuper(obj, args); // 调用父类(目标类)原始方法// 后置增强(如事务提交)return result;
}

二、字节码生成

1. 动态生成原理
  • 基于 ASM:直接操作字节码生成 .class 文件的二进制内容,避免源码编译。
  • 方法重写
    生成子类时,为每个非 final 方法生成重写版本,插入拦截逻辑(调用 MethodInterceptor)。
2. 性能优化
  • FastClass 机制
    为代理类和目标类的方法建立索引,直接通过索引调用方法,绕过了反射的Method.invoke(),提升了性能。
// FastClass 通过索引调用方法(伪代码)
public Object invoke(int methodIndex, Object obj, Object[] args) {switch (methodIndex) {case 0: return ((TargetClass)obj).method1();case 1: return ((TargetClass)obj).method2();}
}
3. 示例:生成的字节码文件(伪代码)

生成的代理类可能如下(简化后的伪代码):

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
public class Cat$$EnhancerByCGLIB$$8ca2de8b extends Cat implements Factory {private boolean CGLIB$BOUND;public static Object CGLIB$FACTORY_DATA;private static final ThreadLocal CGLIB$THREAD_CALLBACKS;private static final Callback[] CGLIB$STATIC_CALLBACKS;private MethodInterceptor CGLIB$CALLBACK_0;private static Object CGLIB$CALLBACK_FILTER;private static final Method CGLIB$sleep$0$Method;private static final MethodProxy CGLIB$sleep$0$Proxy;private static final Object[] CGLIB$emptyArgs;private static final Method CGLIB$wakeup$1$Method;private static final MethodProxy CGLIB$wakeup$1$Proxy;private static final Method CGLIB$equals$2$Method;private static final MethodProxy CGLIB$equals$2$Proxy;private static final Method CGLIB$toString$3$Method;private static final MethodProxy CGLIB$toString$3$Proxy;private static final Method CGLIB$hashCode$4$Method;private static final MethodProxy CGLIB$hashCode$4$Proxy;private static final Method CGLIB$clone$5$Method;private static final MethodProxy CGLIB$clone$5$Proxy;static void CGLIB$STATICHOOK4() {CGLIB$THREAD_CALLBACKS = new ThreadLocal();CGLIB$emptyArgs = new Object[0];Class var0 = Class.forName("com.example.demo.cglibproxy.vo.Cat$$EnhancerByCGLIB$$8ca2de8b");Class var1;Method[] var10000 = ReflectUtils.findMethods(new String[]{"equals", "(Ljava/lang/Object;)Z", "toString", "()Ljava/lang/String;", "hashCode", "()I", "clone", "()Ljava/lang/Object;"}, (var1 = Class.forName("java.lang.Object")).getDeclaredMethods());CGLIB$equals$2$Method = var10000[0];CGLIB$equals$2$Proxy = MethodProxy.create(var1, var0, "(Ljava/lang/Object;)Z", "equals", "CGLIB$equals$2");CGLIB$toString$3$Method = var10000[1];CGLIB$toString$3$Proxy = MethodProxy.create(var1, var0, "()Ljava/lang/String;", "toString", "CGLIB$toString$3");CGLIB$hashCode$4$Method = var10000[2];CGLIB$hashCode$4$Proxy = MethodProxy.create(var1, var0, "()I", "hashCode", "CGLIB$hashCode$4");CGLIB$clone$5$Method = var10000[3];CGLIB$clone$5$Proxy = MethodProxy.create(var1, var0, "()Ljava/lang/Object;", "clone", "CGLIB$clone$5");var10000 = ReflectUtils.findMethods(new String[]{"sleep", "()V", "wakeup", "()V"}, (var1 = Class.forName("com.example.demo.cglibproxy.vo.Cat")).getDeclaredMethods());CGLIB$sleep$0$Method = var10000[0];CGLIB$sleep$0$Proxy = MethodProxy.create(var1, var0, "()V", "sleep", "CGLIB$sleep$0");CGLIB$wakeup$1$Method = var10000[1];CGLIB$wakeup$1$Proxy = MethodProxy.create(var1, var0, "()V", "wakeup", "CGLIB$wakeup$1");}final void CGLIB$sleep$0() {super.sleep();}public final void sleep() {MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;if (this.CGLIB$CALLBACK_0 == null) {CGLIB$BIND_CALLBACKS(this);var10000 = this.CGLIB$CALLBACK_0;}if (var10000 != null) {var10000.intercept(this, CGLIB$sleep$0$Method, CGLIB$emptyArgs, CGLIB$sleep$0$Proxy);} else {super.sleep();}}final void CGLIB$wakeup$1() {super.wakeup();}public final void wakeup() {MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;if (this.CGLIB$CALLBACK_0 == null) {CGLIB$BIND_CALLBACKS(this);var10000 = this.CGLIB$CALLBACK_0;}if (var10000 != null) {var10000.intercept(this, CGLIB$wakeup$1$Method, CGLIB$emptyArgs, CGLIB$wakeup$1$Proxy);} else {super.wakeup();}}public static MethodProxy CGLIB$findMethodProxy(Signature var0) {String var10000 = var0.toString();switch(var10000.hashCode()) {case -1385928386:if (var10000.equals("sleep()V")) {return CGLIB$sleep$0$Proxy;}break;case -508378822:if (var10000.equals("clone()Ljava/lang/Object;")) {return CGLIB$clone$5$Proxy;}break;case 391780310:if (var10000.equals("wakeup()V")) {return CGLIB$wakeup$1$Proxy;}break;case 1826985398:if (var10000.equals("equals(Ljava/lang/Object;)Z")) {return CGLIB$equals$2$Proxy;}break;case 1913648695:if (var10000.equals("toString()Ljava/lang/String;")) {return CGLIB$toString$3$Proxy;}break;case 1984935277:if (var10000.equals("hashCode()I")) {return CGLIB$hashCode$4$Proxy;}}return null;}public Cat$$EnhancerByCGLIB$$8ca2de8b() {CGLIB$BIND_CALLBACKS(this);}public Cat$$EnhancerByCGLIB$$8ca2de8b(String var1) {super(var1);CGLIB$BIND_CALLBACKS(this);}public static void CGLIB$SET_THREAD_CALLBACKS(Callback[] var0) {CGLIB$THREAD_CALLBACKS.set(var0);}public static void CGLIB$SET_STATIC_CALLBACKS(Callback[] var0) {CGLIB$STATIC_CALLBACKS = var0;}private static final void CGLIB$BIND_CALLBACKS(Object var0) {Cat$$EnhancerByCGLIB$$8ca2de8b var1 = (Cat$$EnhancerByCGLIB$$8ca2de8b)var0;if (!var1.CGLIB$BOUND) {var1.CGLIB$BOUND = true;Object var10000 = CGLIB$THREAD_CALLBACKS.get();if (var10000 == null) {var10000 = CGLIB$STATIC_CALLBACKS;if (CGLIB$STATIC_CALLBACKS == null) {return;}}var1.CGLIB$CALLBACK_0 = (MethodInterceptor)((Callback[])var10000)[0];}}public Object newInstance(Callback[] var1) {CGLIB$SET_THREAD_CALLBACKS(var1);Cat$$EnhancerByCGLIB$$8ca2de8b var10000 = new Cat$$EnhancerByCGLIB$$8ca2de8b();CGLIB$SET_THREAD_CALLBACKS((Callback[])null);return var10000;}public Object newInstance(Callback var1) {CGLIB$SET_THREAD_CALLBACKS(new Callback[]{var1});Cat$$EnhancerByCGLIB$$8ca2de8b var10000 = new Cat$$EnhancerByCGLIB$$8ca2de8b();CGLIB$SET_THREAD_CALLBACKS((Callback[])null);return var10000;}public Object newInstance(Class[] var1, Object[] var2, Callback[] var3) {CGLIB$SET_THREAD_CALLBACKS(var3);Cat$$EnhancerByCGLIB$$8ca2de8b var10000 = new Cat$$EnhancerByCGLIB$$8ca2de8b;switch(var1.length) {case 0:var10000.<init>();break;case 1:if (var1[0].getName().equals("java.lang.String")) {var10000.<init>((String)var2[0]);break;}throw new IllegalArgumentException("Constructor not found");default:throw new IllegalArgumentException("Constructor not found");}CGLIB$SET_THREAD_CALLBACKS((Callback[])null);return var10000;}public Callback getCallback(int var1) {CGLIB$BIND_CALLBACKS(this);MethodInterceptor var10000;switch(var1) {case 0:var10000 = this.CGLIB$CALLBACK_0;break;default:var10000 = null;}return var10000;}public void setCallback(int var1, Callback var2) {switch(var1) {case 0:this.CGLIB$CALLBACK_0 = (MethodInterceptor)var2;default:}}public Callback[] getCallbacks() {CGLIB$BIND_CALLBACKS(this);return new Callback[]{this.CGLIB$CALLBACK_0};}public void setCallbacks(Callback[] var1) {this.CGLIB$CALLBACK_0 = (MethodInterceptor)var1[0];}static {CGLIB$STATICHOOK4();}
}

三、类加载与代理对象创建

1. 类加载流程
  • 动态加载:生成的字节码通过类加载器(默认目标类的 ClassLoader)加载到 JVM。
  • 缓存优化:已生成的代理类会被缓存,避免重复生成。
2. 突破限制
  • 绕过构造函数
    使用 Objenesis 库直接实例化代理对象,无需调用父类构造函数(即使父类有无参构造)。
Objenesis objenesis = new ObjenesisStd();
TargetClass proxy = objenesis.newInstance(proxyClass); // 直接实例化

四、与 JDK 动态代理的关键区别

特性

CGLIB

JDK 动态代理

代理方式

继承目标类生成子类

实现目标接口

目标要求

不能是 final 类/方法

必须实现接口

方法调用速度

快(FastClass 直接调用)

慢(反射调用)

内存消耗

较高(需生成子类)

较低

五、典型限制与解决方案

1. 无法代理 final 方法/类
  • 表现:若目标类或方法是 final,CGLIB 无法生成子类。
  • 解决:重构代码移除 final 修饰,或改用 JDK 动态代理。
2. 构造函数调用问题
  • 表现:代理类会调用父类构造函数,若父类没有无参构造函数会报错。
  • 解决:使用 Objenesis 绕过构造函数(需添加依赖)。

六、应用场景

  • Spring AOP:默认对未实现接口的类使用 CGLIB 代理。
  • 性能敏感场景:如高频调用的工具类增强。
  • 历史代码扩展:无法修改原有类/接口时,直接代理实现功能增强。

import org.springframework.cglib.proxy.Enhancer;public class TestProxy {public static void main(String[] args) {BaseService baseService=new BaseService();// 创建 CGLIB 代理的 Enhancer对象Enhancer enhancer=new Enhancer();// 设置代理的目标类// CGLIB通过继承这个目标类来生成代理类enhancer.setSuperclass(BaseService.class);// 设置方法拦截器,代理会调用这个拦截器enhancer.setCallback(new CglibLoggingMethodInterceptor(baseService));// 创建代理对象 当前的代理对象不是原始对象了 而是通过继承目标类BaseService得到的新类BaseService baseServiceProxy = (BaseService) enhancer.create();baseServiceProxy.save(new Object());}
}
import lombok.extern.slf4j.Slf4j;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;/*** 目标类*/
class BaseService {public void save(Object obj) {System.out.println("对象保存成功");}}@Slf4j
public class CglibLoggingMethodInterceptor  implements MethodInterceptor {private Object target;public CglibLoggingMethodInterceptor(Object target) {this.target = target;}@Overridepublic Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {log.info("save obj:{}",args);// 调用父类方法Object result = proxy.invokeSuper(obj, args);log.info("保存: {}对象成功",args);return result;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/77101.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Cache与Redis集成原理

一、核心架构图解 #mermaid-svg-aiWGQLhmWx7kOfLz {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-aiWGQLhmWx7kOfLz .error-icon{fill:#552222;}#mermaid-svg-aiWGQLhmWx7kOfLz .error-text{fill:#552222;stroke:#5…

编程技能:调试02,设置断点与删除断点

专栏导航 本节文章分别属于《Win32 学习笔记》和《MFC 学习笔记》两个专栏&#xff0c;故划分为两个专栏导航。读者可以自行选择前往哪个专栏。 &#xff08;一&#xff09;WIn32 专栏导航 上一篇&#xff1a;编程技能&#xff1a;调试01&#xff0c;调试介绍 回到目录 下…

flink写doris时的优化

1.概念 doris并不擅长高频、小量数据的导入&#xff1b; 因为doris每一次数据导入都会在be节点上生成数据文件&#xff1b;如果高频导入小量数据&#xff0c;就会在存储层产生大量的小文件&#xff08;必然会影响到后续的查询效率&#xff0c;也会对系统产生更多的compaction…

ElementNotInteractableException原因及解决办法

在自动化测试中,ElementNotInteractableException是一个常见的异常,它通常发生在尝试与网页上的某个元素进行交互(例如点击、输入等操作)时,但由于该元素当前不可交互。这可能由多种原因引起,以下是一些常见的原因及其解决方法: 元素未完全加载 如果尝试与页面上的元素交…

如何从 GitHub 镜像仓库到极狐GitLab?

最近 GitHub 封禁中国用户的事情闹得沸沸扬扬,虽然官方发布的报道说中国用户被限制登录是因为配置错误导致,已经撤回了更新,中国用户已经可以正常使用。但是这就像横在国内开发者和企业头上的“达摩克利斯之剑”。为了避免 GitHub 不可用而带来的影响,国内开发者和企业可以…

服务器安装nacos

1.下载依赖 docker pull nacos/nacos-server:v2.4.3安装 docker run -d --name nacos-server -p 8848:8848 -e MODEstandalone nacos/nacos-server:v2.4.3把nacos中的data 文件和conf 文件copy到自己服务的文件夹 docker cp nacos-server:/home/nacos/data /home/admin1/…

Matter协议暗战:苹果、谷歌、亚马逊的智能家居霸权争夺

原文地址&#xff1a;Matter协议暗战&#xff1a;苹果、谷歌、亚马逊的智能家居霸权争夺 一、Matter 协议&#xff1a;巨头联手打造的 “智能家居联合国” 1.1 从 CHIP 到 Matter&#xff1a;标准统一的十年长跑 智能家居发展多年&#xff0c;却始终被 “孤岛效应” 困扰。各…

软件设计师2009-2022历年真题与答案解析(附pdf下载)

软考在即&#xff0c;现在给大家分享一下软件设计师2009-2022真题与答案解析 pdf全套&#xff0c;文末提供大家免费下载&#xff0c;大家都知道在软考备考过程中&#xff0c;拥有一套全面且实用的考试资料对于考生来说至关重要。目录如下&#xff1a; 历年真题及详解2004-2019 …

基于EasyX库开发的球球大作战游戏

目录 球球大作战 一、开发环境 二、流程图预览 三、代码逻辑 1、初始化时间 2、设置开始界面大小 3、设置开始界面 4、让玩家选择速度 5、设置玩家小球、人机小球、食物的属性 6、一次性把图绘制到界面里 7、进入死循环 8、移动玩家小球 9、移动人机 10、食物刷新…

aslist和list的区别

‌Arrays.asList和List的主要区别在于它们的固定长度和不可变性、与原始数组的关系、性能以及使用场景。 一、固定长度和不可变性 ‌Arrays.asList‌&#xff1a;通过Arrays.asList方法创建的List是一个固定长度的List&#xff0c;其长度与原始数组相同。这意味着你不能通过添…

大模型预标注和自动化标注在OCR标注场景的应用

OCR&#xff0c;即光学字符识别&#xff0c;简单来说就是利用光学设备去捕获图像并识别文字&#xff0c;最终将图片中的文字转换为可编辑和可搜索的文本。在数字化时代&#xff0c;OCR&#xff08;光学字符识别&#xff09;技术作为处理图像中文字信息的关键手段&#xff0c;其…

stm32工程,拷贝到另一台电脑编译,错误提示头文件找不到cannot open source input file “core_cm4.h”

提示 cannot open source input file “core_cm4.h” ,找不到 [ core_cm4.h ] 这个头文件 . 于是我在原电脑工程文件里找也没有找到这个头文件 接下来查看原电脑keil的头文件引入配置,发现只引入了工程文件下的头文件, 那么core_cm4.h到底哪里来的? (到现在我也不清楚怎…

STM32 模块化开发指南 · 第 2 篇 如何编写高复用的外设驱动模块(以 UART 为例)

本文是《STM32 模块化开发实战指南》的第 2 篇,聚焦于“串口驱动模块的设计与封装”。我们将从一个最基础的裸机 UART 初始化开始,逐步实现:中断支持、环形缓冲收发、模块接口抽象与测试策略,构建一个可移植、可扩展、可复用的 UART 驱动模块。 一、模块化 UART 的设计目标…

【NLP 59、大模型应用 —— 字节对编码 bpe 算法】

目录 一、词表的构造问题 二、bpe(byte pair encoding) 压缩算法 算法步骤 示例&#xff1a; 步骤 1&#xff1a;初始化符号表和频率统计 步骤 2&#xff1a;统计相邻符号对的频率 步骤 3&#xff1a;合并最高频的符号对 步骤 4&#xff1a;重复合并直至终止条件 三、bpe在NLP中…

TMS320F28P550SJ9学习笔记15:Lin通信SCI模式结构体寄存器

今日初步认识与配置使用Lin通信SCI模式&#xff0c;用结构体寄存器的方式编程 文章提供完整工程下载、测试效果图 我的单片机平台是这个&#xff1a; LIN通信引脚&#xff1a; LIN通信PIE中断&#xff1a; 这个 PIE Vector Table 表在手册111页&#xff1a; 这是提到LINa的PI…

linux-设置每次ssh登录服务器的时候提醒多久需要修改密码

在 Linux 系统中,你可以通过设置 motd(Message of the Day)或 sshd 配置来在用户通过 SSH 登录时提醒他们密码即将过期。以下是具体步骤: 方法 1: 使用 motd 文件 motd 文件在用户登录时显示,你可以通过脚本动态生成内容,提醒用户密码过期时间。 编辑 /etc/motd 文件:…

matlab求和∑函数方程编程?

matlab求和∑函数方程编程&#xff1f; 一 题目&#xff1a;求下列函数方程式的和 二&#xff1a;代码如下&#xff1a; >> sum_result 0; % 初始化求和变量 for x 1:10 % 设…

electron桌面端开发-打开指定软件和文件

electron桌面端开发 现在越来越多的软件开发已经趋向于简单化&#xff0c;桌面端开发已经不在依赖之前的java、c等主流技术&#xff0c;目前基于node的开发越来越广泛。功能点也越来越多元化。 文章目录 electron桌面端开发前言一、打开文件的方式&#xff1f;二、exec使用步骤…

ShenNiusModularity项目源码学习(17:ShenNius.Admin.Mvc项目分析-2)

ShenNiusModularity项目的后台管理主页面如下图所示&#xff0c;该页面为ShenNius.Admin.Mvc项目的Views\Home\Index.cshtml&#xff0c;使用的是layuimini后台模板&#xff08;参考文献2&#xff09;&#xff0c;在layuimini的GitHub主页中提供有不同样式的页面模版链接&#…

SpringBoot 与 Vue3 实现前后端互联全解析

在当前的互联网时代&#xff0c;前后端分离架构已经成为构建高效、可维护且易于扩展应用系统的主流方式。本文将详细介绍如何利用 SpringBoot 与 Vue3 构建一个前后端分离的项目&#xff0c;展示两者如何通过 RESTful API 实现无缝通信&#xff0c;让读者了解从环境搭建、代码实…