Spring全家桶源码解析--2.6 Spring scope 限制bean的作用范围

文章目录

  • 前言
  • 一、Scope是什么?
  • 二、Scope使用
    • 2.1 单例:
      • 2.1.1 单例Bean的特点如下:
      • 2.1.2 单例设计模式 与单例bean:
    • 2.2 原型bean:
      • 2.2.1 原型Bean的特点:
      • 2.2.2 原型Bean的销毁:
    • 2.3 Request bean 和 Session bean:
      • 2.3.1 Request bean :
        • 2.3.1.1 Request bean 特点:
      • 2.3.2 session bean :
        • 2.3.1.2 session bean 特点:
      • 2.3.3 Request、Session bean的 实现:
      • 2.3.4 bean 的request,session作用域 同 web 浏览器 session会话和http 请求的关系:
  • 总结:


前言

在Maven 中我们可以通过 标签来现在依赖的作用范围,在Spring 中也提供@Scope 注解来限制bean 的作用范围。


一、Scope是什么?

在平时的开发中我们可能通过@Bean,@Component,@Import ,@Configuration 等注解来为容器中注入bean ,虽然我们没有声明,但是他们都被默认成了单例对象,Spring 开发者是基于什么考虑来提供@Scope 注解。
我们看下Scope 注解的作用:
@Scope注解是Spring框架中用于限定Bean作用范围的注解。它可以应用于Bean定义的类级别,以指定Bean的实例化和销毁规则

在Spring中,Bean的作用域定义了Bean对象在应用程序中的存在范围和生命周期,包括多例(prototype)、单例(singleton)、会话(session)、请求(request)等几种常用的作用域。
作用域的意义主要包括以下几个方面:

  • 控制对象的创建和销毁:通过定义不同的作用域,可以指定Bean对象的创建和销毁行为。例如,使用单例作用域的Bean在应用程序启动时创建,直到应用程序关闭时才销毁;而使用多例作用域的Bean在每次通过容器获取时都会创建一个新的实例,不由容器管理其生命周期。

  • 控制对象的共享与隔离:不同的作用域决定了Bean对象在应用程序中的共享程度。单例作用域的Bean是全局唯一的,可以在应用的任何地方共享;而多例作用域的Bean每次获取都会创建一个新的实例,相互之间是隔离的。

  • 提供灵活的对象管理:通过合理使用不同作用域的Bean,可以满足不同业务场景下的需求。比如,使用会话作用域的Bean可以在用户会话期间保持状态,而请求作用域的Bean可以处理每个HTTP请求的相关逻辑。

  • 提高性能和资源利用率:单例作用域的Bean在应用程序启动时被创建,可以在整个应用程序生命周期内重复使用,避免了重复创建对象的开销。这样可以提高应用程序的性能并节省资源的使用。

  • 作用域在Spring中起到了控制对象创建和销毁、对象共享与隔离、对象管理和性能优化等作用。根据实际需求,合理选择适当的作用域可以帮助我们更好地管理Bean对象,提高应用程序的性能和灵活性。

二、Scope使用

2.1 单例:

在Spring中,单例(Singleton)作用域是最常用的一种作用域。当一个Bean被定义为单例作用域时,Spring容器只会创建一个该类型的实例,并在容器的生命周期中共享这个实例

单例Bean在Spring容器启动时被创建,并保存在容器内部的一个缓存中。当应用程序需要获取该类型的Bean时,容器会返回已经存在的实例。这样可以避免重复创建对象的开销,提高应用程序的性能和资源利用率。

在平时的开发中我们没有定义@Scope 那么这个bean 默认就是单例的;

2.1.1 单例Bean的特点如下:

  • 全局唯一性:单例Bean在应用程序的整个生命周期内只会创建一个实例,不管在哪里调用都会得到同一个实例。

  • 共享:所有使用该单例Bean的组件将共享同一个实例,这样可以方便地进行数据的共享和状态的管理。

  • 线程安全:单例Bean默认对多线程是线程安全的,即使多个线程同时访问该Bean的方法,也不会出现线程冲突的问题。

  • 生命周期管理:单例Bean的生命周期由Spring容器管理,容器会在必要时创建和销毁单例Bean。

  • 要将一个Bean定义为单例作用域,可以使用@Scope("singleton")注解,或者直接省略@Scope注解,默认为单例作用域。

  • 需要特别注意的是,单例Bean应该是无状态的,不应该包含可变状态。如果单例Bean包含可变状态,可能会引发线程安全问题。如果需要在单例Bean中保存状态,应该使用线程安全的方式进行操作,或者考虑将Bean的作用域定义为原型(Prototype)。

  • 单例Bean是Spring中最常用的作用域之一,它提供了全局唯一性、共享和线程安全的特性。合理使用单例Bean可以提高应用程序的性能和资源利用率,并方便管理对象的状态和数据共享。

