Spring的创建及使用

news2025/1/18 20:07:32

文章目录

    • 什么是Spring
    • Spring项目的创建
    • 存储Bean对象
    • 读取Bean对象
      • getBean()方法
    • 更简单的读取和存储对象的方式
      • 路径配置
      • 使用类注解存储Bean对象
      • 关于五大类注解
      • 使用方法注解@Bean存储对象
        • @Bean重命名
      • Bean对象的读取
    • 使用@Resource注入对象
    • @Resource VS @Autowired
    • 同一类型多个bean对象的读取问题

什么是Spring

Spring一般是指Spring Framework,即Spring框架。它是一个强大的java开发框架,可以支持多种应用场景;通过使用Spring框架,可以极大程度地简化开发流程,提高开发效率。

常见的对Spring的概括是:Spring是包含了众多工具方法的Ioc容器;

包含了“众多工具方法”不难理解,那么什么是所谓的Ioc容器呢?

Ioc实际上就是Inversion Control,即控制反转。在传统的开发模式中,对于在A类中使用B类这样一个场景,就需要在A类中创建B类的对象。这样,关于这个被创建的B类的对象,A就可以控制这个对象的所有行为,包括创建、使用、销毁。而如果是使用控制反转的开发模式,就是将这个对B类对象的控制权交出去,交给Spring去控制。也就是说,控制反转实际就是控制权的反转。

很明显,传统的开发模式存在一定的问题,即当代码或程序之间的调用关系过于复杂,就会存在修改了一个程序的代码之后,可能就需要修改对应的调用链上的一系列代码。但控制反转的开发模式则是可以很好地解决这个问题,实现代码之间的解耦。

可以使用代码来进行理解:

既然Ioc就是控制权反转的意思,那么容器又是指什么呢?在日常生活中的容器,就是用来容纳某种东西的一个装置,容器最大的作用也就是存和取。因此对于这样一个Ioc容器而言,两个最基础的功能应该也就是:存入对象到容器、从容器取出对象;

而对于Spring而言,将对象存储到Spring中,再根据需要从Spring中获取对象的过程其实也就是它最核心的功能或步骤;

Spring项目的创建

  1. 创建一个Maven项目;

在这里插入图片描述

在这里插入图片描述

  1. 添加需要的Spring框架依赖到pom.xml,包括Spring上下文和spring对象;
<dependencies>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.2.3.RELEASE</version>
        </dependency>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-beans</artifactId>
            <version>5.2.3.RELEASE</version>
        </dependency>
</dependencies>

添加完成以后一定要进行刷新,确保依赖已经下载成功;

在这里插入图片描述
如上,表示下载成功;

  1. 在java文件夹下创建一个启动类(包含main方法即可);

在这里插入图片描述

至此,一个spring文件就创建成功了!

存储Bean对象

首先就是spring的第一个关键功能,存储对象到spring中,具体操作如下:

  1. 创建一个Bean对象;

Bean对象实际也只是java中的一个普通对象,创建在java文件夹下即可:

在这里插入图片描述

  1. 将bean对象注册到spring中;

在resources文件夹下创建一个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">

</beans>

(上面的格式是固定内容,不需要记忆)

将之前创建的bean对象注册到spring中:

在这里插入图片描述
id是对象的标识,在之后取对象时会用到;class是指明了bean对象的位置(包名+类名);

读取Bean对象

  1. 得到Spring上下文;
  2. 从spring上下文获取bean对象;
  3. 根据需要使用bean对象;
package com.yun;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Start {
    public static void main(String[] args) {

        //1.得到spring上下文对象
        ApplicationContext context=
                new ClassPathXmlApplicationContext("spring-config.xml");

        //2.从spring上下文中取出bean对象
        User user=(User) context.getBean("user");

        //3.使用bean对象
        user.fun();
    }
}

在得到spring上下文对象时,需要指明对应的spring配置文件;

从spring上下文获取bean对象时,括号中的内容需要与配置文件中id一一对应;

getBean()方法

关于读取Bean对象的getBean()方法,其实还有多种重载方法可以来获取Bean对象:

  • 直接根据对象的id(名称)来获取bean对象;
  • 根据类型来获取bean对象;
  • 使用名称+类型的方式来获取bean对象;

使用代码进行演示:

