当前位置: 首页 > news >正文

Spring5学习笔记—AOP编程

Springgif.gif

✅作者简介:大家好,我是Leo,热爱Java后端开发者,一个想要与大家共同进步的男人😉😉
🍎个人主页:Leo的博客
💞当前专栏: Spring专栏
✨特色专栏: MySQL学习
🥭本文内容:Spring5学习笔记—AOP编程
🖥️个人小站 :个人博客,欢迎大家访问
📚个人知识库: 知识库,欢迎大家访问

1. 静态代理设计模式

1.1 为什么需要代理设计模式

  • 在JavaEE分层开发开发中,那个层次对于我们来讲最重要

    DAO ---> Service --> Controller JavaEE分层开发中,最为重要的是Service层
    
  • Service层中包含了哪些代码?

    Service层中 = 核心功能(几十行 上百代码) + 额外功能(附加功能)
    1. 核心功能业务运算DAO调用
    2. 额外功能(事务、日志、性能...)1. 不属于业务2. 可有可无3. 代码量很小 
    
  • 额外功能书写在Service层中好不好?

    Controller层(Service层的调用者)除了需要核心功能,还需要这些额外功能。

    但是从软件设计者角度看:Service层最好不要写额外功能。

  • 现实生活中的解决方式

    image-20230815111510825

代理模式是一种比较好理解的设计模式。简单来说就是 我们使用代理对象来代替对真实对象(real object)的访问,这样就可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标对象的功能。

代理模式的主要作用是扩展目标对象的功能,比如说在目标对象的某个方法执行前后你可以增加一些自定义的操作。

代理模式: 为一个对象提供一个替身,以控制对这个对象的访问。即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。

被代理的对象可以是远程对象、创建开销大的对象或需要安全控制的对象

举个例子:当我们工作之后需要出去租房子,房东张贴广告带我看房子,最后签合同,但是房东只想坐着签合同并不想到处跑着看房子,于是就找了一个中介专门来宣传广告并且带租户看房子,而房东只负责签合同收钱!中介在这里就可以看作是代理你的代理对象代理的行为(方法)是带租户看房子。

img

1.2 代理设计模式分析

2.1 概念

通过代理类,为原始类(目标)增加额外的功能 好处:利于原始类(目标)的维护

2.2 名词解释

1. 目标类 原始类 指的是 业务类 (核心功能 --> 业务运算 DAO调用)
2. 目标方法,原始方法目标类(原始类)中的方法 就是目标方法(原始方法)
3. 额外功能 (附加功能)日志,事务,性能

2.3 代理开发的核心要素

代理类 = 实现和目标类相同的接口 + 在同名方法中添加额外功能 + 调用原始类同名方法房东 ---> public interface UserService{m1m2}UserServiceImpl implements UserService{m1 ---> 业务运算 DAO调用m2 }UserServiceProxy implements UserServicem1m2

2.4 编码

静态代理:为每一个原始类,手动编写一个代理类 (.java .class)

image-20230815112248257

2.5 静态代理存在的问题

1. 静态类文件数量过多,不利于项目管理UserServiceImpl  UserServiceProxyOrderServiceImpl OrderServiceProxy
2. 额外功能维护性差代理类中 额外功能修改复杂(麻烦)

2. Spring的动态代理开发

2.1 Spring动态代理的概念

概念:通过代理类为原始类(目标类)增加额外功能,代理类由Spring动态生成。
好处:利于原始类(目标类)的维护

2.2 搭建开发环境

<dependency><groupId>org.springframework</groupId><artifactId>spring-aop</artifactId><version>5.1.14.RELEASE</version>
</dependency><dependency><groupId>org.aspectj</groupId><artifactId>aspectjrt</artifactId><version>1.8.8</version>
</dependency><dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.8.3</version>
</dependency>

2.3 Spring动态代理的开发步骤

1. 创建原始对象(目标对象)

public class UserServiceImpl implements UserService{@Overridepublic void register(User user) {System.out.println("UserServiceImpl.register");}@Overridepublic boolean login(String name, String password) {System.out.println("UserServiceImpl.login");return true;}
}
<bean id="userServiceImpl" class="com.Leo.dynamic.service.impl.UserServiceImpl"/>

2. 定义额外功能

实现MethodBeforeAdvice接口

