Spring

程序的耦合

耦合:程序间的依赖关系
包括:类之间的依赖,方法之间的依赖
解耦:降低程序见的依赖关系
实际开发中,应该做到编译期不依赖,运行时才依赖

解耦的思路:

第一步,使用反射来创建对象,而避免使用new关键字
第二部,通过读取配置文件来获取要创建的对象全限定类名

//1.注册驱动
//DriverManager.registerDriver(new com.mysql.jabc.DriverI());依赖于jar包,如果缺少则会在编译期时就报错
Class.forName("com.mysql.jdbc.Driver");//一个字符串
//2.获取连接
Connection conn=DriverManager.getConnection(url:"jdbc:mysql://localhost:3306");
//3.获取操作数据库的预处理对象
PreparedStatement pstm=conn.prepareStatement(sql:"select * from emp");
//4.执行SQL,得到结果集
ResultSet rs=pstm.executeQuery();
//5.遍历结果集
while(rs.nest()){
  System.out.println(rs.getString(columnLabel:"enam"));
}
//6.释放资源
rs.close();
pstm.close();
conn.close();

创建Bean对象的工厂

Bean:在计算机英语中,有可重用组件的含义。
JavaBean:用Java语言编写的可重用组件。
JavaBean > 实体类
它就是创建我们的service和dao对象的。
第一个:需要一个配置文件来配置我们的service和dao配置的内容:唯一标识=全限定类名(key=value)
第二个:通过读取配置文件中配置的内容,反射和创建对象
配置文件可以是xml也可以是properties

获取spring的Ioc核心容器,并根据id获取对象

ApplicationContext的三个常用实现类:

ClassPathXmlApplicationContext:它可以加载类路径下的配置文件,要求配置文件必须在类路径下,不在的话加载不了。
FileSystemXmlApplicationContext:它可以加载磁盘任意路径下的配置文件(必须有访问权限)
AnnotationConfigApplicationContext:它是用于读取注解创建容器的,

核心容器的两个接口引发出的问题

ApplicationContext:
单例对象适用
它在构建核心容器时,创建对象采取的策略是采用立即加载的方式,也就是说,只要一读取完配置文件马上就创建配置文件中的对象。
BeanFactory:
多例对象适用
它在构建核心容器时,创建对象采取的策略时采用延迟加载的方式,也就是说,什么时候根据id获取对象了,什么时候才真正的创建对象。

//1.获取核心容器对象
ApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");
//2.根据id获取bean对象
IAccountService as=(IAccountService)ac.getBean("accountService");
IAccountDao ad=ad.getBean("accountDao.class");
//BeanFactory
Resource resource=new ClassPathResource("bean.xml");
BeanFactory factory=new XmlBeanFactory(resource);
IAccountService as=(IAccountService)ac.getBean("accountService");

把对象的创建交给spring来管理

spring对bean的管理细节
1.创建bean的三种方式
2.bean对象的作用范围
3.bean对象的生命周期

创建Bean的三种方式

第一种方式:使用默认构造函数创建
在spring的配置文件中使用bean标签,配以id和class属性之后,且没有其他属性和标签时,采用的就是默认构造函数创建bean对象,此时如果类中没有默认构造函数,则对象无法创建

<bean id="accountService" class="com.neusoft.service.imp.AccountServiceImp"></bean>
这种就没有默认无参构造方法,不能创建

第二种方式:使用普通工厂中的方法创建对象
先配置一个类交给spring,再使用某个类中的方法创建对象,并存入spring容器

<bean id="instanceFactory" class="com.neusoft.factory.InstanceFactory"></bean>
<bean id="accountService" factory-bean="instanceFactory" factory-method="getAccountService"></bean>

第三种方式:使用工厂中的静态方法创建对象
使用某个类中的静态方法创建对象,并存入spring容器

<bean id="accountService" class="com.neusoft.factory.StaticFactory" factory-method="getAccountService"></bean>

bean的作用范围类调整

bean标签的scope属性
作用:用于指定bean的作用范围
取值:
singleton:单例的(默认值)
prototype:多例的
request:作用于web应用的请求范围
session:作用于web应用的会话范围
global-session:作用于集群环境的会话范围(全局会话范围),当不是集群环境时,就是session

bean对象的生命周期

单例对象
出生:当容器创建时,对象出生
活着:只要容器还在,对象一直活着
死亡:容器销毁,对象消亡
总结:单例对象的生命周期和容器相同

<bean id="accountService" class="com.neusoft.service.impl.AccountServiceImpl" scope="singleton" init-method="init" destory-method="destory"></bean>
//获取核心容器对象
//在这一步时就已经创建了对象
 ApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");
        
IAccountService as=(IAccountService) ac.getBean("accountService");

运行结果

无销毁的原因:
main方法是一切方法的入口,当main方法执行完以后,里面所有内存全部释放,包括容器也被释放了,所以容器还没来得及调用销毁方法就已经消失了,所以需要手动提交销毁方法:ac.close()
报红

