程序的耦合
耦合:程序间的依赖关系
包括:类之间的依赖,方法之间的依赖
解耦:降低程序见的依赖关系
实际开发中,应该做到编译期不依赖,运行时才依赖
解耦的思路:
第一步,使用反射来创建对象,而避免使用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:
作用:用于指定初始化方法