Spring5学习笔记—AOP编程

news2024/10/6 8:29:00

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{
               m1
               m2
          }
          UserServiceImpl implements UserService{
               m1 ---> 业务运算 DAO调用
               m2 
          }
          UserServiceProxy implements UserService
               m1
               m2

2.4 编码

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

image-20230815112248257

2.5 静态代理存在的问题

1. 静态类文件数量过多,不利于项目管理
   UserServiceImpl  UserServiceProxy
   OrderServiceImpl 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{
    @Override
    public void register(User user) {
        System.out.println("UserServiceImpl.register");
    }

    @Override
    public 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 {
	//作用:给原始方法添加额外功能
    //注意:会在原始方法运行之前运行此方法
    @Override
    public 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 抛出的异常
         */
        @Override
        public 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 可能抛出的异常
         */
        @Override
        public 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() {
                @Override
                public 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 创建Enhancer
            Enhancer enhancer = new Enhancer();
            //  2.2 设置借用类加载器
            enhancer.setClassLoader(TestCGlibProxy.class.getClassLoader());
            //  2.3 设置父类(目标类)
            enhancer.setSuperclass(userService.getClass());
            //  2.4 设置回调,额外功能写在里面
            enhancer.setCallback(new MethodInterceptor() {
                //相当于 InvocationHandler --> invoke()
                @Override
                public 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 {
        @Override
        public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
            return bean;
        }
    
        @Override
        public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
            InvocationHandler invocation = new InvocationHandler() {
                @Override
                public 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动态代理创建方式 
      如果要切换Cglib
           1. 基于注解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;


    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
              this.ctx = applicationContext;
    }

    @Log
    @Override
    public 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");
    }

    @Override
    public boolean login(String name, String password) {
        System.out.println("UserServiceImpl.login");
        return true;
    }
}

8. AOP阶段知识总结

AOP 总结

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

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

相关文章

什么是服务网格,为什么 Kubernetes 需要它?

​企业现在热衷于采用微服务架构&#xff0c;因为它具有敏捷性和灵活性。容器和作为首选的容器编排工具—Kubernetes的兴起使得从单体架构向微服务架构的转变变得更加容易。然而&#xff0c;在大规模使用微服务架构时出现了一系列新的挑战&#xff1a; DevOps和架构师很难管理…

SpringBoot的日志信息及Lombok的常用注解

文章目录 一. 日志的介绍1. 什么是日志2. 日志的作用 二. 日志的使用1. 日志格式说明2. 自定义日志的输出3. 日志级别4. 日志级别的配置5. 日志持久化6. 更简单的输出日志-Lomok7. Lombok框架实现原理以及其他常见注解 一. 日志的介绍 1. 什么是日志 日志是我们程序重要组成部…

UE4/5数字人MetaHuman的控制绑定资产使用

开始操作 首先我们创建一个关卡序列&#xff1a; 打开后将我们的数字人放进去【右键&#xff0c;第一个添加进去】&#xff1a; 我们会自动进入动画模式&#xff0c;没有的话&#xff0c;就自己进入一下&#xff0c; 然后我们去寻找我们的控制绑定资产。 找到控制绑定资产 …

pdf文件如何编辑修改?学会这几种简单编辑方法

pdf文件如何编辑修改&#xff1f;在日常工作和学习中&#xff0c;我们经常需要编辑和修改PDF文件。无论是工作中的合同、报告或学习中的笔记、课件&#xff0c;PDF文件都是我们常用的文档格式。然而&#xff0c;PDF文件的特性使得它们很难被直接编辑。那么&#xff0c;如何编辑…

移远通信推出一站式Matter解决方案,构建智能家居开放新生态

近日&#xff0c;全球领先的S物联网整体解决方案供应商移远通信宣布&#xff0c;正式推出全新Matter解决方案&#xff0c;从模组、APP、平台、认证、生产五大层面为客户提供一站式服务&#xff0c;赋能智能家居行业加快融合发展。 过去十年&#xff0c;得益于物联网生态的发展&…

Hlang--用Python写个解析器

文章目录 前言语法树节点语法描述异常处理InvalidSyntaxError错误内容定位解析实现bin_op函数factory函数模拟流程总结前言 先前,我们实现了一个基本的词法分析器。那么现在的话,我们要在这个基础上面,实现一个解析器,那么实现效果的话,是这样的: 注意此时此刻,我们还…

一文教你快速 Cloudreve搭建云盘系统,实现随时访问

文章目录 1、前言2、本地网站搭建2.1 环境使用2.2 支持组件选择2.3 网页安装2.4 测试和使用2.5 问题解决 3、本地网页发布3.1 cpolar云端设置3.2 cpolar本地设置 4、公网访问测试5、结语 1、前言 自云存储概念兴起已经有段时间了&#xff0c;各互联网大厂也纷纷加入战局&#…

FLatten Transformer

FLatten Transformer: Vision Transformer using Focused Linear Attention ICCV 2023 聚焦式线性注意力模块 关于Transformer 在Transformer模型应用于视觉领域的过程中&#xff0c;降低自注意力的计算复杂度是一个重要的研究方向。线性注意力通过两个独立的映射函数来近似S…

springBoot 配置文件 spring.resources.add-mappings 参数的作用

在Spring Boot应用中&#xff0c;spring.resources.add-mappings参数用于控制是否将特定路径的资源文件添加到URL路径映射中。 默认情况下&#xff0c;该参数的值为true&#xff0c;即会自动将静态资源&#xff08;例如CSS、JavaScript、图片等&#xff09;的URL路径添加到Spr…

Mybatis Plus中使用LambdaQueryWrapper进行分页以及模糊查询对比传统XML方式进行分页

传统的XML分页以及模糊查询操作 传统的XML方式只能使用limit以及offset进行分页&#xff0c;通过判断name和bindState是否为空&#xff0c;不为空则拼接条件。 List<SanitationCompanyStaff> getSanitationStaffInfo(Param("name") String name,Param("bi…

〔013〕Stable Diffusion 之 图片自动评分和不健康内容过滤器 篇

✨ 目录 &#x1f388; 下载咖啡美学评价插件&#x1f388; 咖啡美学评价使用&#x1f388; 不健康内容过滤器插件 &#x1f388; 下载咖啡美学评价插件 想让系统帮你的图片作品打分评价&#xff0c;可以下载咖啡美学自动评价插件插件地址&#xff1a;https://github.com/p1at…

记录--webpack和vite原理

这里给大家分享我在网上总结出来的一些知识&#xff0c;希望对大家有所帮助 前言 每次用vite创建项目秒建好&#xff0c;前几天用vue-cli创建了一个项目&#xff0c;足足等了我一分钟&#xff0c;那为什么用 vite 比 webpack 要快呢&#xff0c;这篇文章带你梳理清楚它们的原理…

Python“牵手”1688商品评论数据采集方法,1688API申请指南

1688平台API接口是为开发电商类应用程序而设计的一套完整的、跨浏览器、跨平台的接口规范&#xff0c;1688API接口是指通过编程的方式&#xff0c;让开发者能够通过HTTP协议直接访问1688平台的数据&#xff0c;包括商品信息、店铺信息、物流信息等&#xff0c;从而实现1688平台…

【校招VIP】java语言类和对象之map、set集合

考点介绍&#xff1a; map、set集合相关内容是校招面试的高频考点之一。 map和set是一种专门用来进行搜索的容器或者数据结构&#xff0c;其搜索效率与其具体的实例化子类有关系。 『java语言类和对象之map、set集合』相关题目及解析内容可点击文章末尾链接查看&#xff01; …

dm达梦数据库 创建 YEARWEEK 外部函数

这里写目录标题 数据库环境创建方式一、C 外部函数二、JAVA 外部函数 数据库环境 类型版本官网下载地址基础环境Windows 11–Dm8dm8_20230418_x86_win_64.zip链接: 官网 创建方式 一、C 外部函数 &#xff08;1&#xff09;C 外部函数是使用 C、C 语言编写&#xff0c;在数据…

PostgreSQL-UDF用户自定义函数-扩展插件

目录 PostgreSQL-UDF用户自定义函数-扩展插件零、前置条件一、创建 .c 和 .sql 文件创建.c文件创建.sql文件 二、创建 .control 和 Makefile 文件创建 .control 文件创建 Makefile 文件 三、编译 & 链接四、psql&#xff08;或者其他PG backend&#xff09;中创建扩展 Post…

illustrator-形状模式-路径查找器-路径偏移-扩展

文章目录 1.形状模式1.1.联集&#xff1a;形状相加1.2.减去顶层&#xff1a;形状减去共同部分1.3.交集&#xff1a;相同部分1.4.差集&#xff1a;形状不同部分 2.路径查找器2.1.分割2.2.修边2.3.合并2.4.裁剪2.5.轮廓2.6.减去后方对象 3.路径偏移4.扩展外观 打开路径查找器快捷…

吃雪糕也能涨知识了!猿辅导携手中街1946推出“冷知识冰棍”

为了给孩子们的暑假学习加点“料”&#xff0c;猿辅导近日脑洞大开&#xff0c;和中街1946携手推出了“冷知识冰棍”&#xff0c;以数学、英语、语文、科学4个科目为外包装&#xff0c;分别对应草莓山楂、青提菠萝、茉莉蓝莓和蜜桃乌龙等4种口味&#xff0c;为孩子们开启了夏日…

在 SwiftUI 中创建一个环形 Slider

文章目录 前言初始化环形轮廓将进度值和拇指位置绑定添加触摸手势为不同的坐标值设置滑块位置总结 前言 Slider 控件是一种允许用户从一系列值中选择一个值的 UI 控件。在 SwiftUI 中&#xff0c;它通常呈现为直线上的拇指选择器。有时将这种类型的选择器呈现为一个圆圈&#…

centos7、ky10_server(arm版、x86版)将程序做成系统服务

前提条件 提供启动脚本、停止脚本 启动脚本 生成app启动脚本 touch app_start.sh cat > app_start.sh << EOF chmod -R 777 /home/zenglg/appcd /home/zenglg/app/apache-tomcat/bin./startup.sh EOF 停止脚本 生成app.停止脚本 touch app_stop.sh cat > app_sto…