AOP设计思想与实用

news2025/1/17 21:34:17

文章目录

  • 一、AOP思想
    • (一)什么是AOP
    • (二)为什么要使用AOP
  • 二、Spring AOP
    • (一)AOP 的组成
      • 1. Join Point(连接点)
      • 2. Pointcut(切点)
      • 3. Advice(通知)
      • 4. Aspect(切面)
    • (二)AOP实现
      • 1. 添加 Spring AOP 框架的支持
      • 2. 定义切面和切点
      • 3. 定义 Advice
    • (三)Spring AOP 实现原理
  • 三、基于 AOP 思想实现 Spring Boot 统一功能处理
    • (一)统一用户登录身份校验
      • 1. Spring AOP 用户统一登录验证的问题
      • 2. Spring 拦截器
      • 3. 拦截器实现原理
      • 4. 扩展:添加统一访问前缀
    • (二)统一异常处理
    • (三)统一数据返回格式
      • 1. 为什么要统一数据返回格式
      • 2. 统一数据返回格式的实现

一、AOP思想

(一)什么是AOP

  百度百科:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
  AOP它是一种思想,是对某一类事情的集中处理。换句话说,如果我们在做某件事之前,必须先进行另一件事,而另一件事很多地方都会用到,此时我们就可以把另一件事单独放在一处集中处理。例如访问页面前的登录校验

(二)为什么要使用AOP

用户校验
  往常我们进行登录校验一般的做法是将登录校验的过程封装成一个方法,如果某个路由需要进行登录校验,就进行调用,但是这么做也有其弊端,那就是污染了业务代码,如果有一天我们方法的参数类型或者数量改变,那么就必须对所有调用了这个方法的业务代码进行修改,如果有几千个,那么无疑是一个很大的工程

  AOP的思想正是解决这个问题,往常方法都是被动调用,而 AOP 的做法却是主动出击,如果某个路由需要进行登陆验证,AOP会直接将请求拦截,先进行判断,没问题了之后才可以访问,这么做的好处是一方面不需要具体的业务代码进行手动判断,也就不会对业务代码造成污染;另一方面也就降低了开发和维护的成本

AOP 的用途不止是统一的用户登录验证,还可以实现:

  • 统一日志管理
  • 统一方法执行时间统计
  • 统一的返回格式设置
  • 统一的异常处理
  • 事务的开启和提交等

换句话说,使用 AOP 可以扩充多个对象的某个能力,这也就是为什么说 AOP 是 OOP 的补充和完善

二、Spring AOP

  AOP 和 Spring AOP 的关系就像 IoC 和 DI 的关系,Spring AOP 框架是对 AOP 思想的具体实现
  如果要学习 Spring AOP,那么我们需要先学习 AOP 是如何组成的

(一)AOP 的组成

1. Join Point(连接点)

  Join Point表示在程序中明确定义的点,典型的包括方法调用,对类成员的访问以及异常处理程序块的执行等等,它自身还可以嵌套其它 joint point
  通俗来讲就是各种业务代码中需要借助 AOP 实现的部分

2. Pointcut(切点)

  Pointcut 的作⽤就是提供⼀组规则(使⽤ AspectJ pointcut expression language 来描述)来匹配 Join Point,给满⾜规则的 Join Point 添加 Advice
  通俗来讲 Pointcut 就是 Join Point 的集合,这个集合中的每一个连接点就是具体要解决的某个问题,而存在这个问题的类或方法常常是用通配、正则表达式等方式进行统计

3. Advice(通知)

  Advice 定义了 Pointcut 需要进行的统一操作,它通过 before、after 和 around 来区别是在 Join Point 之前、之后还是替代执行的代码
  也就是执行统一操作的具体代码实现,往往是一个一个方法