2.1.2 单例设计模式 与单例bean:

单例 bean 是有spring 容器进行生成 和销毁的,其生命周期由spring 控制,一个类可以生成多个相同类型但是不同名称的单例bean
设计单例模式,是一种开发的规范模式,一个类只会有一个实例存在,并提供全局访问点,从概念上讲可以说两个根本就是不同的东西;

2.2 原型bean:

容器只负责创建和初始化,spring 本身不进行存储,原型(Prototype)作用域的Bean是Spring中的一种作用域,与单例作用域相对。当Bean被定义为原型作用域时,在每次通过容器获取该Bean时,都会创建一个新的实例。

2.2.1 原型Bean的特点:

  • 多实例:每次通过容器获取原型Bean时,都会创建一个新的实例。相同的原型Bean可以有多个独立的实例。

  • 灵活性:原型Bean适用于需要频繁创建新实例且不需要保持状态或数据共享的情况。每次获取原型Bean时,都可以获得一个全新的实例。

  • 生命周期管理:容器在创建原型Bean实例后,不负责对其进行生命周期的管理。这意味着如果原型Bean依赖其他Bean,容器不会自动处理依赖的初始化和销毁。

  • 对象创建时机:原型Bean的创建时机是在每次通过容器获取Bean时。而单例Bean在容器启动时就会创建。

  • 要将一个Bean定义为原型作用域,可以@Scope(“prototype”)注解

  • 需要注意的是,原型Bean在被注入到其他单例Bean中时,每次注入都会创建一个新的实例。因此,在使用原型Bean时要注意管理生命周期和资源释放,特别是在具有复杂依赖关系和状态的情况下。

  • 原型Bean是Spring框架中的一种作用域,每次通过容器获取原型Bean时都会创建一个新的实例。原型作用域适用于需要频繁创建新实例的情况,且不需要保持状态或数据共享。通过合理使用原型Bean,可以满足动态创建对象实例的需求。

  • 原型Bean是由Spring容器创建和管理的,但是它的生命周期并不受Spring容器的控制。因此,原型Bean不由Spring销毁,而是由使用该Bean的客户端负责销毁。

2.2.2 原型Bean的销毁:

当从Spring容器获取一个原型Bean时,Spring容器会创建一个新的实例,并将其返回给客户端。客户端可以使用该实例进行操作,但是在实例不再使用时,客户端需要负责销毁它。

原型Bean的销毁工作可以通过以下几种方式来实现:
(1). 显式销毁:客户端可以在不再使用原型Bean时,主动调用销毁方法进行销毁。这可以是一个自定义的销毁方法,例如实现了DisposableBean接口的destroy()方法,或者使用@PreDestroy注解标注的方法。在该方法中,可以执行必要的清理操作,例如释放资源。

@Component
@Scope("prototype")
public class PrototypeBean implements DisposableBean {
    
    @Override
    public void destroy() throws Exception {
        // 执行销毁操作
    }
}

(2). 手动销毁:客户端可以在原型Bean不再使用时,手动调用销毁方法进行销毁。这可以使用通过Spring容器提供的ApplicationContext对象获取原型Bean,并调用销毁方法。

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
PrototypeBean bean = context.getBean(PrototypeBean.class);
// 使用原型Bean
// 销毁Bean
((Disposable) bean).destroy();

需要注意的是,Spring容器会管理单例Bean的生命周期,括创建和销毁。但是对于原型Bean,Spring容器只负责创建,不负责销毁。因此,在使用原型Bean时,应该确保在不使用时及时销毁,以避免资源泄漏和内存泄漏问题。

另外,Spring容器在创建原型Bean时不会对其进行任何的依赖注入和初始化操作。每次获取原型Bean时,都会创建一个新的实例,并返回给客户端。因此,原型Bean的状态不会被其他Bean所影响,每次获取到的实例都是全新的。

2.3 Request bean 和 Session bean:

2.3.1 Request bean :