public class Before implements MethodBeforeAdvice {//作用:给原始方法添加额外功能//注意:会在原始方法运行之前运行此方法@Overridepublic void before(Method method, Object[] args, Object target) throws Throwable {System.out.println("-----method before advice log------");}
}
<bean id="before" class="com.Leo.dynamic.service.Before"/>

3. 定义切入点

1. 切入点:额外功能加入的位置2. 目的:由程序员根据自己的需要,决定额外功能加入给那个原始方法
register()
login()简单的测试:所有方法都做为切入点,都加入额外的功能。
<aop:config><aop:pointcut id="pc" expression="execution(* *(..))"/>
</aop:config>

4. 组装

<!-- 组装切入点与额外功能 -->
<aop:advisor advice-ref="before" pointcut-ref="pc"/>

5. 测试调用

目的:获得Spring工厂创建的动态代理对象,并进行调用
ApplicationContext ctx = new ClassPathXmlApplicationContext("/applicationContext.xml");
注意:1. Spring的工厂通过	原始对象的id值获得的是代理对象2. 获得代理对象后,可以通过声明接口类型,进行对象的存储UserService userService=(UserService)ctx.getBean("userServiceImpl");userService.login("","");
userService.register(new User());

控制台打印: 可以发现在日志之前输入了

image-20230815114453445

2.4 动态代理细节分析

4.1 Spring创建的动态代理类在哪里?

Spring框架在运行时,通过动态字节码技术,在JVM创建的,运行在JVM内部,等程序结束后就消失了。

什么叫动态字节码技术:通过第三方动态字节码框架,在JVM中创建对应类的字节码,进而创建对象,当虚拟机结束,动态字节码跟着消失。

结论:动态代理不需要定义类文件,都是JVM运行过程中动态创建的,所以不会造成静态代理,类文件数量过多,影响项目管理的问题。

image-20200423165547079

4.2 动态代理编程简化代理的开发

在额外功能不改变的前提下,创建其他目标类(原始类)的代理对象时,只需要指定原始(目标)对象即可。

3. Spring动态代理详解

3.1 额外功能的详解

  • MethodBeforeAdvice分析

    作用:原始方法执行之前,运行额外功能。

    public class Before implements MethodBeforeAdvice {/*** 作用:给原始方法添加额外功能* 注意:会在原始方法运行之前运行此方法* @param method 原始方法 login() register() ...* @param objects 原始方法的参数列表 name password ...* @param o 原始对象 UserServiceImpl OrderServiceImpl* @throws Throwable 抛出的异常*/@Overridepublic void before(Method method, Object[] objects, Object o) throws Throwable {System.out.println("---- MethodBeforeAdvice  log... ----");}
    }
    

