[笔记]Spring AOP

Spring AOPAspect Oriented Programming

  • AOP将应用程序分为核心业务和非核心的公共功能,AOP的关注点是系统中的非核心的公共功能;

  • AOP可以通过预编译或者运行期动态代理的方式,为横跨多个对象(没有继承关系..)的业务逻辑添加统一的功能横切关注点Cross Cutting Concern:事务、日志记录、权限检查、异常处理等)

  • 使用AOP可以实现在不修改核心业务源代码的情况下为核心业务添加统一的功能,实现了核心业务和非核心的公共功能之间的解耦,提高了代码的重用性;

基于注解的AOP

在Spring Boot中,AOP主要通过注解来实现,以简化配置和提升代码的可读性。

@Aspect注解
  • 作用:将一个类定义为切面类。

  • 用法:与@Component@Service等注解结合使用,以便Spring容器管理这个切面。

定义切点(Pointcut)
  • @Pointcut:定义了何处应用AOP(比如哪些方法)。

  • 表达式:使用execution表达式指定方法模式。

@Pointcut("execution(* com.example.service.*.*(..))")
public void serviceLayer() {
}
定义通知(Advice)
  • @Before:在方法执行之前执行。

  • @AfterReturning:方法成功执行后执行。

  • @AfterThrowing:方法抛出异常后执行。

  • @After:方法执行后执行,无论其结果如何。

  • @Around:在方法执行前后执行,提供了最大的灵活性。

@Before("serviceLayer()")
public void logBeforeService(JoinPoint joinPoint) {
    // 实现逻辑
}
实际应用示例
  • 切面类:结合@Aspect@Component定义切面。

  • 通知方法:使用@Before@After等注解定义不同类型的通知。

@Aspect
@Component
public class LoggingAspect {
    @Before("serviceLayer()")
    public void logBefore(JoinPoint joinPoint) {
        // 日志逻辑
    }

    // 其他通知定义
}
在Spring Boot中的使用
  • 依赖:确保已添加AOP相关依赖(如spring-boot-starter-aop)。

  • 注解:直接在类和方法上使用@Aspect和通知相关的注解。

  • 无需XML:不需要XML配置,Spring Boot会自动处理这些注解。

静态代理 vs 动态代理

  1. 静态代理

    1. 实现方式:

      • 通过继承或实现接口。

      • 需要为每个目标类创建一个代理类。

    2. 缺点:

      • 高耦合度:代理类和目标类紧密绑定。

      • 可扩展性差:每个新类都需要新的代理类。

  2. 动态代理

    1. 在运行时动态创建代理类,不需要显式地为每个目标类编写代理类。

    2. 分为两种类型:

      • JDK动态代理:

        • 只能代理实现了接口的类。

        • 使用java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口实现。

      • CGLIB动态代理:

        • 可以代理没有实现接口的类。

        • 通过继承目标类来实现代理。

        • 使用net.sf.cglib.proxy.Enhancer类和net.sf.cglib.proxy.MethodInterceptor接口实现。

补充示例:静态代理与动态代理

静态代理示例:

在Spring框架中,静态代理通常不通过XML配置来实现,因为静态代理涉及到手动创建代理类,而不是在运行时动态生成代理对象。在静态代理中,你会直接编写一个代理类,它实现与目标对象相同的接口,并在代理类中显式调用目标对象的方法。

由于这种方式并不涉及Spring的AOP特性,所以没有特定的XML配置来声明静态代理。

public interface UserService {
    void addUser();
    void deleteUser();
}

public class UserServiceImpl implements UserService {
    public void addUser() {
        System.out.println("添加用户");
    }

    public void deleteUser() {
        System.out.println("删除用户");
    }
}

public class UserServiceProxy implements UserService {
    private UserServiceImpl userService;

    public UserServiceProxy(UserServiceImpl userService) {
        this.userService = userService;
    }

    public void addUser() {
        System.out.println("执行前置逻辑");
        userService.addUser();
        System.out.println("执行后置逻辑");
    }