Request 作用域的bean 需要和web 进行关联,在Web应用中,每个请求都会有一个独立的Bean实例。即每个请求都会有自己的Bean实例。请求(Request)作域是Spring框架中一种作用域,表示在每次HTTP请求中创建一个新的Bean实例。请求作用域的Bean将在每个请求过程中保持独立,这使得我们可以在每个请求中使用不同的Bean实例
要将Bean定义为请求作用域,可以使用@Scope(“request”)注解或RequestScope注解。

示例代码如下:

@Component
@Scope("request")
public class MyRequestScopedBean {
    // Bean的定义
}

在上述示例中,MyRequestScopedBean被定义为请求作用域的Bean。每次HTTP请求都会创建一个新的MyRequestScopedBean实例,并且该实例仅在当前请求中有效。当请求结束后,该实例将被销毁。

2.3.1.1 Request bean 特点:
  • 请求作用域的Bean只能在Web应用程序中使用,并且仅适用于使用了Spring Web MVC或Spring Boot的Web应用。
  • 在使用请求作用域的Bean之前,需要在Spring上下文中启用对请求作用域的支持。可以通过配置RequestContextListener或使用注解@EnableWebMvc来启用。
  • 在同一个请求中多次使用请求作用域的Bean将会得到同一个实例。但不同请求之间的Bean实例是独立的。
  • 请求作用域的Bean对于并发请求是线程安全的。每个线程在处理请求时都会获得自己的Bean实例。
  • 请求作用域是Spring框架中的一种作用域,用于在每个HTTP请求中创建独立的Bean实例。它适用于需要在每个请求中维持独立状态的Bean。通过使用@Scope(“request”)注解或RequestScope注解,我们可以将Bean定义为请求作用域,实现请求级别的状态管理。

2.3.2 session bean :

在Web应用中,每个用户会话都会有一个独立的Bean实例。即每个用户在会话期间都有自己的Bean实例

Session作用域是Java Web应用中的一种作用域范围,用于在用户会话期间保持数据的独立状态。可以将数据存储在Session对象中,并在整个会话过程中进行共享和访问。

在Java EE中,Session作用域可以通过HttpSession对象来实现。HttpSession对象是由Servlet容器在每个用户会话期间创建和管理的,它表示了与特定用户相关联的会话对象。通过HttpSession对象,可以在会话作用域中存储和获取数据。

2.3.1.2 session bean 特点:
  • 会话范围:Session作用域的生命周期与用户会话的启动和结束相对应。当用户首次访问Web应用时,Servlet容器会为该用户创建一个唯一的会话,并为其分配一个Session对象。该Session对象在整个会话期间都是有效的,直到用户主动结束会话或会话过期。

  • 跨页面共享数据:Session作用域中的数据可以在用户会话期间的不同页面中进行共享和访问。当一个页面将数据存储在Session对象中后,其他页面可以通过相同的Session对象来获取存储的数据。这对于需要在多个页面之间传递数据或保持用户的状态特别有用。

  • 容器管理:Session对象由Servlet容器自动创建和管理,无需显式创建或销毁。当用户会话结束或会话过期时,Servlet容器会自动销毁相关的Session对象及其对应的数据。

  • 在使用Session作用域时,一般会借助HttpServletRequest对象来获取和操作Session对象。以下是一些示例代码,演示了如何在Session作用域中存储和获取数据。

在使用Session作用域时,一般会借助HttpServletRequest对象来获取和操作Session对象。以下是一些示例代码,演示了如何在Session作用域中存储和获取数据。

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Component
public class SessionScopedBean {
    
    @Autowired
    private HttpServletRequest request;

    public void storeData() {
        HttpSession session = request.getSession();
        session.setAttribute("key", "value");
    }

    public void retrieveData() {
        HttpSession session = request.getSession();
        String value = (String) session.getAttribute("key");
        // 使用值进行其他操作
    }
}

在上述示例中,SessionScopedBean是一个会话作用域的Bean,通过注入HttpServletRequest对象,可以获取当前会话的HttpSession对象。然后可以使用HttpSession对象将数据存储在会话作用域中,并在整个会话期间进行获取和使用。

需要注意的是,Session作用域中的数据是与用户会话绑定的,并且在跨多个用户会话之间是独立的。每个用户会话都有自己独立的Session对象和相应的数据。

综上所述,Session作用域是Java Web应用中一种用于在会话期间保持数据独立状态的作用域范围。通过HttpSession对象,可以在会话作用域中存储和获取数据。

2.3.3 Request、Session bean的 实现:

浏览器在发起第一次请求到达服务器,服务发现浏览器请求头没有携带requestedSessionId / 携带了但是 没有找到session(如重新启动tomcat) 则通过.createSession(sessionId) 方法创建一个新的session并返回;

AbstractRequestAttributesScope 类中get 方法:

public Object get(String name, ObjectFactory<?> objectFactory) {
	// 从一个请求上下文容器中拿到请求的Attributes
    RequestAttributes attributes = RequestContextHolder.currentRequestAttributes();
	// 从属性中获取scope对象, 需要传入name 和 scope类型
	// 每次都是新的request 请求,所有从属性中获取对应的bean 是空的 ,则通过objectFactory.getObject()
	// 完成bean 的创建,创建完成放入到request 的属性中;
	// 如果是session 域在进入该方法之前 回先去获取当前request 请求对应的那个session 通过
	// requestedSessionId  获取session 如果获取到则返回,获取不到则认为是新窗口则创建session;
	// 从reques 或者session 获取 对应的bean 如果获取不到则进行创建,并放入到属性中,
	// 如果获取到了则直接返回
    Object scopedObject = attributes.getAttribute(name, this.getScope());
    if (scopedObject == null) {
        scopedObject = objectFactory.getObject();
        attributes.setAttribute(name, scopedObject, this.getScope());
        Object retrievedObject = attributes.getAttribute(name, this.getScope());
        if (retrievedObject != null) {
            scopedObject = retrievedObject;
        }
    }

    return scopedObject;
}

RequestContextHolder ServletRequestAttributes#getAttribute:

public Object getAttribute(String name, int scope) {
	// 如果是request
	if (scope == SCOPE_REQUEST) {
		if (!isRequestActive()) {
			throw new IllegalStateException(
					"Cannot ask for request attribute - request is not active anymore!");
		}
		// 从request中根据name获取 bean
		return this.request.getAttribute(name);
	}
	else {
		// 如果是session
		HttpSession session = getSession(false);
		if (session != null) {
			try {
				// 从session中根据name获取 bean
				Object value = session.getAttribute(name);
				if (value != null) {
					this.sessionAttributesToUpdate.put(name, value);
				}
				return value;
			}
			catch (IllegalStateException ex) {
				// Session invalidated - shouldn't usually happen.
			}
		}
		return null;
	}
}

2.3.4 bean 的request,session作用域 同 web 浏览器 session会话和http 请求的关系:

在Web开发中,Spring框架提供了一些特殊的作用域来管理bean对象的生命周期,这些作用域与浏览器的会话(session)和HTTP请求息息相关。

  • Request作用域:
    Request作用域是指将bean的生命周期限定在单个HTTP请求之内。当一个HTTP请求到达服务器,并且在处理请求的过程中需要使用到某个bean对象,Spring会创建该bean对象并将其置于Request作用域中。这意味着在同一个请求中多次获取该bean对象,会得到同一个实例。

  • Session作用域:
    Session作用域是指将bean的生命周期限定在用户的会话期间。当用户通过浏览器与Web应用程序建立会话后,Spring会创建各个bean对象,并将其置于Session作用域中。这样,在同一个会话中多次获取该bean对象,会得到同一个实例。不同的用户会话之间拥有不同的实例。

现在来看一下Web浏览器的会话(session)和HTTP请求之间的关系。当用户使用浏览器访问Web应用程序时,浏览器会与服务器建立一个会话。会话可以跨越多个HTTP请求,直到会话超时或用户关闭浏览器。

对于每个HTTP请求,浏览器会将请求发送到服务器,并附带上之前建立的会话ID(可能通过Cookie或URL重写机制)。服务器接收到请求后,可以通过会话ID找到与该会话相关的信息,包括Session作用域中的bean对象。

在Spring中,使用Request作用域或Session作用域可以方便地管理bean对象的生命周期,并实现在请求或会话期间共享数据。可以通过在bean的声明中添加相应的作用域注解来指定相应的作用域,例如@RequestScope@SessionScope

综上所述:
(1) 他们是两个概念,一个是spring 中bean 的作用域,一个是浏览器与服务器的会话请求,但是他们直接又有关联,当声明为request 的作用域,每次进行http 请求都户创建一个新的bean 并被放入到该request的属性中,所以在同一个http 请求中多次获取到的这个bean 是同一个对象,但是在不同的http 请求中获取到的bean 是不同的
对于session作用域的bean,在一个http请求中去获取改bean ,首先判断当前request 请求是否携带了sessionId 如果没有携带或者携带了但是没有找到session,此时session 为空,会重新创建一个新的session 并放入到response 中,后续多次进行http 请求 ,在使用改bean 的时候会先从session 的属性中获取,获取不到则进bean 的创建并吧创建之后的bean 放入的session 的属性中,所以在一个浏览器窗口的多次http 请求对应session作用域的bean 获取的是同一个对象,不同窗口因为session会话是两个,则取到的bean 是不同的


