base黑马 视频链接:https://www.bilibili.com/video/BV1WZ4y1P7Bp?t=94.4
此系列文章可以当做视频的配套笔记,也可以当做自学SSM框架的教程。
Spring配置文件详解1
id不允许重复
默认情况下的反射是通过无参构造来创建对象的,Spring默认情况下也是找的无参构造。
那就必须保证UserDaoImpl这个类中必须存在无参构造
Spring配置文件详解2
bean的创建时机 :
如果bean的scope是singleton时它的创建时机就是,当加载配置文件创建Spring容器时,这个bean就创建了。
加载配置文件创建Spring容器就是执行这一行代码:
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
如果bean的scope是prototype,它的创建时机是在每次getBean时创建一个对象。
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
UserDao userDao2 = (UserDao) applicationContext.getBean("userDao");
Spring配置文件详解3
初始化方法、销毁方法
在bean中配置自己定义的初始化方法 和 销毁方法
<bean id="userDao" class="com.lyh.dao.impl.UserDaoImpl" init-method="init" destroy-method="destroy"/>
@Test //初始化方法 和 销毁方法的 测试
public void test2(){
//这个是一个单元测试,没有在一个TomCat服务器内部
//只要这个单元测试执行完毕,那么该销毁的都给销毁了
//这里没有出现销毁的方法 是 因为 还没有来得及打印
//如果想测试出打印 只需要让容器 手动的关闭就是可以了
//但是这个ApplicationContext接口,没有对应的关闭的方法
//所以只能用 ClassPathXmlApplicationContext 这个实现类中的关闭的方法
ApplicationContext applicationContext =
new ClassPathXmlApplicationContext("applicationContext.xml");
// 这里就是当前xml配置文件的名称
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
}
@Test //初始化方法 和 销毁方法的 测试
public void test3(){
ClassPathXmlApplicationContext applicationContext =
new ClassPathXmlApplicationContext("applicationContext.xml");
// 这里就是当前xml配置文件的名称
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
applicationContext.close();
}
test3的效果:
Spring配置文件详解4(bean实例化的三种方式)
1、工厂静态方法实例化:
package com.lyh.factory;
import com.lyh.dao.UserDao;
import com.lyh.dao.impl.UserDaoImpl;
public class StaticFactory {
//对象的工厂创建完毕
public static UserDao getUserDao(){
//返回的是创建一个对象
return new UserDaoImpl();
}
}
applicationContext.xml配置文件中的内容
<bean id="userDao" class="com.lyh.factory.StaticFactory" factory-method="getUserDao"/>
测试代码:工厂静态方法实例化
@Test //工厂静态方法实例化
public void test4(){
ClassPathXmlApplicationContext applicationContext =
new ClassPathXmlApplicationContext("applicationContext.xml");
// 这里就是当前xml配置文件的名称
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
}
执行效果:
2、工厂实例方法实例化
package com.lyh.factory;
import com.lyh.dao.UserDao;
import com.lyh.dao.impl.UserDaoImpl;
public class DynamicFactory {
//这里就必须得有 工厂对象,再去调用方法
public UserDao getUserDao(){
return new UserDaoImpl();
}
}
<!--先把这个工厂对象让Spring容器给产生 再调用 这个工厂内部的 某个方法 -->
<bean id="factory" class="com.lyh.factory.DynamicFactory"/>
<!--我要获得userDao这个对象,我要找factory工厂的 getUserDao 方法-->
<bean id="userDao" factory-bean="factory" factory-method="getUserDao"/>
测试方法:
@Test //工厂实例方法实例化
public void test4(){
ClassPathXmlApplicationContext applicationContext =
new ClassPathXmlApplicationContext("applicationContext.xml");
// 这里就是当前xml配置文件的名称
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
}
效果:
Spring配置文件-依赖注入1
bean依赖注入的概念
依赖注入的方法:
service层的代码:
package com.lyh.service.impl;
import com.lyh.dao.UserDao;
import com.lyh.service.UserService;
public class UserServiceImpl implements UserService {
private UserDao userDao;
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
@Override
public void save() {
userDao.save();
}
}
applicationContext.xml文件配置
<bean id="userDao" class="com.lyh.dao.impl.UserDaoImpl"/>
<bean id="userService" class="com.lyh.service.impl.UserServiceImpl">
<!--这里的name指的是在 UserServiceImpl这个类中的 set方法 去掉set首字母小写的名字就是userDao-->
<!-- ref就是上面的id="userDao"中的值-->
<property name="userDao" ref="userDao"/>
</bean>
测试方法:
package com.lyh.demo;
import com.lyh.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class UserController {
public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
}
}
效果:
延伸:
package com.lyh.demo;
import com.lyh.service.UserService;
import com.lyh.service.impl.UserServiceImpl;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class UserController {
public static void main(String[] args) {
UserService userService = new UserServiceImpl();
userService.save();
}
}
这里会报空指针:
因为现在的controller层不是从 容器中拿的,容器中的Service层注着Dao层,现在自己new的对象的Service层没有注入Dao对象是空的,所以会报空指针异常。
Spring配置文件-依赖注入4
set方法注入,P命名空间注入:
就是把 property 标签给省略掉了
<bean id="userDao" class="com.lyh.dao.impl.UserDaoImpl"/>
<bean id="userService" class="com.lyh.service.impl.UserServiceImpl">
<!--这里的name指的是在 UserServiceImpl这个类中的 set方法 去掉set首字母小写的名字就是userDao-->
<!-- ref就是上面的id="userDao"中的值-->
<property name="userDao" ref="userDao"/>
</bean>
引入P命名空间放到 applicationContext.xml文件中的头文件
xmlns:p=“http://www.springframework.org/schema/p”
applicationContext.xml文件bean
<bean id="userService" class="com.lyh.service.impl.UserServiceImpl" p:userDao-ref="userDao"/>
p:userDao-ref=" " 注入对象属性
p:userDao=" " 注入普通属性
测试代码:
package com.lyh.demo;
import com.lyh.service.UserService;
import com.lyh.service.impl.UserServiceImpl;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class UserController {
public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
}
}
运行效果:
这种方法也是可以的
Spring配置文件-依赖注入5(有参构造方法注入)
UserServiceImpl中的代码
package com.lyh.service.impl;
import com.lyh.dao.UserDao;
import com.lyh.service.UserService;
public class UserServiceImpl implements UserService {
private UserDao userDao;
//有参构造
public UserServiceImpl(UserDao userDao) {
this.userDao = userDao;
}
//无参构造
public UserServiceImpl() {
}
@Override
public void save() {
userDao.save();
}
}
applicationContext.xml配置文件
<bean id="userDao" class="com.lyh.dao.impl.UserDaoImpl"/>
<bean id="userService" class="com.lyh.service.impl.UserServiceImpl">
<!-- 这里的name是构造方法中的参数名字,ref就是上面的id="userDao"-->
<constructor-arg name="userDao" ref="userDao"/>
</bean>
测试代码:
package com.lyh.demo;
import com.lyh.service.UserService;
import com.lyh.service.impl.UserServiceImpl;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class UserController {
public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
}
}
测试效果:
Spring配置文件-依赖注入6(普通数据类型注入)
普通数据类型的注入
UserDaoImpl类中的代码
package com.lyh.dao.impl;
import com.lyh.dao.UserDao;
public class UserDaoImpl implements UserDao {
private String username;
private int age;
//这里可以通过 set方法 来注入,也可以通过 构造方法 来注入
//这里就使用 set 方法来注入了 相对来说set方法用的比较多一些
public void setUsername(String username) {
this.username = username;
}
public void setAge(int age) {
this.age = age;
}
@Override
public void save() {
System.out.println(username+"===="+age);
System.out.println("save方法执行了。。。");
}
}
applicationContext.xml中的内容
<bean id="userDao" class="com.lyh.dao.impl.UserDaoImpl">
<property name="username" value="李永恒"/>
<property name="age" value="18"/>
</bean>
测试代码进行测试:
package com.lyh.demo;
import com.lyh.service.UserService;
import com.lyh.service.impl.UserServiceImpl;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class UserController {
public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
}
}
执行结果:
Spring配置文件-依赖注入7(集合数据类型的注入)
package com.lyh.dao.impl;
import com.lyh.dao.UserDao;
import com.lyh.domain.User;
import java.util.List;
import java.util.Map;
import java.util.Properties;
public class UserDaoImpl implements UserDao {
private List<String> strList;
private Map<String,User> userMap;
private Properties properties;
public void setStrList(List<String> strList) {
this.strList = strList;
}
public void setUserMap(Map<String, User> userMap) {
this.userMap = userMap;
}
public void setProperties(Properties properties) {
this.properties = properties;
}
@Override
public void save() {
System.out.println(strList);
System.out.println(userMap);
System.out.println(properties);
System.out.println("save方法执行了。。。");
}
}
User.java文件
package com.lyh.domain;
public class User {
private String name;
private String addr;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getAddr() {
return addr;
}
public void setAddr(String addr) {
this.addr = addr;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", addr='" + addr + '\'' +
'}';
}
}
applicationContext.xml 配置文件
<!-- bean就是创建对象的-->
<bean id="user1" class="com.lyh.domain.User">
<property name="name" value="tom"/>
<property name="addr" value="北京"/>
</bean>
<!-- 在创建一个对象-->
<bean id="user2" class="com.lyh.domain.User">
<property name="name" value="lucy"/>
<property name="addr" value="天津"/>
</bean>
<bean id="userDao" class="com.lyh.dao.impl.UserDaoImpl">
<!-- strList既不是一个普通值,也不是容器内部的对象引用-->
<property name="strList">
<list>
<!-- 如果 list泛型不是一个普通字符串 而是一个其他的对象,那个这里就用<ref></ref>标签-->
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</list>
</property>
<property name="userMap">
<map>
<!-- 这里的这个key叫什么名字都行 value-ref 代表 id的一个引用-->
<entry key="u1" value-ref="user1"/>
<entry key="u2" value-ref="user2"/>
</map>
</property>
<property name="properties">
<props>
<prop key="p1">ppp1</prop>
<prop key="p2">ppp2</prop>
<prop key="p3">ppp3</prop>
</props>
</property>
</bean>
<bean id="userService" class="com.lyh.service.impl.UserServiceImpl">
<!-- 这里的name是构造方法中的参数名字,ref就是上面的id="userDao"-->
<constructor-arg name="userDao" ref="userDao"/>
</bean>
测试代码:
package com.lyh.demo;
import com.lyh.dao.UserDao;
import com.lyh.service.UserService;
import com.lyh.service.impl.UserServiceImpl;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class UserController {
public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
}
}
测试效果:
Spring配置文件-import和知识要点
Spring相关API(1)
Spring相关API(2)
存在两个常用的API
一个是传递一个字符串,就是常用的id
还有一个是传递一个字节码对象类型
UserService userService = applicationContext.getBean(UserService.class);
UserService userService = applicationContext.getBean(“userService”,UserService.class); //这种也是可以的
UserService userService = (UserService) applicationContext.getBean(“userService”);
使用这种方式允许出现多个相同类型的bean
<!-- 这两个bean的类型是相同的 -->
<bean id="userService" class="com.lyh.service.impl.UserServiceImpl">
<!-- 这里的name是构造方法中的参数名字,ref就是上面的id="userDao"-->
<constructor-arg name="userDao" ref="userDao"/>
</bean>
<bean id="userService2" class="com.lyh.service.impl.UserServiceImpl">
<!-- 这里的name是构造方法中的参数名字,ref就是上面的id="userDao"-->
<constructor-arg name="userDao" ref="userDao"/>
</bean>
UserService userService = applicationContext.getBean(UserService.class);
这种方式,只适用于容器中存在的某一种类型的bean是一个的情况。