Spring 切面类中,可以在方法上使用以下注释,会设置方法为通知方法,在满足(切点)条件之后通知本方法进行调用:

  • 前置通知使用 @Before:通知方法会在目标方法(即要访问的具体业务方法)调用之前执行
  • 后置通知使用 @After:通知方法会在目标方法返回或者抛出异常后调用
  • 返回之后通知使用 @AfterReturning:通知方法会在目标方法返回后调用
  • 抛异常后通知使用 @AfterThrowing:通知方法会在目标方法抛出异常之后调用
  • 环绕通知使用 @Around:通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行相应代码

4. Aspect(切面)

  Aspect由切点(Pointcut)和通知(Advice)组成
  Aspect 就是包含了 Advice、Pointcut 甚至是 Aspect的类,相当于 AOP 实现的某个功能的集合

(二)AOP实现

  接下来我们使用 Spring AOP 来实现以下 AOP 的功能,完成的目标是拦截所有 UserController 中的方法,并执行相应的通知事件
实现步骤如下:

  1. 添加 Spring AOP 框架的支持
  2. 定义切面和切点
  3. 定义通知

1. 添加 Spring AOP 框架的支持

  我们仍然是创建 Spring Boot 项目,添加的依赖仍然是Spring Boot DevToolsLombokSpring Web这三个,有同学好奇,为什么不添加 Spring AOP 的依赖,原因是我们在创建项目时顺便可以添加的依赖被称作起步依赖,是有限的,不包含所有依赖,Edit里也没有。因此需要我们去 Maven 仓库复制之后手动添加
  由于我们创建的是 Spring Boot项目,因此我们需要添加的是 Spring Boot Starter AOP依赖,又由于作者创建的 Spring Boot 项目版本是 2.7.11,因此也需要选择 2.7.11 版本的 Spring AOP依赖,最后复制并粘贴到 pom.xml 即可
Spring Boot AOP
引入依赖

2. 定义切面和切点

对切面和切点的示例代码如下:

@Aspect // 告诉框架我是一个切面类
@Component // 保证切面类随着项目的启动而启动
public class UserAspect {
    /**
     * 定义切点,配置拦截规则
     * execution 固定写法,表示拦截
     * * 表示拦截所有类型
     * com...UserController路径表示拦截的类
     * .*表示拦截该类的满足后面括号里的条件的所有方法
     * (..) 表示不管参数个数都进行拦截
     */
    @Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
    public void pointcut() {}
}

UserController中的方法:

@RequestMapping("/user")
@RestController
public class UserController {

    @RequestMapping("/hello")
    public void hello(String name) {
        System.out.println("hello " + name);
    }

    @RequestMapping("/hi")
    public void hi() {
        System.out.println("hi world");;
    }
}

  拦截规则限定了哪些类的哪些方法需要进入该切点,pointcut 方法为空方法,它不需要有方法体,此方法起到“标识的作用”,标识下面 Advice 方法具体指的是哪一个切点(因为切点可能有很多个)


切点表达式说明:
AspectJ 支持三种通配符

  • *:匹配任意字符,只匹配一个元素(包、类、方法、方法参数)
  • ..:匹配任意字符,可以匹配多个元素,在表示类时,必须和 *联合使⽤
  • +:表示按照类型匹配指定类的所有类,必须跟在类名后⾯,如 com.cad.Car+,表示继承该类的所有⼦类包括本身

切点表达式由切点函数组成,其中 execution() 是最常用的切点函数,用来匹配方法,语法为:

execution(<修饰符><返回类型><包.类.方法(参数)><异常>)

其中修饰符和异常可以省略,具体值和表达的含义如下:
1. 修饰符,一般省略

public 公共方法
*任意

2. 返回值,不能省略

void 没有返回值
String 返回值为字符串类型
等等各种类型的返回
*返回任意类型

3. 包

com.example.demo 固定包
com.example.demo.*.service demo 包下面,最后包名为service的包
com.exampli.demo… demo包下的所有子包
com.exampli.demo.*.service… demo包下,任意service包名下的所有子包

4. 类