总结:

不同bean的作用范围 图解:
在这里插入图片描述

Spring 通过@Scope 注解为bean 定义需要的声明周期,Singleton(单例):默认的作用范围,每个Spring容器中只会存在一个Bean实例。Prototype(原型):每次获取Bean时都会创建一个新的实例。Request(请求):每个HTTP请求都会创建一个新的实例,该实例仅在当前请求内部有效。Session(会话):每个HTTP会话都会创建一个新的实例,该实例在整个会话期间有效。

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

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

相关文章

Unity中使用VS常用技巧

文章目录 1、Unity预览代码中文显示乱码解决方案2、VS中格式化代码快捷键 1、Unity预览代码中文显示乱码解决方案 ①打开VS&#xff0c;在 扩展 中找到 管理扩展 打开 ②在 管理扩展面板 联机 中搜索 Force UTF-8(No BOM) &#xff0c;点击下载即可 ③重启VS 如果以前编写的脚…

gpt-4-turbo、gpt-4v、dall-e-3 api实测!

上周GPT大更新&#xff0c;不仅开放了GPT-4-Turbo、GPT-4-Vision等模型api&#xff0c;还发布了GPTs&#xff0c;使得用户能够根据需要定义自己的GPT应用&#xff0c;OpenAI在这波AI革命上又一次震撼世人。 笔者也在上周拿到了几个新模型的api资格&#xff0c;一直盼着可以测试…

C语言题目逻辑实战总结

eg1: 已知有1,2,3,4个数字&#xff0c;能组成多少个互不相同且无重复数字的三位数&#xff1f;都是多少 已知&#xff1a;1&#xff1a;三位数 &#xff0c;2:1-4,3&#xff1a;各不相同&#xff0c; 输出&#xff1a;1&#xff1a;有多少个这样的三位数&#xff0c;2&#xff…

免费内网穿透?让外网能访问到自己的接口springboot,暴露自己的接口给外网

内网穿透&#xff1f;让外网能访问到自己的接口java&#xff0c;暴露自己的接口给外网 内网穿透听着就很吊&#xff0c;不就是把没有备案的服务器暴露出去&#xff0c;给别人访问嘛。 自己免费使用测试的话。 下载一个Cpolar 然后注册一下&#xff0c;进来后&#xff0c;直接…

都dubbo3了,别再用xml了配置dubbo服务了

这里是weihubeats,觉得文章不错可以关注公众号小奏技术&#xff0c;文章首发。拒绝营销号&#xff0c;拒绝标题党 背景 最近项目再准备升级dubbo3,所以打算简单实现一个dubbo3的demo。 来学习一下dubbo dubbo3 dubbo3主要是为了融入云原生打造的 Dubbo 3 提供的核心特性列…

Milvus性能优化提速之道:揭秘优化技巧,避开十大误区,确保数据一致性无忧,轻松实现高性能

Milvus性能优化提速之道&#xff1a;揭秘优化技巧&#xff0c;避开十大误区&#xff0c;确保数据一致性无忧&#xff0c;轻松实现高性能 Milvus 是全球最快的向量数据库&#xff0c;在最新发布的 Milvus 2.2 benchmark中&#xff0c;Milvus 相比之前的版本&#xff0c;取得了 …

竞赛 题目:基于LSTM的预测算法 - 股票预测 天气预测 房价预测

文章目录 0 简介1 基于 Keras 用 LSTM 网络做时间序列预测2 长短记忆网络3 LSTM 网络结构和原理3.1 LSTM核心思想3.2 遗忘门3.3 输入门3.4 输出门 4 基于LSTM的天气预测4.1 数据集4.2 预测示例 5 基于LSTM的股票价格预测5.1 数据集5.2 实现代码 6 lstm 预测航空旅客数目数据集预…

PieCloudDB Database 自研内存管理器 ASanAlloc:为产品质量保驾护航

内存管理是计算机科学中至关重要的一部分&#xff0c;它涉及到操作系统、硬件和软件应用之间的动态交互。有效的内存管理可以确保系统的稳定性和安全性&#xff0c;提高系统运行效率&#xff0c;帮助我们最大限度地利用有效的内存资源&#xff0c;合理分配和回收内存&#xff0…