    public void deleteUser() {
        System.out.println("执行前置逻辑");
        userService.deleteUser();
        System.out.println("执行后置逻辑");
    }
}
XML配置示例
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
                           http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- 声明原始的UserService实现 -->
    <bean id="userService" class="com.example.service.impl.UserServiceImpl"/>

    <!-- 声明UserService的静态代理类 -->
    <bean id="userServiceProxy" class="com.example.service.impl.UserServiceProxy">
        <!-- 注入原始的UserService -->
        <constructor-arg ref="userService"/>
    </bean>

</beans>
动态代理示例:
  JDK动态代理

  JDK动态代理,也称为Java动态代理,是基于接口的代理方式,使用Java自带的代理机制来实现。

  • 原理: 通过实现目标对象的接口并在调用处理器(InvocationHandler)中定义拦截逻辑。

  • 使用场景: 适用于目标对象实现了接口的情况。

  • 示例代码:

public interface Teacher {
    void teach();
}
public class Wang implements Teacher {
    @Override
    public void teach() {
        System.out.println("老师讲课");
    }
}
/**
 * 代理对象的执行逻辑
 */
public class MyHandler implements InvocationHandler {

    Teacher teacher;

    public MyHandler(Teacher teacher) {
        this.teacher = teacher;
    }

    /**
     * Object proxy: 代理对象
     * Method method: 目标方法
     * Object[] args: 方法参数
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        /*
           // 前置 公共的功能
           System.out.println("上课前 - 检查手机");
           // 核心业务  -  执行目标对象的业务
           teacher.teach();
           // 后置 公共的功能
           System.out.println("放学后 - 解决问题");
         */
        System.out.println("上课前 - 检查手机");

        // 执行代理对象的业务逻辑
        Object result = method.invoke(teacher, args);

        System.out.println("放学后 - 解决问题");

        // 代理对象的方法返回值
        return result;
    }
}

 

XML配置

使用JDK动态代理时,通常不需要特别指定,因为这是Spring的默认行为。

但可以通过设置[aop:aspectj-autoproxy](aop:aspectj-autoproxy)标签的proxy-target-class属性为false来明确指示使用JDK动态代理。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
                           http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/aop
                           http://www.springframework.org/schema/aop/spring-aop.xsd">

    <!-- 管理目标类 -->
    <bean id="userService" class="com.example.service.impl.UserServiceImpl"/>

    <!-- 管理切面类 -->
    <bean id="myAspect" class="com.example.aspect.MyAspect"/>

    <!-- AOP配置 -->
    <aop:config>
        <aop:aspect id="aspect" ref="myAspect">
            <!-- 配置通知和切点 -->
            <!-- 示例: 前置通知 -->
            <aop:before method="beforeAdvice" pointcut="execution(* com.example.service.*.*(..))"/>
            <!-- 其他通知配置 -->
        </aop:aspect>
    </aop:config>

    <!-- 明确指定使用JDK动态代理 -->
    <aop:aspectj-autoproxy proxy-target-class="false"/>
</beans>
CGLIB动态代理:

CGLIB(Code Generation Library)代理是一种基于类的代理方式,能够代理没有实现接口的类。

  • 原理: 通过继承目标类并在方法拦截器(MethodInterceptor)中定义拦截逻辑。

  • 使用场景: 适用于目标对象没有实现任何接口的情况。

  • 示例代码:

/**
 * 目标对象 - 被代理对象
 */
public class JayZhou {

    public void teach() {
        System.out.println("老师讲课!");
    }

}
public class MyHandler implements MethodInterceptor {

    /**
     * @param proxy 代理对象
     * @param method 被拦截的方法
     * @param args argument 方法参数
     * @param methodProxy 用于执行父类的方法
     */
    @Override
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        System.out.println("上课前检查手机");

        // 执行目标对象的方法   super.method
        Object result = methodProxy.invokeSuper(proxy, args);

        System.out.println("放学后解决问题");

        return result;
    }
}
XML配置

要使用CGLIB代理,您需要将[aop:aspectj-autoproxy](aop:aspectj-autoproxy)标签的