UserController 指定类
Controller 以 Controller 名字结尾的类
User
以 User 开头的类
*任意类

5. 方法名,不能省略

hello 固定方法
hello* 以 hello 开头的方法
*hello 以 hello 结尾的方法
*任意方法

6. 参数

() 无参
(Integer) 一个整型
(Integer, Integer) 两个整型
其他类型与上述类似
(…) 参数任意

7. throws,可省略,一般不写

3. 定义 Advice

  Advice 定义的是被拦截的方法具体要执行的业务,比如用户登录权限验证方法就是具体要执行的业务,接下来我们来演示一下上述提到的前置、后置以及环绕通知注解的使用,具体实现如下:

@Aspect
@Component
public class UserAspect {
    @Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
    public void pointcut() {}

    // 前置通知
    @Before("pointcut()")
    public void beforeAdvice() {
        System.out.println("执行 before 通知");
    }

    // 后置通知
    @After("pointcut()")
    public void afterAdvice() {
        System.out.println("执行 after 通知");
    }

    // 环绕通知
    @Around("pointcut()")
    // ProceedingJoinPoint 就是执行连接点对象,即我们需要知道我们拦截的具体是哪个方法
    public Object aroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
        Object obj = null;
        System.out.println("Around 方法开始执行");
        // 执行被拦截的业务代码
        // 相当于我们执行了一半的环绕通知方法,调用执行业务代码
        // 等业务代码执行完毕,继续执行后面的环绕通知方法
        obj = joinPoint.proceed();
        System.out.println("Around 方法结束执行");
        return obj;
    }
}

测试结果:
通知实现

(三)Spring AOP 实现原理

  Spring AOP 是构建在动态代理基础上,因此 Spring 对 AOP 的⽀持局限于⽅法级别的拦截
我们需要明确两个概念:

  • 静态代理:目标类和代理类实现了相同的接口,在代理类中依赖了目标类,代理类的方法中调用了目标类的方法,并做了一些增强性的工作
  • 动态代理:在程序的执行过程中,使用jdk的反射机制,创建代理对象,并动态的指定代理的目标类

  静态代理和动态代理最大的区别就是动态代理可以在不改变原来目标方法功能的前提下,在代理中对目标方法功能进行增强
  Spring AOP 支持 JDK Proxy 和 CGLIB 方式实现动态代理。Spring 默认情况下,实现了接口的类,使用 AOP 会基于 JDK 生成代理类,没有实现接口的类,会基于 CGLIB 生成代理类。Spring Boot默认情况下使用 CGLIB
JDK 和 CGLIB 实现的区别

  1. JDK 实现:要求被代理类必须实现接⼝,之后是通过 InvocationHandlerProxy,在运⾏时动态的在内存中⽣成了代理类对象,该代理对象是通过实现同样的接⼝实现(类似静态代理接⼝实现的⽅式),只是该代理类是在运⾏时期,动态的织⼊统⼀的业务逻辑字节码来完成,节省了我们开发和维护的成本
  2. CGLIB 实现:被代理类可以不实现接⼝,是通过继承被代理类,在运⾏时动态的⽣成代理类对象。但是也导致一个问题,它无法代理被 final 修饰的类,此时只能使用 JDK 实现

织入(Weaving):代理的生成时机
  织入是把切面应用到目标对象并创建新的代理对象的过程,切⾯在指定的连接点被织⼊到⽬标对象中
  在目标对象的声明周期里有多个点可以进行织入,包括编译器,类加载期,运行期。Spring AOP 就是在运行期,也就是在应用运行的某一时刻将切面织入

最后,Spring AOP做的业务方向更偏向于事务、日志、分布式锁等,统一用户登录验证,统一异常处理等业务操作实现 Spring 提供了更好的方式

三、基于 AOP 思想实现 Spring Boot 统一功能处理

  接下来我们介绍 Spring Boot 统⼀功能处理模块

  1. 统一用户登录身份校验
  2. 统一数据格式返回
  3. 统一异常处理