【Linux】进程替换|exec系列函数

文章目录 一、看一看单进程版的进程替换二、进程替换的原理三、多进程版——验证各种程序替换接口exec系列函数execlexeclpexecvexecvp tipsexecleexecve 四、总结 一、看一看单进程版的进程替换 #include<stdio.h> #include<unistd.h> #include<stdlib.h>i…

Elasticsearch的配置学习笔记

文/朱季谦 Elasticsearch是一个基于Lucene的搜索服务器。它提供一个分布式多用户能力的全文搜索引擎&#xff0c;基于RESTful web接口&#xff0c;Elasticsearch是用Java语言开发的。 关于Elasticsearch系列笔记&#xff0c;主要从Elasticsearch的配置、核心组件、架构设计、使…

跟着 Tubi 同事吃遍全世界

在过去的一年里&#xff0c;Tubi 北京办公室的 Pantry 非常忙&#xff0c;忙于接收 Tubi 同事从全球各地带回的美食。而我们也有幸跟随慷慨的同事们尝遍了大江南北的味道。 细数这 30 多次美食分享&#xff0c;我们发现&#xff0c;大家分享的不仅是食物&#xff0c;还是…… …

第二证券:尾盘选股方法?

股票出资是一种危险较高的出资办法&#xff0c;因而怎样挑选合适的股票成为出资者面临的重要问题之一。尾盘选股办法就是一种常用的战略。尾盘是指股票商场生意时间的究竟一段时间&#xff0c;通常是下午3点到收盘之间。在这段时间里&#xff0c;生意量较大&#xff0c;不坚决性…

在Python中使用sqlite3进行数据持久化操作

目录 引言 一、安装sqlite3模块 二、创建数据库连接 三、创建游标对象 四、执行SQL命令 五、提交更改 六、关闭连接 七、使用参数化查询 八、使用ORM进行数据操作 九、备份和恢复数据库 十、处理大量数据 十一、优化查询性能 十二、处理并发访问 十三、处理数据持…

仿真算法收敛与初值的关系

问题&#xff1a; 当电路中存在大电容时&#xff0c;由于初值设置不合理可能导致的仿真算法不收敛的问题。 解决方法&#xff1a;设置初始节点值。 疑问&#xff1a;Node set和Initial Condition的区别。 [求助] node set 和initial condition有很么区别呢&#xff1f; 注&…

7-4链表去重

题目 给定一个带整数键值的链表 L&#xff0c;你需要把其中绝对值重复的键值结点删掉。即对每个键值 K&#xff0c;只有第一个绝对值等于 K 的结点被保留。同时&#xff0c;所有被删除的结点须被保存在另一个链表上。例如给定 L 为 21→-15→-15→-7→15&#xff0c;你需要输出…

前端算法面试之堆排序-每日一练

如果对前端八股文感兴趣&#xff0c;可以留意公重号&#xff1a;码农补给站&#xff0c;总有你要的干货。 今天分享一个非常热门的算法--堆排序。堆的运用非常的广泛&#xff0c;例如&#xff0c;Python中的heapq模块提供了堆排序算法&#xff0c;可以用于实现优先队列&#xf…

C/C++ stm32基础知识超详细讲解(系统性学习day14)

目录 前言 一、ARM和STM32是什么&#xff1f; 二、STM32的开发方式 三、GPIO----寄存器开发方式 1.八种输入输出模式分析 2.寄存器 四、stm32芯片图片 五、怎么学好stm32 总结 前言 stm32的广泛含义及背景&#xff1a; STM32是一款由意法半导体&#xff08;ST&…

mmdetection安装与训练

一、什么是mmdetection 商汤科技&#xff08;2018 COCO 目标检测挑战赛冠军&#xff09;和香港中文大学最近开源了一个基于Pytorch实现的深度学习目标检测工具箱mmdetection&#xff0c;支持Faster-RCNN&#xff0c;Mask-RCNN&#xff0c;Fast-RCNN等主流的目标检测框架&#…

前端js面试题 (四)

文章目录 ES6新增的proxy手写&#xff0c;proxy访问某对象输出别的数字深度拷贝&#xff0c;为啥无法使用JSON.parse(JSON.stringify(obj))异步编程有哪些&#xff0c;async await来由&#xff0c;本质原理是什么事件队列输出题第一题第二题第三题 粘性布局的原理&#xff0c;以…