【Java EE进阶 --- SpringBoot】Spring IoC

乐观学习,乐观生活,才能不断前进啊!!!
我的主页:optimistic_chen
我的专栏:c语言Java,
Java EE初阶Java数据结构
欢迎大家访问~
创作不易,大佬们点赞鼓励下吧~

文章目录

IoC思想

前面Spring MVC博客提到过,Spring 是一个开源框架,具体来说:Spring是包含了众多⼯具⽅法的IoC容器,Spring提供了框架,填充框架需要用到众多的IoC容器。
总结一下:Spring是容器,IoC是修饰这个容器的工具。比如:List(数据存储容器),Tomcat(Web容器)

IoC( Inversion of Control 控制反转):我们在前⾯讲到,在类上⾯添加@RestController 和@Controller 注解,就是把这个对象交给Spring管理 ,Spring框架启动时就会加载该类。把对象交给Spring管理,就是IoC思想

那么就说:Spring是一个"控制反转"的容器。也就是"控制权反转",当需要某个对象时,我们通常去new一个即可,现在有了IoC就不需要自己创建对象,把这个任务交给容器(IoC),程序只需要依赖注⼊(DependencyInjection, DI)就可以了。

|--------------------------------------------|
| 举个例子:在公司中,员工的招聘,解雇等不是老板负责控制,而是老板下放权力给HR来处理 |

传统程序的思路

假如设计一辆汽车时:汽车根据车身设计、车身根据底盘设计、底盘根据车轮设计。

汽车跑起来,要有汽车,汽车要有车身,车身要有底盘,底盘要有车轮;按照这个逻辑写:

java 复制代码
//车类
public class Car {
    private Framework framework;
    
    public Car(Integer size) {
        this.framework=new Framework(size);
        System.out.println("Car init");
    }

    public void run(){
        System.out.println("car run");
    }
}

//车身类
public class Framework {
    private Bottom bottom;
    public Framework(Integer size){
        this.bottom=new Bottom(size);
        System.out.println("framework init");
    }
}

//底盘类
public class Bottom {
    private Tire tire;
    public Bottom(Integer size){
        this.tire=new Tire(size);
        System.out.println("bottom init");
    }
}

//车轮类
public class Tire {
    int size;
    public Tire(Integer size){
        this.size=size;

        System.out.println("tire init,size:"+size);
    }
}

再添加一个main就能运行了。现在我们只是对车轮的大小进行了设计,如果需要再添加参数,比如颜色等等。就需要对上面的程序进行修改。

从以上代码可以看出,当底层的代码有改动,那么整个调用链上都要修改,代码耦合度非常高。

IoC程序思路

把创建⼦类的⽅式,改为注⼊传递的⽅式

java 复制代码
public class Main {
    public static void main(String[] args) {
        //所有的对象都在这
        Tire tire=new Tire(17,2);
        Bottom bottom=new Bottom(tire);
        Framework framework=new Framework(bottom);
        Car car=new Car(framework);
        car.run();
    }
}

现在如果要修改车轮只要修改两个地方就可以了。

我们发现,传统思路是Car控制并创建了Framework,Framework创建并创建了Bottom,依次往下,而IoC是把依赖对象注⼊将当前对象中,依赖对象的控制权不再由当前类控制了

观察下图,是不是控制权反转呢?

那么这就是控制反转容器,也就是IoC容器:对象交给容器管理

总结:

  1. 资源集中管理: IoC容器会帮我们管理⼀些资源(对象等),我们需要使⽤时,只需要从IoC容器中去取就可以了
  2. 我们在创建实例的时候不需要了解其中的细节,降低了使⽤资源双⽅的依赖程度,也就是耦合度.

Bean的存储

在 Spring 中,你不需要自己创建对象,你只需要告诉 Spring,哪些类我需要创建出对象,然后在启动项目的时候 Spring 就会自动帮你创建出该对象,并且只存在一个类的实例。这个类的实例在 Spring 中被称为 Bean。而这种模式,我们称之为"单例模式"。也就是一个类只有一个实例的意思。

那么我如何告诉Spring 哪些类需要创建对象?

  1. 类注解:@Controller、@Service、@Repository、@Component、@Configuration.
  2. ⽅法注解:@Bean

Controller(控制器存储)

控制层,接收请求,对请求进行处理,并响应

java 复制代码
//将对象存储到Spring中
@Controller
public class HelloController {
    public void  print(){
        System.out.println(" do controller");
    }
}

我们怎么知道这个对象已经成功存在Spring中了呢?如何Spring容器中获取对象?

java 复制代码
@SpringBootApplication
public class DemoApplication {
	public static void main(String[] args) {
		ApplicationContext context=SpringApplication.run(DemoApplication.class, args);
		HelloController bean = context.getBean(HelloController.class);
  		bean.print();
	}
}

运行程序,成功打印,说明从Spring中获取到了Controller对象,并执行了print方法

获取Bean对象的其他方法

如果Spring面对同一类型存在多个bean,使用上图多个方法进行区分。Spring Bean是Spring框架运行时管理的对象,Spring会给管理的对象起名字(Bean命名约定)。