在这里插入图片描述

三种方式各有优劣,一般来说,如果使用名称获取的方式,必须保证在配置文件中的id是唯一的;如果使用类型的方式获取,当出现同一个类型被多次注册到配置文件中时(即一下面所示的情况),程序就会出错;

在这里插入图片描述

更简单的读取和存储对象的方式

关于前面整个Bean对象的存储和读取的过程,实际还是较为繁琐的;为了简化其流程,我们使用注解来完成一种更加简单的存储和读取过程;

路径配置

在进行更简单的方式之前,我们首先需要完成Bean对象扫描路径的配置工作:

即在spring-config.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"
       xmlns:content="http://www.springframework.org/schema/context"
       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">
    <content:component-scan base-package="com.yun">
</content:component-scan>
</beans>

在这里插入图片描述
需要根据自己的代码的具体路径来修改上面图片红色方框中的路径,这实际就是Spring的扫描路径,只有在该路径下的Bean对象才会被存储到Spring中;

使用类注解存储Bean对象

在Spring中,提供了五大类注解可以实现将对象存储到Spring,分别是@Controller @Service @Repository @Component @Configuration ;

  • @Controller
    该注解主要是负责控制器存储,使用代码演示:
package com.yun.controller;

import org.springframework.stereotype.Controller;

@Controller

public class UserController {
    public void sayHello(){
        System.out.println("Hello~Controller");
    }
}

此时就可以读取到这里存储的UserController对象:

public class App {
    public static void main(String[] args) {

        //1.获取到Spring上下文对象
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");

        //2.从Spring上下文取出bean 对象
        // 使用注解默认的命名规则是小驼峰
        UserController userController=context.getBean("userController",UserController.class);

        //3.使用bean对象
        userController.sayHello();
    }
}

在这里插入图片描述

@Controlller注解使用中文翻译就是控制器的意思,主要是负责验证前端传递过来的参数,起到一个“安全检查”的作用;

  • @Service

该注解是负责服务存储;

package com.yun.service;

import org.springframework.stereotype.Service;

@Service
public class UserService {
    public void helloSer(){
        System.out.println("Hello~Service");
    }
}

在这里插入图片描述

@Service注解主要是负责了服务调用的编排和汇总;

  • @Repository

该注解负责仓库存储;

package com.yun.repository;

import org.springframework.stereotype.Repository;

@Repository
public class UserRepository {
    public void doRepository(){
        System.out.println("Hello~Repository");
    }
}

@Repository该注解的中文意思为仓库,使用该注解可以直接操作数据库;

  • @Component

该注解负责组件存储;

package com.yun.component;

import org.springframework.stereotype.Component;

@Component
public class UserComponent {

    public void doComponent(){
        System.out.println("Hello~Component");
    }
}

@Component 该注解表示了组件的意思,主要是负责一些通用化的工具类;

  • @Configuration

该注解负责配置存储;

package com.yun.configuration;

import org.springframework.context.annotation.Configuration;

@Configuration
public class UserConfiguration {

    public void doConfiguration(){
        System.out.println("Hello~Configuration");
    }
}


@Configuration该注解表示配置,负责了项目所需要的相关的所有配置;

使用五大类注解存储Bean对象的方法如上,而关于使用相关注解的Bean对象的读取,也是首先获取Spring的上下文,再通过上下文得到bean对象,最后使用bean对象;

关于五大类注解

  • 首先,为什么需要如此多的类注解呢?

如果是单从上面使用类注解来存储Bean对象的操作来看,似乎每一个类注解起到的作用都是相同的。但在实际的业务开发中,使用不同的类注解可以清晰的表明当前类的用途,这也就是前面说到不同的类注解负责的业务或模块是不同的;

  • 五大类注解之间的关系?

如果我们尝试去溯源五大注解的源码,就会发现:

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

可以看到, @Controller / @Service / @Repository / @Configuration四个注解的实现实际上都借助了@Component注解来实现,所有它们之间的关系也就显而易见,即可以简单地理解为前面四种注解是@Component的子类;

  • 使用五大类注解时Bean对象的命名

在上面的代码中,我们关于类名都是使用了大驼峰的方式进行标准的命名,在读取bean时则是默认使用了首字母小写的方式,最后也如愿读取成功了,关于这样使用的原因,我们同样可以溯源到相关的源码:

