文章目录:一.Spring项目的创建1.先创建maven项目
2.添加国内源
3.添加spring依赖
4.创建spring配置文件
5.创建启动类
二.Bean对象的创建和读取1.Bean对象的创建与存储方式(1)类注解
(2)方法注解
(3)
2.Bean对象的获取方式(1)通过id
(2)通过类名
(3)id+类名
(4)注解@Autowired
(5)@Autowired @Resource区别
一.Spring项目的创建
1.先创建一个maven项目
2.添加国内源
查看Local reposittory目录下是否有settings这个文件
将settings.xml文件里在
3.添加spring-content依赖
4.添加配置文件在resources底下
在文件里添加如下内容
5.添加启动类
二.Bean对象的创建与读取
1.Bean对象的创建
我们先来创立一个Student类
package spring;
public class Student {
public String name="网红";
public int id=3;
public int score=97;
String s=new String();
public Student()
{
System.out.println("学生初始化");
}
public void say()
{
System.out.println("你好学生");
}
}
在resources文件底下,新建一个spring-config(文件名可以随意起) xml文件
在xml文件里引入:
将bean对象存储到spring容器中
然后我们就可以在里面注册bean 对象,我们将student对象注册到spring中
其中 class后面需要写 包名+类名
2.从spring容器中获取bean
1.因为我们是在spring中注册bean,如果我们想要获取到,我们需要先获取到spring上下文,这里我们就涉及到两个类(ApplicationContext、BeanFactory)
2.然后根据注册的id,来获取对应的bean对象
public static void main(String[] args) {
(1)//获取spring上下文
ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
(2)//根据id来取容器里对应的bean对象
Student result=(Student)context.getBean("student");
result.say();
}
3.ApplicationContext BeanFactory有什么区别:
我们再来创建一个Teacher类,然后在spring中将实例化的对象给注册到里面去
package spring;
import com.sun.scenario.effect.impl.sw.sse.SSEBlend_SRC_OUTPeer;
public class Teacher {
public Teacher()
{
System.out.println("教师初始化");
}
public void say()
{
System.out.println("你好,教师");
}
}
我们先来用 ApplicationContext 来演示
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
}
我们可以发现当用去获取上下文时,spring中所有的类都被加载了,也就是在这时我们才真正的去创建bean对象,一开始在spring-config.xml中我们只是去注册、声明bean对象。
我们再来看一下,BeanContext
public static void main(String[] args) {
BeanFactory context=new XmlBeanFactory(new ClassPathResource("spring-config.xml"));
}
当我们去运行这个程序的时候,发现什么都没有
public static void main(String[] args) {
BeanFactory context=new XmlBeanFactory(new ClassPathResource("spring-config.xml"));
Student result=(Student)context.getBean("student");
}
这时我们再调用一下getBean()方法,我们再运行程序
由此我们可以得出当我们去使用BeanFactory的时候,我们发现只有当我们去取这个对象的时候,我们才会真正去加载并创建它
区别:(1)BeanFactory是 ApplicationContext的父类,其功能和方法相较于ApplicationContext要更少,这样效率、性能会高
(2)ApplicationContext一次加载并实例化所有的Bean对象,而BeanFactory是能用到那个Bean对象再去加载初始化这个对象的,这样节省空间,但效率低
4.Spring存取对象的基本流程:
1.Spring读取Bean流程:先扫描xml文件配置,从配置中或注解中获取Bean对象的定义信息,然后把它注册到Spring容器中
2.Spring加载并实例化Bean:先给Bean对象分配空间,然后进行初始化,然后把它设置到Spring容器中
5.获取bean对象的几种方式
(1)通过id名称获取bean
public static void main11(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
Student result=(Student)context.getBean("student");
result.say();
}
(2)通过类名获取bean:
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
Student result=context.getBean(Student.class);
result.say();
}
当我们通过类名的方式去获取对象的时候,我们发现这个时候,我们不用再进行强转成Student类型了
但是这种方式存在一个问题,如果我们再注册一个Student类的对象在里面
这时我们再去通过这种方式获取这就存在问题了。
(3)正确的方式应该是通过 id+类名的方式
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
Student student1 = context.getBean("student", Student.class);
Student student2 = context.getBean("stu", Student.class);
student1.say();
student2.say();
}
(4)更简单的存储Bean对象的方式,添加类注解
五大类注解:1.@Controller (控制器,用来验证前端传来的数据是否正确,如果不正确,返回false)防止别人恶意访问程序,相当于安保系统
2.@Service 服务:负责调度和提供一些方法,但是不实际执行,相当于客服中心
3.@Repository 持久层:和数据层交互,负责实际执行
4.@Component 组件工具类
5.@Configuration 配置项(项目当中的一些配置)
在这里边Compoent是所有这几个类的父类,其他的四个类都继承了这个类,我们随便打开其他四个类里面的一个类看看:
通过在在类外添加注解的方式也能存储Bean对象,但是我们需要在spring-conflig.xml中增加
base-package表明类所在的包
package com.gather;
import org.springframework.context.annotation.Configuration;
@Configuration
public class Student{
public void say()
{
System.out.println("hello student");
}
}
像这种添加类注解的方式,我们用getBean()的方式来获取的话,那么里面的参数应该如何写呢?其实也是一种类似于Bean名称+类名的方式,如果类名第一个字母大写的话,我们只需把第一个字母变成小写
public class Text {
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
Student student=(Student) context.getBean("student",Student.class);
student.say();
}
}
在这里我们看到如果类名第一个字母大写的话,id直接将类名第一个字母小写即可,
如果我们建的类名前两个都是大写的话,这样就不可以了
public class POlice {
public void say()
{
System.out.println("hello police");
}
}
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
POlice police=context.getBean("police",POlice.class);
police.say();
}
如果id直接也是把第一个字母小写的话,我们发现会报错,
这时候id应该直接写类名
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
POlice police=context.getBean("POlice",POlice.class);
police.say();
}
那么为啥是这样呢?我们来看一下这里相关的源码:
Bean命名规则:首字母默认是小写的,如果首字母第一个大写 ,Bean名称需要将第一个字母小写,如果前两个字母都是大写,那么Bean名称就是类名。
在这里我们还需要注意一个问题:
如果在spring-conflig.xml里只有这一种通过添加注解的方式来存取Bean对象
我们要存取和Bean对象与之对应的类必须在这个包底下或者在这个包的子目录里,另外如果想要访问这个包底下与之对应的所有Bean对象,这些类都要加注解,举个例子:
因为Farmer这个类没在com.gather这个包底下,也不在这个包的子包底下,所以相当于我们没把它注册到Spring容器中,所以就取不到这个对象
那应该如何解决呢?
我们在spring-conflig.xml再加一行Bean注册内容即可
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
Farmer farmer=context.getBean("farmer",Farmer.class);
farmer.say();
}
通过在方法上加Bean注解的方式将Bean对象存储在服务器上,但是在类上我们要加上类注解,因为这样我们是存在一些类不需要去存储在Spring中,这样再去读取扫描和配置文件的时候,可以快速的定位存储在Spring中的类,方便快速取出对象,这样可以提高性能。
@Configuration
public class Use2 {
@Bean
public User getuser2()
{
User user=new User();
user.name="小凯";
user.age=5;
return user;
}
}
但是要注意这里的方法名必须是要存储Bean对象的类名
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
User user=context.getBean("getuser2",User.class);
System.out.println(user.name);
}
这里要注意getBean()方法里的第一个参数是方法名
我们来看一下打印信息
getBean()方法里面第一个参数也就是Bean名称是一个方法名,感觉怪怪的,其实我们是可以重新起一个名字的
@Configuration
public class User1 {
@Bean(value = "user2")
public User getuser2()
{
User user=new User();
user.name="小凯";
user.age=5;
return user;
}
}
也就是在Bean注解之后用value重新起一个名,getBean()方法里第一个参数也就可以写这个名了
User user=context.getBean("user2",User.class);
但是这里要注意如果重新起了名,getBean()方法里第一个参数就不能再用原来的方法名了(getuser2)
大家在这里想一个问题,如果我们再在另一个类里面写一个和User1里面一模一样的方法,那么当我们再去取得时候
@Configuration
public class User2 {
@Bean//(value="user7")
public User getuser2() {
User user = new User();
user.name = "小中";
user.age = 5;
return user;
}
}
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
User user=context.getBean("getuser2",User.class);
System.out.println(user.name);
}
这取决于谁先注入,谁先被加载,谁后注入,后注入的会覆盖先注入的,我们再来写段代码看看,
我们在两个类前面分别加上两个@Order(10) 、@Order(20) 这里的注解表示的是谁先注入,数字越小表示注入的越早,一开始是User1里面的先注入,当我们把数值改一@Order(20)@Order(10)
这时就会产生不同的结果了。
取Bean更简单的方式使用@Autowired、Resource
@Autowired 注解表示将需要的对象从Spring容器里取出来,
1.属性注入
我们先建立一个Police类
@Configuration
public class Police {
public String name;
public int age;
public void say()
{
System.out.println("hello police");
}
}
然后再创建一个UsePolice类,我们通过属性注入的方式将Police对象注入,
@Configuration
public class UsePolice {
@Autowired
Police police;
public void getpolice()
{
police.say();
}
}
然后我们拿到UsePolice这个类型的对象
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
UsePolice usePolice=context.getBean("usePolice",UsePolice.class);
usePolice.getpolice();
}
属性注入的优点:简单方便 弊端:(1)不符合单一设计的原则,我们可以将多个类型的对象共同注入(2)不能注入被final修饰的对象
因为final修饰的变量要么在后面直接赋值,要么利用构造方法赋值。
(3) 只适用于Ioc容器,不具备通用性
2.Setter注入
@Configuration
public class UsePolice2 {
public Police police;
@Autowired
public void setPolice(Police police) {
this.police = police;
}
public void getpolice()
{
police.say();
}
}
Setter方法注入的优点:(1)符合单一设计的原则 缺点:(1)不能注入被final修饰的对象
(2)使用Setter注入的对象可能被修改 比如我们在getPolice()方法里将police置为null
@Configuration
public class UsePolice2 {
public Police police;
@Autowired
public void setPolice(Police police) {
this.police = police;
}
public void getpolice()
{
this.police=null;
police.say();
}
}
3.构造方法注入
@Component
public class UsePolice3 {
public Police police;
@Autowired
public UsePolice3(Police police)
{
this.police=police;
}
public void getpolice()
{
police.say();
}
}
构造方法注入的特点:如果当前类里面只有一个构造方法的话,那么@Autowired可以被省略
构造方法注入优点分析:1.能注入被final修饰的对象
2.构造方法只执行一次,注入对象不会再被修改
3.构造方法注入不止适用于Ioc容器,具有通用性。
4.构造方法注入会将注入的对象完全初始化
缺点:构造方法里可以传多个参数,可能会违背单一设计原则。
与@Autowired注解作用差不多的还有@Resource注解
@Resource同样可以属性注入和Setter注入,但是如果是构造方法注入就会报错
接下来我们来思考一个问题:如果有两个Police对象,这时注入的是谁呢?我们启动一下
@Component
public class Useuser4 {
@Bean(value="police1")
public Police getpolice()
{
Police police=new Police();
police.name="小海";
police.age=9;
return police;
}
@Bean
public Police getpolice2()
{
Police police =new Police();
police.name="小小";
police.age=9;
return police;
}
}
Component
public class Usepolice {
@Resource
public Police police;
public void say()
{
police.say();
System.out.println(police.name);
}
}
public static void main(String[] args) {
ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
Usepolice useuser4=context.getBean("usepolice",Usepolice.class);
useuser4.say();
}
打印结果:
我们发现会报错,这是因为Resource先根据名称去查,然后再根据类型去查,
这里的名称也就是police ,我们发现注册的两个Police实例都不是这个名称,然后再根据Police类型去找,发现两个Police实例也不是以Police为名称的
这个问题的解决办法就是在Resource后面加上具体的Bean对象的名称
@Resource(name="police1")
public Police police;
这时候就指定是注入police1这个对象了
那使用@Autowired能不能指定指定的对象呢?其实也是可以的
@Autowired
@Qualifier(value="police1")
总结:@Resource注解和@Autowired的区别
(1)@Autowired来自Spring @Resource来自jdk的注解
(2)@Autowired 支持属性注入、Setter注入、和构造方法注入,@Resource不支持构造方法注入
(3)相较于@Autowired@Resource可以设置更多的参数可以在后面指定name,依次来获取Bean
Bean的作用域