proxy-target-class属性设置为true

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
                           http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/aop
                           http://www.springframework.org/schema/aop/spring-aop.xsd">

    <!-- 管理目标类 -->
    <bean id="userService" class="com.example.service.impl.UserServiceImpl"/>

    <!-- 管理切面类 -->
    <bean id="myAspect" class="com.example.aspect.MyAspect"/>

    <!-- AOP配置 -->
    <aop:config>
        <aop:aspect id="aspect" ref="myAspect">
            <!-- 配置通知和切点 -->
            <!-- 示例: 前置通知 -->
            <aop:before method="beforeAdvice" pointcut="execution(* com.example.service.*.*(..))"/>
            <!-- 其他通知配置 -->
        </aop:aspect>
    </aop:config>

    <!-- 明确指定使用CGLIB代理 -->
    <aop:aspectj-autoproxy proxy-target-class="true"/>
</beans>

JDK代理和CGLIB代理的比较

  • 应用场景:

    • JDK代理只适用于接口方法。

    • CGLIB可以代理类方法,无需接口。

  • 性能:

    • CGLIB通常性能更优,直接操作字节码。

    • JDK代理使用反射,性能略逊。

  • 实现方式:

    • JDK代理使用Java原生反射。

    • CGLIB通过字节码生成。

无论是使用JDK代理还是CGLIB代理,Spring的配置方式基本相同,主要区别在于[aop:aspectj-autoproxy](aop:aspectj-autoproxy)标签的proxy-target-class属性的设置。


AOP的用途

  1. 横切关注点

    1. AOP允许将应用程序中跨越多个点的功能(如日志、事务管理、安全等)模块化为特殊的类,称为"切面"(Aspects)。

    2. 这些横切关注点通常与业务逻辑无关,但对多个模块都有影响。

  2. 解耦

    1. 通过将非业务代码(如日志和安全)从业务代码中分离,AOP有助于降低模块间的耦合度。

    2. 提高了代码的可维护性和可重用性。

Spring AOP 关键术语和概念

  1. JoinPoint(连接点): 指那些可能被拦截到的点。在Spring AOP中,这通常指的是方法的执行。

  2. PointCut(切点): 指明哪些JoinPoint(方法)需要被拦截的规则集合。切点的表达式决定了哪些方法会被增强。

  3. Advice(通知/增强): 绑定到特定JoinPoint(通过PointCut选择)上的动作。常见的类型有:

    1. Before(前置通知)

    2. After Returning(后置通知)

    3. After Throwing(异常通知)

    4. After(最终通知)

    5. Around(环绕通知)

  4. Target(目标对象): 包含JoinPoint的对象。它是被代理的对象。

  5. Introduction(引介): 用于给类添加新的方法和属性的AOP概念。

  6. Proxy(代理): 为Target对象提供的代理,用于拦截对Target对象的调用。

  7. Weaving(织入): 将通知应用到目标对象以创建新的代理对象的过程。织入可以在编译时、类加载时或运行时进行。

  8. Aspect(切面): 通知(Advice)和切点(PointCut)的结合。切面定义了何时(切点)和如何(通知)进行跨越应用程序多个点的行为。

依赖:

        <!--aspectJ的依赖-->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.8.7</version>
        </dependency>

Spring AOP与AspectJ密切相关,但在实际使用中,Spring AOP通常被认为是AspectJ的简化版,专注于方法拦截。

xml演示:

<!--管理目标类-->
    <bean id="userService" class="com.xq.service.impl.UserServiceImpl"></bean>

    <!--管理切面类-->
    <bean id="myAspect" class="com.xq.aspect.MyAspect"></bean>

    <!--
      配置目标类和切面类
    -->
    <aop:config>
        <!--
           配置切点
           aop:pointcut  配置切点的标签
                id: 切点的名称 自定义的 唯一即可
                expression: 切点表达式
                     execution: 切点表达式的固定写法  切点方法的形参类型 包名+类名+方法名(参数类型)
                                * 切点方法的返回值任意
                                (..) 代表的是切点方法的参数任意
        -->
        <aop:pointcut id="p1" expression="execution(* com.xq.service.impl.UserServiceImpl.deleteUser(..))"></aop:pointcut>
        <aop:pointcut id="p2" expression="execution(* com.xq.service.impl.UserServiceImpl.findAll(..))"></aop:pointcut>
        <aop:pointcut id="p3" expression="execution(* com.xq.service.impl.UserServiceImpl.updateUser(..))"></aop:pointcut>
        <aop:pointcut id="p4" expression="execution(* com.xq.service.impl.UserServiceImpl.addUser(..))"></aop:pointcut>
        <!--
          配置代理对象 将通知应用到切点方法上
          aop:aspect标签  配置代理对象
               ref: 引用的就是切面类的id
        -->
        <aop:aspect ref="myAspect">
            <!--
               aop:before 前置通知的配置
                   method: 增强的方法的名称
                   pointcut-ref:引用哪一个切点
            -->
            <aop:before method="checkPrivilege" pointcut-ref="p1"></aop:before>
            <!--
               后置通知的配置
               aop:after-returning: 描述后置通知的标签
            -->
            <aop:after-returning method="printLog" pointcut-ref="p2"></aop:after-returning>

            <!--
               配置环绕通知
               aop:around  环绕通知的标签
            -->
            <aop:around method="around" pointcut-ref="p3"></aop:around>

            <!--
              抛出异常通知
              aop:after-throwing 抛出异常通知的标签
            -->
            <aop:after-throwing method="throwing" pointcut-ref="p4"></aop:after-throwing>

            <!--
               最终通知的标签 aop:after
            -->
            <aop:after method="after" pointcut-ref="p4"></aop:after>
        </aop:aspect>
    </aop:config>

注解:

  别忘了先对切面类扛注释

  @Aspect注解标记了一个类作为切面。

  @Before, @AfterReturning, @Around, @AfterThrowing, 和 @After 注解分别用于定义不同类型的通知。

@Component
@Aspect //标识当前类是一个切面类

  逐条注解:



/**
 * 切面类
 */
@Component
@Aspect //标识当前类是一个切面类
public class MyAspect {

   

    //权限校验的方法  @Before注解: 前置通知的注解
    @Before(value = "execution(* com.xq.service.impl.UserServiceImpl.deleteUser(..))")
    public void checkPrivilege(){
        System.out.println("开启了权限校验......");
    }

    //打印日志的方法 @AfterReturning注解  后置通知的注解
    @AfterReturning(value = "execution(* com.xq.service.impl.UserServiceImpl.findAll(..))")
    public void printLog(){
        System.out.println("开启了日志打印功能");
    }

    //开启环绕通知的方法 环绕通知:就是在目标方法前后都执行的方法  @Around 环绕通知的注解
    @Around(value = "execution(* com.xq.service.impl.UserServiceImpl.updateUser(..))")
    public void around(ProceedingJoinPoint joinPoint){
        try {
            System.out.println("开启了环绕通知1");
            joinPoint.proceed();
            System.out.println("开启了环绕通知2");
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
    }

    //抛出异常通知 只有目标方法出现异常之后,才会执行的增强方法 @AfterThrowing 抛出异常通知的注解
    @AfterThrowing(value = "execution(* com.xq.service.impl.UserServiceImpl.addUser(..))")
    public void throwing(){
        System.out.println("抛出了运行时异常....");
    }

    //最终通知 不管目标方法有没有出现异常 都会执行该增强方法
    @After(value = "execution(* com.xq.service.impl.UserServiceImpl.addUser(..))")
    public void after(){
        System.out.println("最终通知的方法执行了.....");
    }
}

  抽象方法注解:

  • @Pointcut 注解用于定义切点表达式。

  • 其他注解(如 @Before, @Around 等)则用于将方法标记为特定类型的通知,并通过它们的value属性指向对应的切点。



/**
 * 切面类
 */
@Component
@Aspect //标识当前类是一个切面类
public class MyAspect {

    //定义切点
    @Pointcut(value = "execution(* com.xq.service.impl.UserServiceImpl.deleteUser(..))")
    public void pointCut1(){

    }

    //权限校验的方法  @Before注解: 前置通知的注解
    //@Before(value = "execution(* com.xq.service.impl.UserServiceImpl.deleteUser(..))")
    @Before(value = "pointCut1()")
    public void checkPrivilege(){
        System.out.println("开启了权限校验......");
    }

}

applicationContext.xml配置:

applicationContext.xml文件中的配置启用了基于注解的AOP。这样,Spring可以自动检测带有@Aspect注解的类,并根据其中定义的注解来配置AOP。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--
       开启包扫描
    -->
    <context:component-scan base-package="com.xq"></context:component-scan>