在这里插入图片描述

在这里插入图片描述

bean的命名方式在默认情况下使用类名首字母小写的方式进行;
特殊情况下,当类名的前两个字母均为大写的情况下,bean的命名直接使用原类名即可;

使用方法注解@Bean存储对象

方法注解@Bean,顾名思义就是使用在方法上的;方法注解正常使用的前提是:搭配类注解一起使用;

使用代码进行演示:

package com.yun.controller;

import com.yun.model.User;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;

@Component
public class UserBeans {


    @Bean
    public User user(){
        User user=new User();
        user.setId(1);
        user.setName("张三");
        user.setAge(18);

        return user;
    }
}

在这里插入图片描述

使用方法注解存储的Bean对象,在后续使用Bean对象时,是直接使用方法名来命名Bean对象;

另外,关于方法注解,只能使用在无参的方法上,因为Spring在初始化存储时,无法提供相应的参数;

@Bean重命名

当然,除了直接使用方法名,在Spring中关于方法注解的使用,还可以通过为Bean对象设置name属性来达到重命名的目的;

重命名的设置方法有三种方式:

  • 显示使用name属性重命名

在这里插入图片描述

  • 直接使用双引号重命名

在这里插入图片描述

  • 显式使用name属性进行多个重命名;

在这里插入图片描述
当然,在对Bean进行了重命名以后,就不能再使用原来的方法名获取Bean对象了;

Bean对象的读取

Bean对象的读取即,将对象读到以后放到某个类中,也称为对象装配或对象注入;

对象注入的方式有下面3种:

  • 属性注入

借助@Autowired注解实现,使用代码演示:

service部分的原始代码:

package com.yun.service;

import org.springframework.stereotype.Service;
@Service
public class UserService {
    
    public boolean helloSer(){
        System.out.println("Hello~Service");
        return true;
    }
}

将UserService注入到UserController类中:

package com.yun.controller;

import com.yun.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.stereotype.Repository;

@Controller

public class UserController {

    //将service中的UserService注入到了该类中
    @Autowired
    private UserService userService;
    
    public void sayHello(){
        System.out.println(userService.helloSer());
    }
}


使用bean对象进行验证注入是否成功;

import com.yun.controller.UserController;
import com.yun.model.User;
import com.yun.repository.UserRepository;
import com.yun.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class App {
    public static void main(String[] args) {

        //1.获取到Spring上下文对象
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");

        //2.从Spring上下文取出bean 对象
       
        UserController userController=context.getBean("userController",UserController.class);
        //使用bean对象
        userController.sayHello();

        
    }
}



运行结果:

在这里插入图片描述

  • Setter注入
    需要在属性的set方法上加上@Autowired注解来实现;

service中的代码基本保持不变;

package com.yun.service;

import org.springframework.stereotype.Service;


@Service
public class UserService {

    public boolean helloSer(){
        System.out.println("setter~Service");
        return true;
    }
}

在controller中改变注入对象的方式:

package com.yun.controller;

import com.yun.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.stereotype.Repository;

@Controller

public class UserController {

    //将service中的UserService注入到了该类中
    private UserService userService;

    @Autowired
    public void setUserService(UserService userService){
        this.userService=userService;
    }

    public void sayHello(){
        System.out.println(userService.helloSer());
    }
}


最后进行验证的代码与前面相同,下面是具体的运行结果:

在这里插入图片描述

  • 构造方法注入

构造方法注入是在当前类的构造方法中实现注入,同样使用到了@Autowired注解;

package com.yun.controller;

import com.yun.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.stereotype.Repository;

@Controller

public class UserController {

    //将service中的UserService注入到了该类中
  
    private UserService userService;

    @Autowired
    public UserController(UserService userService){
        this.userService=userService;
    }

    public void sayHello(){
        System.out.println(userService.helloSer());
    }
}


其余部分的代码与前面基本相同;

三种对象注入的方式各有特点,下面是对其各自优缺点的分析:

  1. 属性注入

优点:

  • 代码简洁,使用方便;

缺点:

  • 只适用于IOC容器,代码的可移植性不强;
  • 无法注入不可变的对象(final修饰的对象);
  • 容易违反单一设计原则;