(一)统一用户登录身份校验

1. Spring AOP 用户统一登录验证的问题

  为什么不能在 Spring AOP 切面中实现用户登录信息统一校验,主要是存在两个问题

  1. 很难获取到 HttpServletRequest 对象,也就很难获取到 HttpSession 对象
  2. 我们要对一部分方法进行拦截,而另一部分方法不拦截,比如注册方法和登录方法,这样的话排除方法的规则很难定义,甚至无法定义

那么我们该如何解决呢?

2. Spring 拦截器

  Spring 拦截器和 Spring AOP都是对 AOP 思想的具体实现,但是它是直接内置到 Spring 框架中的,不需要像 Spring AOP 一样引入依赖。对于上述问题 Spring 提供了具体的实现拦截器:HandlerInterceptor,拦截器的实现分为以下两个步骤:

1. 创建自定义拦截器,实现 HandlerInterceptor 接口的 preHandle (执行具体方法之前的预处理)方法
代码实现:

public class LoginInterceptor implements HandlerInterceptor {
    /**
     * 此方法返回 boolean 类型,如果为 true,表示验证通过,进入业务代码
     * 如果为 false,表示验证未通过,进行处理
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        // 仍然是通过 request 获取到 HttpSession 对象
        HttpSession session = request.getSession(false);
        // 假设 session 中存储了 userInfo 的数据,对 session 进行校验
        if(session != null && session.getAttribute("userInfo") != null) {
            // 用户登录信息校验通过
            return true;
        }
        // 如果用户信息校验未通过,跳转到登录界面 login.html
        // 或返回 403 / 401 没有权限码
        response.sendRedirect("/login.html");
        // response.setStatus(403);
        return false;
    }
}

2. 将自定义拦截器加入 WebMvcConfigurer 的 addInterceptors 方法中
代码示例:

/**
 * Created with IntelliJ IDEA.
 * Description: 对当前系统的配置文件操作的类
 */
@Configuration
public class AppConfig implements WebMvcConfigurer {
    // 添加一个拦截器
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new LoginInterceptor2())
                .addPathPatterns("/**") // 拦截所有请求
                .excludePathPatterns("/user/login")
                .excludePathPatterns("/user/reg")
                .excludePathPatterns("/login.html");
    }
}

3. 拦截器实现原理

请求执行过程
  Spring 底层的逻辑是:在执行 Controller 之前,会先调用预处理方法 applyPreHandler,而 applyPreHandler 底层源码会先获取到所有的拦截器 HandlerInterceptor,并for循环执行所以拦截器中的 preHandle 方法,这就是拦截器的实现原理
  Spring 中的拦截器也是通过动态代理和环绕通知的思想实现的

4. 扩展:添加统一访问前缀

  大多时候,我们在一台服务器上不会只跑一个项目,但是我们在给一台服务器配域名的时候只能配一个ip的一个端口,因此我们为了确认用户访问的是一个项目,就要为一个项目的所有请求地址添加统一 api 前缀

我们有两种实现方式:

  1. 在系统的配置文件中配置

代码示例:

// 和 添加拦截器位置一样,也是在 WebMvcConfigurer 类里设置
// 为所有的接口添加统一前缀
@Override
public void configurePathMatch(PathMatchConfigurer configurer) {
    // 第一个参数表示统一前缀名
    // 第二个参数是 lambda 表达式,表示所有的 Controller 都要添加这个前缀
    configurer.addPathPrefix("/zhangsan", c -> true);
}
  1. 直接在application.properties或者application.yml中设置
server.servlet.context-path=/lisi

(二)统一异常处理

  统一异常处理使用的是 @ControllerAdvice + @ExceptionHandler 来实现的,@ControllerAdvice 表示控制器通知类,是 AOP 思想的实现,@ExceptionHandler 是异常处理器,两个结合表示当出现异常的时候执行某个通知,也就是执⾏某个方法事件,具体代码如下:

@ControllerAdvice
@ResponseBody
public class ExHandler {
    // 接收所有异常并规定返回格式
    // Exception.class 表示截取的是所有异常类型
    @ExceptionHandler(Exception.class)
    public HashMap<String, Object> exception(Exception e) {
        // 自定义异常返回数据格式
        HashMap<String, Object> map = new HashMap<>();
        map.put("code", "-1"); // 状态码
        map.put("msg", e.getMessage()); // 状态码的描述信息
        map.put("data", "null");
        return map;
    }

    // 接收空指针异常并返回规定格式
    @ExceptionHandler(NullPointerException.class)
    public HashMap<String, Object> nullException(NullPointerException e) {
        HashMap<String, Object> map = new HashMap<>();
        map.put("code", "-1"); // 状态码
        map.put("msg", "空指针异常:" + e.getMessage()); // 状态码的描述信息
        map.put("data", "null");
        return map;
    }
}

异常代码:

@RequestMapping("/hello")
public String hello() {
    Integer test = null;
    test.toString();
    System.out.println("hello world!");
    return "hello world";
}

@RequestMapping("/login")
public String login() {
    int result = 10 / 0;
    System.out.println("执行了 login");
    return "login";
}

测试结果:
空指针异常处理
算术异常处理

(三)统一数据返回格式

1. 为什么要统一数据返回格式

  在公司开发项目中,因此前后端开发的人员大概率不止一个,不同的人返回数据格式是不同的,而如果不进行统一,前端人员接收响应进行处理时将会很痛苦,因此统一数据返回格式是十分重要的,它有以下好处:

  1. 方便前端程序猿更好的接收和解析后端数据接⼝返回的数据
  2. 因为按照统一格式返回,降低了前端程序员和后端程序员的沟通成本
  3. 有利于项⽬统⼀数据的维护和修改
  4. 有利于后端技术部⻔的统⼀规范的标准制定

2. 统一数据返回格式的实现

  统一的数据返回格式可以使用 @ControllerAdvice + ResponseBodyAdvice 的方式实现,具体实现代码如下:

/**
 * 自定义统一处理返回数据格式
 * @param body 原始的数据返回结果
 * @param returnType
 * @param selectedContentType
 * @param selectedConverterType
 * @param request
 * @param response
 * @return 返回 新的 body
 */
@Override
public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
    // 这里选择返回 json 的数据格式
    HashMap<String, Object> result = new HashMap<>();
    result.put("code", "200");
    result.put("msg", "");
    result.put("body", body);
    return result;
}

  上述写法有些问题,当原 body 是字符串类型时,都 put 到 map 中之后直接返回就会报出java.util.HashMap cannot be cast to java.lang.String异常,具体原因可以参考下面这篇文章(https://blog.csdn.net/qq_37170583/article/details/107470337 ),修正的写法作者在这里推荐一种:

// 操作 jackson 的工具
// Spring Boot内置了 Jackson
@Autowired
private ObjectMapper objectMapper;

/**
 * 自定义统一处理返回数据格式
 * @param body 原始的数据返回结果
 * @param returnType
 * @param selectedContentType
 * @param selectedConverterType
 * @param request
 * @param response
 * @return 返回 新的 body
 */
@Override
public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
    // 这里选择返回 json 的数据格式
    HashMap<String, Object> result = new HashMap<>();
    result.put("code", "200");
    result.put("msg", "");
    result.put("body", body);
    if(body instanceof String) {
        try {
            return objectMapper.writeValueAsString(result);
        } catch (JsonProcessingException e) {
            e.printStackTrace();
        }
    }
    return result;
}

  说明:这个返回结果设置还是比较死板的,我们往往会在它之前设置一个工具类来设置统一对象,只有工具类不行了,才会用到它

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

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

相关文章