······命名约定使⽤Java标准约定作为实例字段名.也就是说,bean名称以⼩写字⺟开头,然后使⽤驼峰式⼤⼩写.

······当有多个字符并且第⼀个和第⼆个字符都是⼤写时,将保留原始的⼤⼩写

根据这些方法,我们做出一些示例:

java 复制代码
@SpringBootApplication
public class DemoApplication {
	public static void main(String[] args) {
		ApplicationContext context=SpringApplication.run(DemoApplication.class, args);
		//HelloController bean=new HelloController();
		HelloController bean=context.getBean(HelloController.class);//根据类型获取对象(适合该类型下该对象只有应该)
		bean.print();

		HelloController bean2=(HelloController) context.getBean("helloController");//根据对象名获取对象
	    bean2.print();

		HelloController bean3=context.getBean("helloController",HelloController.class);//根据类型和对象名获取对象
		bean3.print();
	}

}

获取Bean对象这个功能由⽗类BeanFactory提供的

Service(服务存储)

业务逻辑层,处理业务的具体逻辑

java 复制代码
@Service
public class UserService {
    public void print(){
        System.out.println("do service");
    }
}
java 复制代码
@SpringBootApplication
public class DemoApplication {
	public static void main(String[] args) {
		//获取上下文对象
		ApplicationContext context=SpringApplication.run(DemoApplication.class, args);
		UserService bean = context.getBean(UserService.class);
		bean.print();//使用对象
	}
}

Component(组件存储)

java 复制代码
@Component
public class UserComponent {
    public void print(){
        System.out.println("do Component");
    }
}

java 复制代码
@SpringBootApplication
public class DemoApplication {
	public static void main(String[] args) {
		//获取上下文对象
		ApplicationContext context=SpringApplication.run(DemoApplication.class, args);

		UserComponent bean3=context.getBean(UserComponent.class);
		bean3.print();
	}
}

Repository(仓库存储)

数据访问层,也叫做持久层,负责数据访问操作

java 复制代码
@Repository
public class UserRepository {
    public void print(){
        System.out.println("do UserRepository");
    }
}
java 复制代码
@SpringBootApplication
public class DemoApplication {
	public static void main(String[] args) {
		//获取上下文对象
		ApplicationContext context=SpringApplication.run(DemoApplication.class, args);
		UserRepository bean2=context.getBean(UserRepository.class);
		bean2.print();
	}
}

Configuration(配置存储)

配置层,处理项目中的配置信息

java 复制代码
@Configuration
public class UserConfig {
    public  void print(){
        System.out.println("do Config");
    }
}
java 复制代码
@SpringBootApplication
public class DemoApplication {
	public static void main(String[] args) {
		//获取上下文对象
		ApplicationContext context=SpringApplication.run(DemoApplication.class, args);
		UserConfig bean4=context.getBean(UserConfig.class);
		bean4.print();
	}
}

总结:

观察@Controller / @Service / @Repository / @Configuration四个注解的源码,

说明它们都是@Component的"子类"

方法注解@Bean

单个对象

java 复制代码
@Component
public class StudentComponent {
    @Bean
    public Student student() {
        Student student=new Student();
        student.setName("zhangsan");
        student.setAge(12);
        return student;
    }
}

获取对象

java 复制代码
@SpringBootApplication
public class DemoApplication {
	public static void main(String[] args) {
		//获取上下文对象
		ApplicationContext context = SpringApplication.run(DemoApplication.class, args);
		Student bean6 = context.getBean(Student.class);
		System.out.println(bean6);
     }
}

⽅法注解要配合类注解使⽤

多个Bean对象

如果有多个Bean对象,我们获取对象的时候应该获取哪个呢?

肯定会报错,Spring不知道要选择哪个,所以需要加@Primary注解

java 复制代码
@Component
public class StudentComponent {
    @Bean
    public Student student() {
        Student student=new Student();
        student.setName("zhangsan");
        student.setAge(12);
        return student;
    }
    @Primary
    @Bean
    public Student student2() {
        return new Student("da",13);
    }
}

完结

点一个免费的赞并收藏起来~
点点关注,避免找不到我~
我的主页:optimistic_chen我们下期不见不散 ~ ~ ~

相关推荐
wuk9986 小时前
在Spring MVC中使用查询字符串与参数
java·spring·mvc
YXWik66 小时前
java 使用 spring AI 实战 RAG (Chroma 向量数据库+Advisor)
java·人工智能·spring
bobz9656 小时前
ipsec mtu 问题
后端
深圳蔓延科技6 小时前
NIO和传统 BIO的性能较量
后端
程序员鱼皮7 小时前
Java 8 终于要被淘汰了!带你速通 Java 8~24 新特性 | 又能跟面试官吹牛皮了
java·后端·程序员
上官浩仁7 小时前
springboot jackson json入门与实战
java·spring boot·json
YwillD7 小时前
SpringCloud添加ai微服务(2)
spring·spring cloud·微服务
IT_陈寒9 小时前
Spring Boot 3 + GraalVM:5个实战技巧让Java应用启动速度提升300%
前端·人工智能·后端
无奈何杨9 小时前
CoolGuard风控系统配置评分卡、权重策略|QLExpress脚本
前端·后端