这是因为我们在前面调用的是接口

ApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");

由于多态,我们无法使用在ClassPathXmlApplicationContext中的方法,所以使用本类即可

ClassPathXmlApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");
public class AccountServiceImpl implements IAccountService
{
//    @Autowired
    @Resource(name = "accountDao")
    private IAccountDao accountDao=null;

    public AccountServiceImpl(){
        System.out.println("对象创建了");
    }

    public void init(){
        System.out.println("对象已初始化");
    }

    public void destory(){
        System.out.println("对象已销毁");
    }

多例对象
出生:当我们使用对象时spring框架为我们创建
活着:对象在使用过程中就一直活着
死亡:当对象长时间不用,且没有别的对象引用时,由Java的垃圾回收器回收

//1.获取核心容器对象
ApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");
//2.根据id获取bean对象
//在这一步时才创建对象
IAccountService as=(IAccountService)ac.getBean("accountService");

依赖注入

依赖注入:Dependency Injection
IOC的作用:降低程序间的耦合(依赖关系)
依赖关系的管理:以后都交给spring来维护
在当前类需要用到其他类的对象,由spring为我们提供,我们只需要在配置文件中说明
依赖关系的维护:就称之为依赖注入

依赖注入:能注入的数据,有三类

基本类型和String
其他bean类型(在配置文件中或者注释配置过的bean)
复杂类型/集合类型

注入的方式:有三种

第一种:使用构造函数提供
第二种:使用set方法提供
第三种:使用注解提供

构造函数注入:

使用的标签:constructor.arg
标签出现的位置:bean标签的内部
标签中的属性
type:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型
Index:用于指定要注入的数据给构造函数中指定索引位置的参数赋值,索引的位置是从0开始的
name:用于指定给构造函数中指定名称的参数赋值
value:用于提供基本类型和String类型的数据
ref:用于指定其他的bean类型数据,它指的就是在spring的Ioc核心容器 中出现过的bean对象
优势:在获取bean对象时,注入数据是必须的操作,否则对象无法创建成功。
弊端:改变了bean对象的实例化方式,使我们在创建对象时,如果用不到这些参数,也必须提供。



只能赋给构造器中String类型的一个值name,并且如果有多个String类型,那么赋值哪一个也不确定,因此type不适合对于多个参数列表的赋值

set方法注入:

涉及的标签:property
出现的位置:bean标签内部
标签的属性:
name:用于指定注入时所调用的set方法名称
value:用于提供基本类型和String类型的数据
ref:用于指定其他的bean类型数据,它指的就是在spring的Ioc核心容器中出现过的bean对象
优势:创建对象时没有明确的限制,可以直接使用默认构造函数
弊端:如果有某个成员必须有值,则读取对象可能set方法没有执行

复杂类型的注入/集合类型的注入

用于给list结构集合注入的标签:list,array,set
用于给Map结构集合注入的标签:map,props
结构相同,标签可以互换

注解@:

用于创建对象的

作用就和在xml配置文件中编写一个<baen>标签实现的功能是一样的
@Component:
作用:用于把当前类对象存入spring容器中
属性:value:用于指定bean的id。当我们不写时,它的默认值时当前类名,且首字母改小写。
@Controller:一般用于表现层
@Service:一般用在业务层
@Repository:一般用在持久层
以上三个注解作用和属性于component一模一样,是spring框架为我们提供明确的三层使用的注解,使我们的三层对象更加清晰

用于注入数据的

作用就和在xml配置文件中的bean标签中写一个<property>标签的作用一样

Autowired:
作用:自动按照类型注入:只要容器中有唯一的一个bean对象类型和要注入的变量类型匹配,就可以注入成功
如果Ioc容器中没有任何一个bean的类型和要注入的变量类型匹配,则报错
如果Ioc容器中有多个类型匹配时,则再进行一次变量名匹配,当变量名与某个bean的id相同时才能运行

出现位置:可以是变量上,也可以是方法上
细节:在使用注解注入时,set方法就不是必须的
Qualifier:
作用:在按照类中注入的基础之上再按照名称注入。它再给类成员注入时不能单独使用。但是在给方法参数注入时可以。
属性:value,用来指定注入bean的id。


Resource:
作用:直接按照bean的id注入。它可以独立使用
属性:name,用于指定bean的id。

以上三种注入都只能注入其他bean类型的数据,而基本类型和String类型无法使用上述注解实现。
Value:
作用:用于注入基本类型和String类型的数据
属性:value,用于指定数据的值,它可以使用spring中SpEL(也就是spring的el表达式)

用于改变作用范围的

作用就和在bean标签中使用scope属性实现的功能一致
Scope:
作用:用于指定bean的作用范围
属性:value,指定范围的取值,常用取值:singleton,prototy

和生命周期相关

作用和在bean标签中使用init-method和destroy-method的作用一致
PreDestroy:
作用:用于指定销毁方法
PostConstruct:
作用:用于指定初始化方法

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351