怎样做一个优秀的网关技术选型报告~

1、简介 当使用单体应用程序架构时&#xff0c;客户端&#xff08;Web 或移动端&#xff09;通过向后端应用程序发起一次 REST 调用来获取数据。负载均衡器将请求路由给 N 个相同的应用程序实例中的一个。然后应用程序会查询各种数据库表&#xff0c;并将响应返回给客户端。微服…

nodejs进阶(4)—读取图片到页面

我们先实现从指定路径读取图片然后输出到页面的功能。 先准备一张图片imgs/dog.jpg。 file.js里面继续添加readImg方法&#xff0c;在这里注意读写的时候都需要声明binary。&#xff08;file.js 在上一篇文章nodejs进阶3-路由处理中有完整的内容&#xff09; readImg:functi…

思维中的世界

⾝体的空间&#xff0c;以⾏为为导向的空间 感官⼩矮⼈ 当我们观察特定的事物时&#xff0c;⼤脑的相应区域就会被“点亮”&#xff0c;并变得 活跃起来。 ⾝体映射到⼤脑上&#xff0c;映射到“感官⼩矮⼈”上&#xff0c; 即从左⽿延伸&#xff0c;过⼤脑顶⾻&#xff0c;…

OpenCV 实战七 setMouseCallback 鼠标交互画框

鼠标交互画框效果&#xff1a; 目录 1、setMouseCallback()函数 2、on_Mouse函数 3、代码 1、setMouseCallback()函数 函数原型 void setMouseCallback(const String& winname, MouseCallback onMouse, void* userdata 0); 参数说明 winname 窗口名称 onMouse 鼠标…

全网最火爆,性能测试-测试用例与测试方法总结(超详细)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 性能测试用例 无…

视频截取动图怎么做?分享在线视频转gif小窍门

如何将一段视频制作成gif动图表情包呢&#xff1f;Gif表情包常见的制作方法就是将电影、电视剧中的某个画面截取出来做成gif表情包。那么&#xff0c;如何从视频中截取GIF呢&#xff1f; 一、怎样才能完成视频转gif制作&#xff1f; 通过使用GIF中文网的视频转gif&#xff08…

C语言算法--冒泡排序

C语言算法–冒泡排序 1-什么是冒泡排序 冒泡排序是一种简单的排序算法&#xff0c;它通过比较相邻元素的大小&#xff0c;并根据需要交换它们的位置来排序数据。它的名称来自于越小的元素会慢慢“冒泡”到数组的开头。 冒泡排序的基本思想是从数组的第一个元素开始&#xff…

【黑马笔记】web app项目初始化

文章目录 0. 工程介绍1. 使用模版初始化1.1 选择模版1.2 补充文件 2. 使用空项目初始化2.1 新建maven项目&#xff0c;直接新建2.2 补充文件2.2.1 补充packaging形式&#xff1a;war2.2.2 借助Facets自动补齐 0. 工程介绍 打包方式 新建maven项目&#xff0c;使用<packing&…

数据结构总结1:了解数据结构、时间复杂度、空间复杂度

后续可能会有补充和更改 目录 一、数据结构 1.算法介绍 二、时间复杂度、空间复杂度 三、练习 1.时间复杂度 2.空间复杂度 一、数据结构 数据结构是计算机存储、组织数据的方式&#xff0c;指相互之间存在一种或多种特定关系的数据元素的集合。 数据结构和数据库的区…

Jmeter测试POST请求

Jmeter测试POST请求 1、添加http请求 线程组->取样器->http请求 参数说明&#xff1a; 协议&#xff1a;http 服务器&#xff1a;10.1.1.26&#xff08;也就是ip地址&#xff09; 端口&#xff1a;8081 方法&#xff1a;POST 路径&#xff1a;例如&#xff…

滴滴 Java 一面面经