使用属性注入的方式,代码量少,使用方便又简单;但也是由于这一点,违反单一设计原则、代码滥用的概率也相应增加;同时由于spring 是基于java环境实现,也必须遵守final关键字的使用规范,即不可以使用属性注入一个final修饰的对象。

  1. setter注入;

优点:

  • 符合单一设计原则;

缺点:

  • 无法注入一个不可变的对象;
  • 注入的对象存在被修改的概率;

在spring 4.2之前,这是官方推荐使用的注入方式,它遵循了单一设计原则;但由于set方法在代码中可能被多次调用,也相应地被修改的概率要更大。

  1. 构造方法注入;

优点:

  • 可以注入final修饰的对象;
  • 注入的对象没有被修改的概率;
  • 所依赖的对象在使用之前就会被完全初始化;
  • 代码的通用性更强;

缺点:

  • 当有多个注入时,代码略显臃肿;

在spring 4.2 之后,构造方法注入成为了官方推荐的注入方式。由于构造方法是会在类创建之初执行一次,因此使用这种方式注入的对象不会被修改,同时对象在使用之前就进行了初始化;另外因为构造方法是由JDK支持实现,因此使用这种方式注入的代码的通用性要更强。

进行对象的注入,除了使用前面提到的@Autowired注解,实际还有一个注解同样可以实现对象的注入;

使用@Resource注入对象

同样使用代码来演示@Resource的对象注入方式;

  • 属性注入;

创建一个UserService2类;

package com.yun.service;

import org.springframework.stereotype.Service;
@Service
public class UserService2 {

    public void doService(){
        System.out.println("Do Service!");
    }
}

将上面创建的对象使用@Resource注入到UserController2中;

package com.yun.controller;

import com.yun.service.UserService2;
import org.springframework.stereotype.Controller;

import javax.annotation.Resource;
import java.nio.file.attribute.UserPrincipalLookupService;

@Controller
public class UserController2 {

    @Resource
    public UserService2 userService2;
    public void doController(){
        userService2.doService();
    }

}

在这里插入图片描述

  • setter注入;
package com.yun.controller;

import com.yun.service.UserService2;
import org.springframework.stereotype.Controller;

import javax.annotation.Resource;
import java.nio.file.attribute.UserPrincipalLookupService;
@Controller
public class UserController2 {

    private UserService2 userService2;

    @Resource
    public void setUserService2(UserService2 userService2) {
        this.userService2 = userService2;
    }


    public void doController(){
        userService2.doService();
    }

}

直接看运行结果:
在这里插入图片描述

  • 构造方法注入

在这里插入图片描述

可以看到@Resource注解不能使用在构造方法注入的实现上;

@Resource VS @Autowired

既然两种注解都可以实现对象的注入,那么它们又具体有什么区别呢?

  • 方法数量上;

首先从java为两种注解提供的方法就可以发现,@Resource注解是包含了有众多的方法,而@Autowired则只有一个方法;
在这里插入图片描述
在这里插入图片描述
可以看到@Resource相对于@Autowired而言,支持更多的参数设置

  • 匹配对象的顺序;

@Autowired在从spring中查找相应的bean对象时,首先会根据对象的类的类型进行匹配,在未匹配成功的情况下,继续使用bean的名称来匹配;
@Resource则是首先进行bean名称的匹配查找,后进行类型的查找;

  • 出身来源不同;
    @Autowired是由spring提供的,而@Resource则是属于JDK的注解;

同一类型多个bean对象的读取问题

在一些特定的场景中,可以需要将同一类型的多个bean对象存储到spring中,在这种情况下,对于bean对象的读取必然会出现下面的错误:

首先创建一个类,这个类中包含了多个User类型的对象,再将User类型的对象使用方法注解存储到spring中;

package com.yun.controller;

import com.yun.model.User;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;

@Component
public class UserBeans {
    @Bean

    public User user1(){
        User user=new User();
        user.setId(1);
        user.setName("张三");
        user.setAge(18);

        return user;
    }

    @Bean

    public User user2(){
        User user=new User();
        user.setId(2);
        user.setName("李四");
        user.setAge(19);

        return user;
    }

    @Bean

    public User user3(){
        User user=new User();
        user.setId(3);
        user.setName("王五");
        user.setAge(20);

        return user;
    }
}