实战:需要时才用,可能都会用到,也有可能都不用。

  • MethodInterceptor(方法拦截器)

    MethodInterceptor接口:额外功能可定义在原始方法执行 前、后、前和后。

    public class Around implements MethodInterceptor {/*** @param invocation 封装了原始方法 invocation.proceed()表示原始方法的运行* @return 原始方法的返回值* @throws Throwable 可能抛出的异常*/@Overridepublic Object invoke(MethodInvocation invocation) throws Throwable {System.out.println("------ 额外功能 log -----");//原始方法的执行Object ret = invocation.proceed();//返回原始方法的返回值return ret;}
    }
    

额外功能运行在原始方法执行之后

  public Object invoke(MethodInvocation invocation) throws Throwable {Object ret = invocation.proceed();System.out.println("-----额外功能运行在原始方法执行之后----");return ret;}

额外功能运行在原始方法执行之前和之后(实战:事务需要在之前和之后都运行)

@Override
public Object invoke(MethodInvocation invocation) throws Throwable {System.out.println("-----额外功能运行在原始方法执行之前----");Object ret = invocation.proceed();System.out.println("-----额外功能运行在原始方法执行之后----");return ret;
}

额外功能运行在原始方法抛出异常时

  @Override
public Object invoke(MethodInvocation invocation) throws Throwable {Object ret = null;try {ret = invocation.proceed();} catch (Throwable throwable) {System.out.println("-----原始方法抛出异常 执行的额外功能 ---- ");throwable.printStackTrace();}return ret;}

MethodInterceptor可以影响原始方法的返回值

@Override
public Object invoke(MethodInvocation invocation) throws Throwable {System.out.println("------log-----");Object ret = invocation.proceed();//拿到原始方法的返回值后进行一些操作就会影响,直接返回就不影响return false;
}

3.2 切入点详解

切入点决定了额外功能加入的位置。

<aop:pointcut id="pc" expression="execution(* *(..))"/>
exection(* *(..)) ---> 匹配了所有方法    a  b  c 1. execution()  切入点函数
2. * *(..)      切入点表达式 

1. 切入点表达式

  • 方法切入点表达式:

    image-20200425105040237

    *  *(..)  --> 所有方法* ---> 修饰符 返回值
    * ---> 方法名
    ()---> 参数表
    ..---> 对于参数没有要求 (0个或多个)
    

举例:

  # 定义login方法作为切入点* login(..)# 定义register作为切入点* register(..)# 定义名字为login且有两个字符串类型参数的方法 作为切入点* login(String,String)# 注意:非java.lang包中的类型,必须要写全限定名* register(com.Leo.proxy.User)# ..可以和具体的参数类型连用(至少有一个参数是String类型)* login(String,..)# 精准方法切入点限定# 修饰符 返回值    包.类.方法(参数)*             com.yuziayn.proxy.UserServiceImpl.login(..)*             com.Leo.proxy.UserServiceImpl.login(String,String)
  • 类切入点表达式:

    指定特定的类作为切入点,即这个类中所有的方法都会加上额外功能。

    举例:

    # 类中的所有方法都加入额外功能 
    * com.Leo.proxy.UserServiceImpl.*(..)# 忽略包
    # 1. 类只在一级包下  com.UserServiceImpl
    * *.UserServiceImpl.*(..)# 2. 类可在多级包下  com.Leo.proxy.UserServiceImpl
    * *..UserServiceImpl.*(..)
    
  • 包切入点表达式:

    指定包作为切入点,即这个包中的所有类及其方法都会加入额外的功能。

    举例:

    # proxy包作为切入点,即proxy包下所有类中的所有方法都会加入额外功能,但是不包括其子包中的类!
    * com.Leo.proxy.*.*(..)# 当前包及其子包都生效
    * com.Leo.proxy..*.*(..) 
    

2 切入点函数

作用:用于执行切入点表达式。

  1. execution()

    最为重要的切入点函数,功能最全!
    用于执行:方法切入点表达式、类切入点表达式、包切入点表达式 弊端:execution执行切入点表达式 ,书写麻烦execution(* com.Leo.proxy..*.*(..))注意:其他的切入点函数 只是简化execution书写复杂度,功能上完全一致
    
  2. args()

    # 作用:用于函数(方法)参数的匹配# 举例:方法参数必须得是2个字符串类型的参数execution(* *(String,String))等价于:args(String,String)
    
  3. within()

    # 作用:用于进行类、包切入点表达式的匹配
    # 举例:
    # UserServiceImpl类作为切入点:execution(* *..UserServiceImpl.*(..))within(*..UserServiceImpl)
    # proxy包作为切入点:execution(* com.Leo.proxy..*.*(..))within(com.yuziayan.proxy..*)
    
  4. @annotation()

    <!-- 作用:为具有特殊注解的方法加入额外功能 --><aop:pointcut id="" expression="@annotation(com.baizhiedu.Log)"/>
    
  5. 切入点函数间的逻辑运算:

    目的:整合多个切入点函数一起配合工作,进而完成更为复杂的需求。

    • and 与操作(同时满足)

      # 案例:方法名为login,同时有2个字符串类型的参数:execution(* login(String,String))execution(* login(..)) and args(String,String)# 注意:与操作不能用于同种类型的切入点函数 
      # 错误案例:register方法 和 login方法作为切入点(不能用and,而用or!)execution(* login(..)) and execution(* register(..))
      # 上面的语句会发生错误,因为其实际表达的含义是方法名为login同时方法名为register,显然有悖逻辑,此时应该用到的是 or
      
  • or 或操作(满足一种即可)

       # 案例:register方法 和 login方法作为切入点 execution(* login(..)) or  execution(* register(..))
    

4. AOP编程

4.1 AOP概念

# AOP (Aspect Oriented Programing)   面向切面编程 = Spring动态代理开发
# 以切面为基本单位的程序开发,通过切面间的彼此协同,相互调用,完成程序的构建
# 切面 = 切入点 + 额外功能# OOP (Object Oriented Programing)   面向对象编程 Java
# 以对象为基本单位的程序开发,通过对象间的彼此协同,相互调用,完成程序的构建# POP (Procedure Oriented Programing) 面向过程(方法、函数)编程 C 
# 以过程为基本单位的程序开发,通过过程间的彼此协同,相互调用,完成程序的构建# AOP的概念:本质就是Spring的动态代理开发,通过代理类为原始类增加额外功能。好处:利于原始类的维护
# 注意:AOP编程不可能取代OOP,而是OOP编程的补充。

4.2 AOP编程的开发步骤

  1. 原始对象
  2. 额外功能 (MethodInterceptor)
  3. 切入点
  4. 组装切面 (额外功能+切入点)

4.3 切面的名词解释

切面 = 切入点 + 额外功能 几何学面 = 点 + 相同的性质

image-20230529150312064

5. AOP的底层实现原理

5.1 核心问题

  • AOP如何创建动态代理类?(动态字节码技术)
  • Spring工厂如何加工创建代理对象?通过原始对象的id值,获得的是代理对象。

5.2 动态代理类的创建

1. JDK的动态代理

  • Proxy.newProxyInstance()方法参数详解:

    image-20200428175248912

image-20200428175316276

  • 编码:

    public class TestJDKProxy {public static void main(String[] args) {//1.创建原始对象//注意:由于后面匿名子类的方法中用到了userService,所以应该用final修饰//     而JDK1.8以后默认加了final,不需要手动加UserService userService = new UserServiceImpl();//2.JDK创建代理对象InvocationHandler handler = new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("----------- JDKProxy log -----------");\//目标方法运行:Object ret = method.invoke(userService, args);return ret;}};UserService userServiceProxy = (UserService) Proxy.newProxyInstance(userService.getClass().getClassLoader(), userService.getClass().getInterfaces(),handler);userServiceProxy.login("Leo", "123456");userServiceProxy.register(new User());}
    }
    

2. CGlib的动态代理

  • 原理:通过父子继承关系创建代理对象。原始类作为父类,代理类作为子类,这样既可以保证2者方法一致,同时在代理类中提供新的实现(额外功能+原始方法)

image-20200429111709226

  • CGlib编码:

    package com.Leo.cglib;import org.springframework.cglib.proxy.Enhancer;
    import org.springframework.cglib.proxy.MethodInterceptor;
    import org.springframework.cglib.proxy.MethodProxy;import java.lang.reflect.Method;public class TestCGlibProxy {public static void main(String[] args) {//1.创建原始对象UserServiceImpl userService = new UserServiceImpl();//2.通过CGlib创建代理对象//  2.1 创建EnhancerEnhancer enhancer = new Enhancer();//  2.2 设置借用类加载器enhancer.setClassLoader(TestCGlibProxy.class.getClassLoader());//  2.3 设置父类(目标类)enhancer.setSuperclass(userService.getClass());//  2.4 设置回调,额外功能写在里面enhancer.setCallback(new MethodInterceptor() {//相当于 InvocationHandler --> invoke()@Overridepublic Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {//额外功能:System.out.println("========= CGlibProxy log ========");//目标方法执行:Object ret = method.invoke(userService, objects);return ret;}});//  2.5 通过Enhancer对象创建代理UserServiceImpl service = (UserServiceImpl) enhancer.create();//测试:service.register();service.login();}
    }
    

3. 总结

1. JDK动态代理   Proxy.newProxyInstance()  
# 通过目标类实现的接口创建代理类 
2. Cglib动态代理 Enhancer                  
# 通过继承目标类创建代理类 

5.3 Spring工厂如何返回代理对象

  • 思路分析:

image-20200430113353205

  • 编码模拟:

    public class ProxyBeanPostProcessor implements BeanPostProcessor {@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {return bean;}@Overridepublic Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {InvocationHandler invocation = new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("----------- 模拟Spring返回代理对象的方式 log -----------");Object ret = method.invoke(bean, args);return ret;}};return Proxy.newProxyInstance(ProxyBeanPostProcessor.class.getClassLoader(), bean.getClass().getInterfaces(), invocation);}
    }
    
      <!-- 1.配置原始对象 --><bean id="userService" class="com.Leo.factory.UserServiceImpl"></bean><!-- 2.配置自己模拟的ProxyBeanPostProcessor --><bean id="proxyBeanPostProcessor" class="com.Leo.factory.ProxyBeanPostProcessor"/>
    

6. 基于注解的AOP编程

6.1 开发步骤:

  1. 原始对象

  2. 额外功能

  3. 切入点

  4. 组装切面

    /*** 声明切面类     @Aspect* 定义额外功能   @Around* 定义切入点     @Around("execution(* login(..))")**/
    @Aspect
    public class MyAspect {@Around("execution(* login(..))")//组装了切入点和额外功能public Object around(ProceedingJoinPoint joinPoint) throws Throwable {//额外功能:System.out.println("--------- 基于注解的AOP编程 log --------");//原始方法执行:Object ret = joinPoint.proceed();return ret;}
    }
    
       <!-- 原始对象 --><bean id="userService" class="com.Leo.aspect.UserServiceImpl"></bean><!-- 切面 --><bean id="myAspect" class="com.Leo.aspect.MyAspect"/><!-- 开启基于注解的AOP编程 --><aop:aspectj-autoproxy/>
    

6.2 细节分析:

  • 切入点复用:

    @Aspect
    public class MyAspect {/*** 切入点复用:定义一个函数,加上@Pointcut注解,通过该注解的value定义切入点表达式,以后可以复用。*/@Pointcut("execution(* login(..))")public void myPointcut(){}@Around("myPointcut()")//组装了切入点和额外功能public Object around(ProceedingJoinPoint joinPoint) throws Throwable {//额外功能:System.out.println("--------- 基于注解的AOP编程 log --------");//原始方法执行:Object ret = joinPoint.proceed();return ret;}@Around("myPointcut()")public Object around1(ProceedingJoinPoint joinPoint) throws Throwable {//额外功能:System.out.println("--------- 基于注解的AOP编程 tx --------");//原始方法执行:Object ret = joinPoint.proceed();return ret;}}
    
  • 动态代理的创建方式:

    AOP底层实现  2种代理创建方式1.  JDK   通过实现接口,创建代理对象2.  Cglib 通过继承目标类,创建代理对象默认情况 AOP编程 底层应用JDK动态代理创建方式 如果要切换Cglib1. 基于注解AOP开发<aop:aspectj-autoproxy proxy-target-class="true" />2. 传统的AOP开发<aop:config proxy-target-class="true"></aop>
    

7. AOP开发中的一个坑

坑:在同一个业务类中,业务方法间相互调用时,只有最外层的方法,加入了额外功能(内部的方法,通过普通的方式调用,运行的都是原始方法)。如果想让内层的方法也调用代理对象的方法,就要实现AppicationContextAware获得工厂,进而获得代理对象。

public class UserServiceImpl implements UserService, ApplicationContextAware {private ApplicationContext ctx;@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {this.ctx = applicationContext;}@Log@Overridepublic void register(User user) {System.out.println("UserServiceImpl.register 业务运算 + DAO ");//throw new RuntimeException("测试异常");//调用的是原始对象的login方法 ---> 核心功能/*设计目的:代理对象的login方法 --->  额外功能+核心功能ApplicationContext ctx = new ClassPathXmlApplicationContext("/applicationContext2.xml");UserService userService = (UserService) ctx.getBean("userService");userService.login();Spring工厂重量级资源 一个应用中 应该只创建一个工厂*/UserService userService = (UserService) ctx.getBean("userService");userService.login("Leo", "123456");}@Overridepublic boolean login(String name, String password) {System.out.println("UserServiceImpl.login");return true;}
}

8. AOP阶段知识总结

AOP 总结

相关文章:

Spring5学习笔记—AOP编程

✅作者简介&#xff1a;大家好&#xff0c;我是Leo&#xff0c;热爱Java后端开发者&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a;Leo的博客 &#x1f49e;当前专栏&#xff1a; Spring专栏 ✨特色专栏&#xff1a; M…...

适用于 Docker 用户的 kubectl

适用于 Docker 用户的 kubectl 你可以使用 Kubernetes 命令行工具 kubectl 与 API 服务器进行交互。如果你熟悉 Docker 命令行工具&#xff0c; 则使用 kubectl 非常简单。但是&#xff0c;Docker 命令和 kubectl 命令之间有一些区别。以下显示了 Docker 子命令&#xff0c; 并…...

网络安全设备篇——加密机

加密机是一种专门用于数据加密和解密的网络安全设备。它通过使用密码学算法对数据进行加密&#xff0c;从而保护数据的机密性和完整性。加密机通常被用于保护敏感数据&#xff0c;如金融信息、个人身份信息等。 加密机的主要功能包括&#xff1a; 数据加密&#xff1a;加密机使…...

Rust 基础入门 —— 2.3.所有权和借用

Rust 的最主要光芒&#xff1a; 内存安全 。 实现方式&#xff1a; 所有权系统。 写在前面的序言 因为我们这里实际讲述的内容是关于 内存安全的&#xff0c;所以我们最好先复习一下内存的知识。 然后我们&#xff0c;需要理解的就只有所有权概念&#xff0c;以及为了开发便…...

Node.js-Express框架基本使用

Express介绍 Express是基于 node.js 的web应用开发框架&#xff0c;是一个封装好的工具包&#xff0c;便于开发web应用&#xff08;HTTP服务&#xff09; Express基本使用 // 1.安装 npm i express // 2.导入 express 模块 const express require("express"); // 3…...

阿里云通用算力型u1云服务器CPU性能详细说明

​阿里云服务器u1是通用算力型云服务器&#xff0c;CPU采用2.5 GHz主频的Intel(R) Xeon(R) Platinum处理器&#xff0c;通用算力型u1云服务器不适用于游戏和高频交易等需要极致性能的应用场景及对业务性能一致性有强诉求的应用场景(比如业务HA场景主备机需要性能一致)&#xff…...

设计模式之创建者模式

文章目录 一、介绍二、应用三、案例1. 麦当劳11随心配2. 代码演示3. 演示结果 四、优缺点五、送给读者 一、介绍 建造者模式(Builder Pattern)属于创建型设计模式&#xff0c;很多博客文章的对它的作用解释为用于将复杂对象的创建过程与其细节表示分离。但对于初学者来说&…...

Java之包,权限修饰符,final关键字详解

包 2.1 包 包在操作系统中其实就是一个文件夹。包是用来分门别类的管理技术&#xff0c;不同的技术类放在不同的包下&#xff0c;方便管理和维护。 在IDEA项目中&#xff0c;建包的操作如下&#xff1a; 包名的命名规范&#xff1a; 路径名.路径名.xxx.xxx // 例如&#xff…...

“深入解析JVM:Java虚拟机内部原理揭秘“

标题&#xff1a;深入解析JVM&#xff1a;Java虚拟机内部原理揭秘 摘要&#xff1a;本文将深入探讨Java虚拟机&#xff08;JVM&#xff09;的内部原理&#xff0c;包括JVM的架构、运行时数据区域、垃圾回收机制以及即时编译器等重要组成部分。通过对JVM内部原理的解析&#xf…...

Mac下Jmeter安装及基本使用

本篇文章只是简单的介绍下Jmeter的下载安装和最基本使用 1、初识Jmeter 前一段时间客户端app自测的过程中&#xff0c;有偶现请求某个接口返回数据为空的问题&#xff0c;领导让我循环100次请求这个接口&#xff0c;看看有没有结果为空的问题。听同事说有Jmeter的专业测试工具…...

云计算与边缘计算:加速数字化转型的关键驱动力

云计算和边缘计算技术正以惊人的速度改变着企业的业务和基础架构。这些先进的技术为企业带来了灵活性、可扩展性和成本效益的优势&#xff0c;重新定义了业务运作的方式。 云计算是通过互联网将计算资源提供给用户的一种服务模式。通过云计算&#xff0c;企业可以将应用程序、…...

TheGem主题 - 创意多用途和高性能WooCommerce WordPress主题/网站

TheGem主题概述 – 适合所有人的TheGem 作为设计元素、样式和功能的终极 Web 构建工具箱而设计和开发&#xff0c;TheGem主题将帮助您在几分钟内构建一个令人印象深刻的高性能网站&#xff0c;而无需触及一行代码。不要在编码上浪费时间&#xff0c;探索你的创造力&#xff01…...

Pytorch-day10-模型部署推理-checkpoint

模型部署&推理 模型部署模型推理 我们会将PyTorch训练好的模型转换为ONNX 格式&#xff0c;然后使用ONNX Runtime运行它进行推理 1、ONNX ONNX( Open Neural Network Exchange) 是 Facebook (现Meta) 和微软在2017年共同发布的&#xff0c;用于标准描述计算图的一种格式…...

vue使用websocket

建立websocket.js // 信息提示 import { Message } from element-ui // 引入用户id import { getTenantId, getAccessToken } from /utils/auth// websocket地址 var url ws://192.168.2.20:48081/websocket/message // websocket实例 var ws // 重连定时器实例 var tt // w…...

jmeter入门:接口压力测试全解析

一.对接口压力测试 1.配置 1.添加线程组&#xff08;参数上文有解释 这里不介绍&#xff09; 2.添加取样器 不用解释一看就知道填什么。。。 3.添加头信息&#xff08;否则请求头对不上&#xff09; 也不用解释。。。 4.配置监听器 可以尝试使用这几个监听器。 2.聚合结果…...

go、java、.net、C#、nodejs、vue、react、python程序问题进群咨询

1、面试辅导 2、程序辅导 3、一对一腾讯会议辅导 3、业务逻辑辅导 4、各种bug帮你解决。 5、培训小白 6、顺利拿到offer...

树莓派4B最新系统Bullseye 64 bit使用xrdp远程桌面黑屏卡顿问题

1、树莓派换源 打开源文件 sudo nano /etc/apt/sources.list注释原来的&#xff0c;更换为清华源 deb https://mirrors.tuna.tsinghua.edu.cn/debian/ bullseye main contrib non-free deb https://mirrors.tuna.tsinghua.edu.cn/debian/ bullseye-updates main contrib no…...

EasyExcel入门介绍及工具类,网络下载excel

前言&#xff1a;在这里分享自己第一次使用EasyExcel并且编写工具类&#xff0c;且在接口中支持excel文件下载的一系列流程&#xff0c;包含所有前后端&#xff08;JSJAVA&#xff09;完整代码&#xff0c;可以根据自己需要自行提取&#xff0c;仅供参考。 一.引入EasyExcel依赖…...

【HarmonyOS北向开发】-04 ArkTS开发语言-ArkTS基础知识

飞书原文档&#xff1a;Docs...

【Alibaba中间件技术系列】「RocketMQ技术专题」小白专区之领略一下RocketMQ基础之最!

应一些小伙伴们的私信&#xff0c;希望可以介绍一下RocketMQ的基础&#xff0c;那么我们现在就从0开始&#xff0c;进入RocketMQ的基础学习及概念介绍&#xff0c;为学习和使用RocketMQ打好基础&#xff01; RocketMQ是一款快速地、可靠地、分布式、容易使用的消息中间件&#…...

营销活动:提升小程序的用户活跃度的关键

在现今竞争激烈的商业环境中&#xff0c;小程序已成为企业私域营销的重要工具之一。然而&#xff0c;拥有一个小程序并不足以保证用户的活跃度。营销活动作为推动用户参与的有效方式&#xff0c;对于提升小程序的用户活跃度起着至关重要的作用。本文将深入探讨营销活动在提升小…...

Neo4j之CALL基础

CALL 语句用于调用 Neo4j 数据库中预定义的函数、过程或者自定义的函数。它是用来执行一些特定操作或计算的重要工具。以下是一些常用的 CALL 语句示例和解释&#xff1a; 调用内置函数&#xff1a; CALL db.labels()这个示例中&#xff0c;调用了内置函数 db.labels() 来获取…...

【TypeScript】元组

元组&#xff08;Tuple&#xff09;是 TypeScript 中的一种特殊数据类型&#xff0c;它允许你定义一个固定数量和类型的元素组合。元组可以包含不同类型的数据&#xff0c;每个数据的类型在元组中都是固定的。以下是 TypeScript 中元组的基本用法和特点&#xff1a; // 声明一…...

数据仓库一分钟

数据分层 一、数据运营层&#xff1a;ODS&#xff08;Operational Data Store&#xff09; “面向主题的”数据运营层&#xff0c;也叫ODS层&#xff0c;是最接近数据源中数据的一层&#xff0c;数据源中的数据&#xff0c;经过抽取、洗净、传输&#xff0c;也就说传说中的 ETL…...

提升Python代理程序性能的终极解决方案:缓存、连接池和并发

在开发Python代理程序时&#xff0c;优化性能是至关重要的。本文将为你介绍一套终极解决方案&#xff0c;通过缓存、连接池和并发处理等技术&#xff0c;极大地提升Python代理程序的效率和稳定性。 游戏国内地更换虚拟含ip地址数据库地区 1.缓存技术 缓存是 .0-*-696ES2 0一…...

CSS和AJAX阶段学习记录

1、AJAX的工作原理&#xff1a; 如图所示&#xff0c;工作原理可以分为以下几步&#xff1a; 网页中发生一个事件&#xff08;页面加载、按钮点击&#xff09; 由 JavaScript 创建 XMLHttpRequest 对象 XMLHttpRequest 对象向 web 服务器发送请求 服务器处理该请求 服务器将响应…...

Android自定义View知识体系

View的概念、作用和基本属性 View是Android中的基本UI组件&#xff0c;用于构建用户界面。它可以是按钮、文本框、图像等可见元素&#xff0c;也可以是容器&#xff0c;用于组织其他View。View的作用是展示数据和接收用户的输入。它可以显示文本、图片、动画等内容&#xff0c…...

Springboot 自定义 Mybatis拦截器,实现 动态查询条件SQL自动组装拼接(玩具)

前言 ps&#xff1a;最近在参与3100保卫战&#xff0c;战况很激烈&#xff0c;刚刚打完仗&#xff0c;来更新一下之前写了一半的博客。 该篇针对日常写查询的时候&#xff0c;那些动态条件sql 做个简单的封装&#xff0c;自动生成&#xff08;抛砖引玉&#xff0c;搞个小玩具&a…...

Go 1.21新增的 slices 包详解(三)

Go 1.21新增的 slices 包提供了很多和切片相关的函数&#xff0c;可以用于任何类型的切片。 slices.Max 定义如下&#xff1a; func Max[S ~[]E, E cmp.Ordered](x S) E 返回 x 中的最大值&#xff0c;如果 x 为空&#xff0c;则 panic。对于浮点数 E, 如果有元素为 NaN&am…...

Python 在logging.config.dictConfig()日志配置方式下,使用自定义的Handler处理程序

文章目录 一、基于 RotatingFileHandler 的自定义处理程序二、基于 TimedRotatingFileHandler 的自定义处理程序 Python logging模块的基本使用、进阶使用详解 Python logging.handlers模块&#xff0c;RotatingFileHandler、TimedRotatingFileHandler 处理器各参数详细介绍 …...

汽车门户网站开发/品牌策划包括哪几个方面

【来信】贺老师&#xff0c;您好&#xff0c;我是江西某高校软件学院的一名在校学生。看了您在csdn上公布的博文和视频&#xff0c;我获益良多。不得不承认&#xff0c;之前的大学时光我是荒废了&#xff0c;立即就要大三了&#xff0c;我主攻的是C方面。因此我悔过自新的想从头…...

学做网站赚钱方法/关键词优化公司如何选择

程序在开始处有一条注释&#xff08;使用新的注释风格&#xff09;&#xff0c;给出了文件名和程序的目的。写这种程序说明很简单、不费时&#xff0c;而且在以后浏览或打印程序时很有帮助。...

自己做网站 最好的软件下载/百度收录网站入口

下载示例程序代码 - 1,162.6 KB 前言 这篇文章出自于我尝试学习使用Nhiberbnate的挫败感。我发现好像Nhibernate全部的介绍材料不是很模糊就是太详细。我所需要的就是一个简单直接的教程&#xff0c;能让我尽快对NHibernate熟悉起来。我从来没有找到。幸运的是&#xff0c;这篇…...

绵阳医院网站建设/店铺推广引流的方法

Firebird2.5架构 &#xff0d;Dmitry Yemanov(2008年Firebird会议上的报告) 英文连接为&#xff1a;http://www.slideshare.net/ibsurgeon/firebird-25-architecture-by-dmitry-yemanov-in-english 一、该版本出现的原因 1.1 统一Classic/SuperServer模式的第一…...

怎么做搜索网站/百度推广天天打骚扰电话

♥️作者:小刘在C站 ♥️个人主页:小刘主页 ♥️每天分享云计算网络运维课堂笔记,努力不一定有收获,但一定会有收获加油!一起努力,共赴美好人生! ♥️夕阳下,是最美的绽放,树高千尺,落叶归根人生不易,人间真情 前言 本章Linuxshell讲解,感谢观看,干货满满。 目录…...

网站建设b2b/免费的网站软件

firebug因此&#xff0c;您可能想知道为什么只有在浏览器中打开Firebug时才运行jQuery代码。 嗯&#xff0c;这可能是因为您在代码中使用了console.log命令&#xff0c;而jQuery代码失败是因为控制台不存在。 要解决此简单问题&#xff0c;请将console.log和firebug命令放在以…...