目录 1.了解Java集合嘛&#xff0c;详细说一下Map&#xff1f;2.为什么HashTable线程安全却不常用&#xff1f;3.HashMap不是线程安全&#xff0c;多线程下会出现什么问题&#xff1f;4.什么办法能解决HashMap线程不安全的问题呢5.ConcurrentHashmap是怎么实现的&#xff1f;6.…

ALOHA 开源机械臂(Viper 300 Widow X 250 6DOF机械臂 操作系统)第三部分

详情链接&#xff1a;https://tonyzhaozh.github.io/aloha/ Learning Fine-Grained Bimanual Manipulation with Low-Cost Hardware 用低成本硬件学习细粒度双手操作 Tony Zhao Vikash Kumar Sergey Levine Chelsea Finn Stanford University UC Berkeley Meta 斯坦福大学…

分布式锁解决方案_Zookeeper分布式锁原理

通过召zk实现分布式锁可靠性时最高的 公平锁和可重入锁的原理 取水秩序&#xff1a; &#xff08;1&#xff09;取水之前&#xff0c;先取号&#xff1b; &#xff08;2&#xff09;号排在前面的&#xff0c;就可以先取水&#xff1b; &#xff08;3&#xff09;先到的排在…

Go语言的学习【1】基础语法之前的准备事项

目录 什么是云原生学习方法go语言的IDE配置之VScode写go代码要注意的事情一些基本命令基础语法Go 语言原生自带测试Go vetPrint-format 错误&#xff0c;检查类型不匹配的printBoolean 错误&#xff0c;检查一直为 true、false 或者冗余的表达式Range 循环&#xff0c;比如如下…

ThingsBoard部署tb-gateway并配置OPCUA

1、安装 我实在自己的虚拟机上安装,使用官方Docker的安装方式 docker run -it -v ~/.tb-gateway/logs:/thingsboard_gateway/logs -v ~/.tb-gateway/extensions:/thingsboard_gateway/extensions -v ~/.tb-gateway/config:/thingsboard_gateway/config --name tb-gateway --…

《安富莱嵌入式周报》第312期:开源磁场照相机,仿生神经元PCB,开源无线耳机,手机系统PalmOS移植到各种单片机,开放系统组装协议OSAP

周报汇总地址&#xff1a;嵌入式周报 - uCOS & uCGUI & emWin & embOS & TouchGFX & ThreadX - 硬汉嵌入式论坛 - Powered by Discuz! 更新一期视频教程&#xff1a; BSP视频教程第26期&#xff1a;CAN/CANFD/CANopen专题&#xff0c;CANFD整个运行机制精…

PyTorch RNN的原理及其手写复现。

PyTorch RNN的原理及其手写复现。 记忆单元(考虑过去的信息)分类包括&#xff1a;1.RNN 2.GRU 3.LSTM模型类别&#xff1a;1.单向循环(左到右) 2.双向循环&#xff08;考虑未来信息&#xff09; 3.多层单向或双向循环优缺点应用场景具体公式 代码实现 记忆单元(考虑过去的信息)…

网络安全合规-数据安全评估

目前&#xff0c;我国在数据管理领域&#xff0c;已经正式出台的国家标准有《数据管理能力成熟度评估模型&#xff08;GB/T 36073-2018&#xff09;》&#xff08;DCMM&#xff09;&#xff0c;在数据安全检测评估、认证领域的标准有《数据安全能力成熟度模型&#xff08;GB/T …

实现取关和关注功能

将关注过的用户id存如数据库中 //关注或者取关 Override public Result follow(Long id, Boolean flag) { //1.获取当前登录用户的id UserDTO user UserHolder.getUser(); if(usernull){ return Result.fail("请先登录"); } Long userId user.getId(); //2.判断是关…

vue3+antDesignVue前端纯导出

效果 <a-buttonsize"default"style"margin-left: 10px"click"exportData">导出</a-button>1.下载所需依赖 npm install xlsx --save npm install file-saver --save<script setup> import { reactive, ref } from "vue…