    <!--
      开启spring对aop的注解支持
    -->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Elasticsearch:Simulate ingest API

Ingest pipeline 为我们摄入数据提供了极大的方便。在我之前的文章中&#xff0c;有非常多的有关 ingest pipeline 的文章。请详细阅读文章 “Elastic&#xff1a;开发者上手指南”。针对一组提供的文档执行摄取管道&#xff0c;可以选择使用替代管道定义。 Simulate ingest AP…

Parallels Desktop 19 mac 虚拟机软件 兼容M1 M2

Parallels Desktop 19 for Mac 是一款适用于 macOS 的虚拟机软件。无需重启即可在 Mac 上运行 Windows、Linux 等系统&#xff0c;具有速度快、操作简单且功能强大的优点。包括 30 余种实用工具&#xff0c;可简化 Mac 和 Windows 上的日常任务。 软件下载&#xff1a;Parallel…

小新22-IAP,24-IAP,27-IAP(F0GG,F0GH,F0GJ)原厂Win11.22H2系统

lenovo联想小新22寸,24寸,27寸IAP原装出厂Windows11系统镜像还原包&#xff0c;恢复出厂开箱状态 适用型号&#xff1a; 联想小新27-IAP(F0GJ),小新24-IAP(F0GH),小新22-IAP(F0GG) IdeaCentre AIO 3 22IAP7,IdeaCentre AIO 3 24IAP7,IdeaCentre AIO 3 27IAP7 链接&#xff1…

【DevOps】Jenkins Extended E-mail 邮件模板添加自定义变量

文章目录 1、配置Jenkins邮箱2、配置告警模板1、配置Jenkins邮箱 略 2、配置告警模板 自定义变量:DYSK_PYTEST_STATUS // Uses Declarative syntax to run commands inside a container. pipeline {agent {kubernetes {cloud "kubernetes" //选择名字是kuberne…

关于网络模型的笔记

1. OSI 七层参考模型&#xff1a; 简介&#xff1a; 七层模型&#xff0c;亦称 OSI&#xff08;Open System Interconnection&#xff09;参考模型&#xff0c;即开放式系统互联。参考模型 是国际标准化组织&#xff08;ISO&#xff09;制定的一个用于计算机或通信系统间互联…

解决docker desktop 登录不上账号的问题

一、背景 点击“Sign in”&#xff0c;一直卡在Verifying credentials...&#xff0c;重试也没用。 二、解决办法 1、macOS下载并安装Proxifier 2、配置Proxifier 配置Proxies 配置rule 其中的Applications填&#xff1a;"Docker.app"; "Docker"; com.…

【极数系列】Flink环境搭建(02)

【极数系列】Flink环境搭建&#xff08;02&#xff09; 引言 1.linux 直接在linux上使用jdk11flink1.18.0版本部署 2.docker 使用容器部署比较方便&#xff0c;一键启动停止&#xff0c;方便参数调整 3.windows 搭建Flink 1.18.0版本需要使用Cygwin或wsl工具模拟unix环境…

web安全学习笔记【07】——非http\https抓包

#知识点&#xff1a; 1、Web常规-系统&中间件&数据库&源码等 2、Web其他-前后端&软件&Docker&分配站等 3、Web拓展-CDN&WAF&OSS&反向&负载均衡等 ----------------------------------- 1、APP架构-封装&原生态&H5&flutter…

网络协议与攻击模拟_07UDP协议

一、简单概念 1、UDP协议简介 UDP&#xff08;用户数据报&#xff09;协议&#xff0c;是传输层的协议。不需要建立连接&#xff0c;直接发送数据&#xff0c;不会重新排序&#xff0c;不需要确认。 2、UDP报文字段 源端口目的端口UDP长度UDP校验和 3、常见的UDP端口号 5…

智能GPT图书管理系统(SpringBoot2+Vue2)、接入GPT接口,支持AI智能图书馆

☀️技术栈介绍 ☃️前端主要技术栈 技术作用版本Vue提供前端交互2.6.14Vue-Router路由式编程导航3.5.1Element-UI模块组件库&#xff0c;绘制界面2.4.5Axios发送ajax请求给后端请求数据1.2.1core-js兼容性更强&#xff0c;浏览器适配3.8.3swiper轮播图插件&#xff08;快速实…

LiveGBS流媒体平台GB/T28181常见问题-如何配置使用自己已有的redis服务替换redis版本升级redis版本

LiveGBS如何配置使用自己已有的redis服务替换redis版本升级redis版本 1、Redis服务2、如何切换REDIS?2.1、停止启动REDIS2.2、配置信令服务2.3、配置流媒体服务2.4、启动 3、搭建GB28181视频直播平台 1、Redis服务 在LivGBS中Redis作为数据交换、数据订阅、数据发布的高速缓存…

Redis(七)复制

文章目录 是什么功能配置配主库不配从库权限细节 案例配置文件修改 一主二仆固定配置文件主从问题命令操作手动指定 薪火相传反客为主复制原理和工作流程存在问题 是什么 https://redis.io/docs/management/replication/ 就是主从复制&#xff0c;master以写为主&#xff0c;S…

Vue中的常用指令

Vue中的常用指令 概念&#xff1a;指令&#xff08;Directives&#xff09;是 Vue 提供的带有 v- 前缀 的特殊标签属性。 为啥要学&#xff1a;提高程序员操作 DOM 的效率。 vue 中的指令按照不同的用途可以分为如下 6 大类&#xff1a; 内容渲染指令&#xff08;v-html、v…

【制作100个unity游戏之23】实现类似七日杀、森林一样的生存游戏2(附项目源码)

本节最终效果演示 文章目录 本节最终效果演示系列目录前言添加小动物模型动画动物AI脚本效果 添加石头石头模型拾取物品效果 源码完结 系列目录 【制作100个unity游戏之23】实现类似七日杀、森林一样的生存游戏1&#xff08;附项目源码&#xff09; 【制作100个unity游戏之23】…

Mac Idea安装后无法启动

1、起因 想安装一个新版的idea2023.3.2&#xff0c;结果安装完之后直接无法启动 以为是卸载不干净&#xff0c;下载了一个腾讯柠檬&#xff0c;结果将2018版也一并卸载了 好家伙&#xff0c;彻底没得用 2、找原因 1&#xff09;查看idea报错信息 网上找了一圈&#xff0c;其…

burp靶场--host攻击

burp靶场–host攻击 https://portswigger.net/web-security/host-header#what-is-an-http-host-header-attack 在本节中&#xff0c;我们将讨论错误配置和有缺陷的业务逻辑如何通过 HTTP 主机标头使网站遭受各种攻击。我们将概述用于识别易受 HTTP 主机标头攻击的网站的高级方…

RabbitMQ中交换机的应用及原理,案例的实现

目录 一、介绍 1. 概述 2. 作用及优势 3. 工作原理 二、交换机Exchange 1. Direct 2. Topic 3. Fanout 三、代码案例 消费者代码 1. 直连direct 生产者代码 测试 2. 主题topic 生产者代码 测试 3. 扇形fanout 生产者代码 测试 每篇一获 一、介绍 1. …

告别无法访问的Github

告别无法访问的Github 最近在使用github的时候又登不上去了&#xff0c;挂着VPN都没用 但是自己很多项目都存在github&#xff0c;登不上去那不得损失很大 所以一行必须整点儿特殊手段来访问&#xff0c;顺便分享一下 1.加速器 网上很多解决方案都是在分享各种加速器来登陆…

深入理解JS语法与变量

深入理解JS语法与变量 前言初识JavaScriptJavaScript的语言风格和特性 JavaScript的书写位置认识输出语句学会处理报错REPL 环境 变量定义变量改变变量变量的合法命名变量的默认值变量的常见错误等号表示赋值同时声明多个变量 变量声明提升注意事项 结语 前言 在现代Web开发中…

idea中debug Go程序报错error layer=debugger could not patch runtime.mallogc

一、问题场景 在idea中配置了Go编程环境&#xff0c;可以运行Go程序&#xff0c;但是无法debug&#xff0c;报错error layerdebugger could not patch runtime.mallogc: no type entry found, use ‘types’ for a list of valid types 二、解决方案 这是由于idea中使用的d…