将User类注入注入到该类中:

package com.yun.controller;

import com.yun.model.User;
import org.springframework.stereotype.Controller;

import javax.annotation.Resource;

@Controller
public class UserController3 {
    @Resource
    private User user;

    public void doCon(){
        System.out.println(user.getName()+" "+user.getAge());
    }
}

在读取时发现出现了下面的报错信息:
在这里插入图片描述
在这里插入图片描述
通过报错信息,我们找到产生问题的原因就是:我们此处的bean对象不是唯一的,在同一类型下找到了多个匹配的bean对象,下面是相关的解决办法:

  • 使用@Resource注解注入,通过设置参数指定bean对象;在这里插入图片描述

  • 使用@Autowired注解注入,搭配@Qualifier 注解一起使用;

在这里插入图片描述
使用上面两种方式修改代码,即可得到正确的运行结果;

至此,关于Spring的创建以及将对象如何存储到Spring中,再从Spring中读取的全过程就介绍完毕啦~

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

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

相关文章

QT服务器练习

#include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);//给服务器指针实例化空间server new QTcpServer(this); }Widget::~Widget() {delete ui; }//启动服务器按钮对…

第1篇:了解Matter物模型翻译器

第1篇&#xff1a;了解Matter物模型翻译器 1. Matter物模型简介2. 物模型翻译成编程语言3. 思考题 1. Matter物模型简介 Matter物模型的介绍可以参考: Matter Core Specification的第7章 Matter物模型直观展示&#xff0c;可以看下图1-1&#xff0c; Matter Dimmable Light的…

【Linux】7、D-Bus 消息总线系统

文章目录 d-bus 官网 d-bus详解 D-Bus 是一种消息总线系统&#xff0c;是应用程序相互通信的一种简单方法。除了进程间通信之外&#xff0c;D-Bus 还有助于协调进程生命周期; 它使得编写“单实例”应用程序或守护进程以及在需要应用程序和守护进程的服务时按需启动它们变得简…

【学习笔记】关于RAW图片的概念学习

这里是尼德兰的喵芯片设计相关文章&#xff0c;欢迎您的访问&#xff01; 如果文章对您有所帮助&#xff0c;期待您的点赞收藏&#xff01; 让我们一起为成为芯片前端全栈工程师而努力&#xff01; 前言 能为我介绍一下raw图片吗&#xff1f; 当谈论"Raw图片"时&am…

c++ ,vs2019, cpp20规范之 forward_list 源码分析

通过阅读源码可知&#xff0c;该单向链表不像list双向链表那样有专门的前导节点。即list._Mypair._Myval2._head._next才指向第一个有效数据节点。而 forward_list ._Mypair._Myval2._head 已经指向了有效数据节点。原因就在于复杂巧妙的类型转换。如下图的构造函数里&#xff…

C语言实现定时器,定时触发函数

最近想到使用C语言实现一个简单的定时器。使用操作系统windows.h提供的多线程API就能实现 首先定义一个定时器结构体&#xff0c;包含定时时间和触发的函数指针 typedef struct Stimer{int valid;//定时器有效long timingMS;//定时时间TriggerFunc tf;//触发函数 }Stimer;创建…

python下的control库使用

文章目录 control的官方网站函数示例强迫响应forced_response control的官方网站 函数示例 强迫响应forced_response import numpy as np import os import sys import control as ctrl import matplotlib.pyplot as pltdef lim_x(x, lim0):res 0if x > lim:res 1else:…

qt代码练习

计时器练习 namespace Ui { class third; }class third : public QWidget {Q_OBJECTpublic:explicit third(QWidget *parent nullptr);~third();QLabel *labth1 new QLabel(this);QTextEdit *txtth1 new QTextEdit("闹钟",this);QLineEdit *leth1 new QLineEdit(t…

Stable Diffusion:网页版 体验 / AI 绘图

一、官网地址 Stable Diffusion Online 二、Stable Diffusion AI 能做什么 Stable Diffusion AI绘图是一种基于Stable Diffusion模型的生成式AI技术&#xff0c;能够生成各种类型的图像&#xff0c;包括数字艺术、照片增强和图像修复等。以下是一些可能的应用&#xff1a; …

《吐血整理》进阶系列教程-拿捏Fiddler抓包教程(13)-Fiddler请求和响应断点调试

1.简介 Fiddler有个强大的功能&#xff0c;可以修改发送到服务器的数据包&#xff0c;但是修改前需要拦截&#xff0c;即设置断点。设置断点后&#xff0c;开始拦截接下来所有网页&#xff0c;直到取消断点。这个功能可以在数据包发送之前&#xff0c;修改请求参数&#xff1b…

【wsl-windows子系统】安装、启用、禁用以及同时支持docker-desktop和vmware方案

如果你要用docker桌面版&#xff0c;很可能会用到wsl&#xff0c;如果没配置好&#xff0c;很可能wsl镜像会占用C盘很多空间。 前提用管理员身份执行 wsl-windows子系统安装和启用 pushd "%~dp0" dir /b %SystemRoot%\servicing\Packages\*Hyper-V*.mum >hyper…

什么是DOTS?

(图片为实机测试) DOTS全称&#xff1a;&#xff08;Burst Job SystemEntity Component System&#xff09; 新型高性能、多线程面向数据的技术堆栈 是由&#xff1a;BrustJob System ECS组合而成&#xff0c;是一种面向数据对象的编程体系&#xff0c;在unity中您也可以对…

机器学习-Gradient Descent

机器学习(Gradient Descent) videopptblog 梯度下降(Gradient Descent) optimization problem: 损失函数最小化 假设本模型有两个参数&#x1d703;1和&#x1d703;2&#xff0c;随机取得初始值 求解偏微分&#xff0c;梯度下降对参数进行更新 Visualize: 确定梯度方向&…

学习使用axios,绑定动态数据

目录 axios特性 案例一&#xff1a;通过axios获取笑话 案例二&#xff1a;调用城市天气api接口数据实现天气查询案例 axios特性 支持 Promise API 拦截请求和响应&#xff08;可以在请求前及响应前做某些操作&#xff0c;例如&#xff0c;在请求前想要在这个请求头中加一些…

leetcode 1372. 二叉树中的最长交错路径

给你一棵以 root 为根的二叉树&#xff0c;二叉树中的交错路径定义如下&#xff1a; 选择二叉树中 任意 节点和一个方向&#xff08;左或者右&#xff09;。 如果前进方向为右&#xff0c;那么移动到当前节点的的右子节点&#xff0c;否则移动到它的左子节点。 改变前进方向&a…

IIS创建网站报错 \\?\C:\Windows\inetsrv\config\applicationHost.config

现象&#xff1a; IIS创建不了网站&#xff0c;IIS配置没有发生改变 原因&#xff1a; 服务器C盘无空间&#xff0c;释放空间后问题解决。

Java-day03(程序流程控制)

程序流程控制 1.顺序结构 程序从上至下逐行执行&#xff0c;无判断与跳转 public class Test1{ public static void main(String[] args){int i 1;int j i 1; System.out.println(j);} }2.分支结构 依据条件&#xff0c;选择性执行某段语句 主要有以下两种 2.1 i…

计算机毕设 深度学习卫星遥感图像检测与识别 -opencv python 目标检测

文章目录 0 前言1 课题背景2 实现效果3 Yolov5算法4 数据处理和训练5 最后 0 前言 &#x1f525; 这两年开始毕业设计和毕业答辩的要求和难度不断提升&#xff0c;传统的毕设题目缺少创新和亮点&#xff0c;往往达不到毕业答辩的要求&#xff0c;这两年不断有学弟学妹告诉学长…

VSCode搭建GCC环境

1. 下载 https://www.mingw-w64.org/downloads/ https://github.com/niXman/mingw-builds-binaries/releases 2.安装 x86_64-12.2.0-release-win32-seh-rt_v10-rev1.7z解压到D盘 我的电脑–属性–系统属性–环境变量–系统变量–path D:\MinGW-w64\x86_64-12.2.0-release…

用html+javascript打造公文一键排版系统10:单一附件说明排版

如果公文有附件&#xff0c;一般会在公文正文下作附件说明。 一、附件说明的格式 一般为&#xff1a; 公文如有附件&#xff0c;在正文下空一行左空二字编排"附件"二字&#xff0c;后标全角冒号和附件名称。如有多个附件&#xff0c;使用阿拉伯数字标注附件顺序号&…