spring学习笔记一 入门及配置

2019-08-16 10:17:21来源:博客园 阅读 ()

新老客户大回馈,云服务器低至5折

spring学习笔记一 入门及配置

Spring的核心是:

IOC:控制反转也叫依赖注入(DI);AOP:面向切面编程,这两个是Spring的灵魂。简单来说,Spring是一个分层的一站式轻量级开源框架。

IoC和DI由什么关系呢?其实它们是同一个概念的不同角度描述,由于控制反转概念比较含糊(ioc不是什么技术,而是一种设计思想,Ioc意味着将你设计好的对象交给容器控制,而不是传统的在你的对象内部直接控制。),所以2004年大师级人物Martin Fowler又给出了一个新的名字:“依赖注入”,相对IoC 而言,“依赖注入”明确描述了“被注入对象依赖IoC容器配置依赖对象”。

原理: 主要用到的设计模式有工厂模式和代理模式。
  1.IOC就是典型的工厂模式,通过sessionfactory去注入实例。
  2.AOP就是典型的代理模式的体现。cglib和动态代理>>invoke反射机制。

IOC说明:将对象交给容器管理,你只需要在spring配置文件总配置相应的bean(ioc-控制生成bean),以及设置相关的属性,让spring容器来生成类的实例对象以及管理对象。//在spring容器启动的时候,spring会把你在配置文件中配置的bean都初始化好,然后在你需要调用的时候,就把它已经初始化好的那些bean分配给你需要调用这些bean的类(假设这个类名是A),动态的向某个对象提供它所需要的其他对象(DI依赖注入-获取对象这时候反转了),而不需要你在A里面new这些bean了。

AOP说明:动态地将代码切入到类的指定方法、指定位置上的编程思想就是面向切面的编程 。

AOP可以说是对OOP的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。

实现aop的两种方式:一是采用动态代理技术,利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行;二是采用静态织入的方式,引入特定的语法创建“方面”,从而使得编译器可以在编译期间织入有关“方面”的代码.

简单点解释举例:比方说你想在你的biz层所有类中都加上一个打印‘你好’的功能,这时就可以用aop思想来做.你先写个类写个类方法,方法经实现打印‘你好’,然后Ioc这个类 ref=“biz.*”让每个类都注入即可实现。 比如事务管理权限控制日志记录、性能统计等。

spring项目搭建:

1、导包

除了上面的4个包之外还需要日志包:

下面的包可选,老版本有需要:

com.springsource.org.apache.log4j-1.2.15.jar

2、创建一个对象

package com.yyb.bean;

public class User {
    
    
    public User() {
            System.out.println("User对象空参构造方法!!!!");
    }
    private String name;
    private Integer age;
    private Car car;
    
    
    
    public User(String name, Car car) {
        System.out.println("User(String name, Car car)!!");
        this.name = name;
        this.car = car;
    }
    
    public User(Car car,String name) {
        System.out.println("User(Car car,String name)!!");
        this.name = name;
        this.car = car;
    }
    
    public User(Integer name, Car car) {
        System.out.println("User(Integer name, Car car)!!");
        this.name = name+"";
        this.car = car;
    }
    
    public Car getCar() {
        return car;
    }
    public void setCar(Car car) {
        this.car = car;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Integer getAge() {
        return age;
    }
    public void setAge(Integer age) {
        this.age = age;
    }
    
    public void init(){
        System.out.println("我是初始化方法!");
    }
    public void destory(){
        System.out.println("我是销毁方法!");
    }
    @Override
    public String toString() {
        return "User [name=" + name + ", age=" + age + ", car=" + car + "]";
    }
    
}
View Code
package com.yyb.bean;

public class Car {
    private String  name;
    private String color;
    
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getColor() {
        return color;
    }
    public void setColor(String color) {
        this.color = color;
    }
    @Override
    public String toString() {
        return "Car [name=" + name + ", color=" + color + "]";
    }
    
    
}
View Code

 

3、书写配置注册对象到容器

配置文件放的位置任意(建议放到src下);配置文件名任意(建议applicationContext.xml)。

在配置文件中添加如下代码:

4、代码测试

复制代码
package com.yyb.springTest;

import com.yyb.bean.User;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

/**
 * Created by Administrator on 2017/8/8.
 */
public class SpringTest {
    @Test
   public void func1(){
       ApplicationContext ac =new ClassPathXmlApplicationContext("applicationContext.xml");
       User user = (User)ac.getBean("user");
       System.out.println(user);
   }
}
复制代码

打印结果如下,可已发现User对象创建成功了:

User对象空参构造方法!!!!
User [name=null, age=null, car=null]

IOC和AOP

IOC: Inverse Of Control  ,即反转控制。指的是对象的创建权反转(交给)给Spring。作用是实现了程序的解耦合。

以前对象的创建是由我们自己创建,比如Service层中需要使用Dao层的类,则直接在Service层中使用new创建Dao层的对象。两层之间严重耦合。使用了Spring之后,对象的创建以及依赖关系可以由spring完成创建以及注入。

注入方式:

  1. set方法注入
  2. 构造方法注入
  3. 接口注入

注入类型:

  1. 值类型注入 (8大基本数据类型)
  2. 引用类型注入(将依赖对象注入)

BeanFactory(过时)

 BeanFactory接口,是spring原始接口。针对原始接口的实现类功能较为单一。BeanFactory接口实现类的容器特点是每次在获得对象时才会创建对象。

ApplicationContext

 每次容器启动时就会创建容器中配置的所有对象,并提供更多功能。

两个典型的实现类:

  •  从类路径下加载配置文件:ClassPathXmlApplicationContext。
  •  从硬盘绝对路径下加载配置文件:FileSystemXmlApplicationContext("d:/xxx/yyy/xxx")。

 结论:web开发中,使用applicationContext。在资源匮乏的环境可以使用BeanFactory。

Spring配置详解

 1、Bean元素

使用Bean元素描述需要spring容器管理的对象。

  • class属性:被管理对象的完整类名。
  • name属性:给被管理的对象起个名字.获得对象时根据该名称获得对象。 可以重复;可以使用特殊字符。
  • id属性:与name属性一模一样.。但名称不可重复;不能使用特殊字符。尽量使用name属性。
<bean  name="user" class="cn.itcast.bean.User" ></bean>

2、Bean元素进阶

scope属性

  • singleton(默认值):单例对象,被标识为单例的对象在spring容器中只会存在一个实例。
  • prototype:多例原型,被标识为多例的对象,每次在获得才会创建。每次创建都是新的对象。整合struts2时,ActionBean必须配置为多例的。
  • request:web环境下,对象与request生命周期一致。
  • session:web环境下,对象与session生命周期一致。
<bean  name="user" class="com.yyb.bean.User" scope="singleton"></bean>

测试代码:

复制代码
@Test
//scope:singleton 单例
//scope:prototype 多例
public void fun4(){
    //1 创建容器对象
    ApplicationContext ac = new ClassPathXmlApplicationContext("cn/itcast/b_create/applicationContext.xml");
    //2 向容器"要"user对象
    User u = (User) ac.getBean("user");
    User u2 = (User) ac.getBean("user");
    User u3 = (User) ac.getBean("user");
    User u4 = (User) ac.getBean("user");
    
    System.out.println(u2==u4);//单例:true
                               //多例:false
    //3 打印user对象
    System.out.println(u);
}
复制代码

生命周期属性

通过配置<bean>标签上的init-method作为Bean的初始化的时候执行的方法,配置destory-method作为Bean的销毁的时候执行的方法。销毁的方法想要执行,需要是单利创建的Bean,而且在工厂关闭的时候,Bean才会被销毁。

    <bean  name="user" class="cn.itcast.bean.User"
         init-method="init" destroy-method="destory" ></bean>

3、spring创建对象的方式

创建方式1:空参构造创建

<bean  name="user" class="cn.itcast.bean.User"></bean>

创建方式2:静态工厂创建 

    <!-- 创建方式2:静态工厂创建 
          调用UserFactory的createUser方法创建名为user2的对象.放入容器
     -->
    <bean  name="user2" class="cn.itcast.b_create.UserFactory" factory-method="createUser" ></bean>

UserFactory类

复制代码
package cn.itcast.b_create;

import cn.itcast.bean.User;

public class UserFactory {
    public static User createUser(){
        System.out.println("静态工厂创建User");
        return new User();
    }
}
复制代码

测试代码:

复制代码
//创建方式2:静态工厂
        @Test
        public void fun2(){
            //1 创建容器对象
            ApplicationContext ac = new ClassPathXmlApplicationContext("cn/itcast/b_create/applicationContext.xml");
            //2 向容器"要"user对象
            User u = (User) ac.getBean("user2");
            //3 打印user对象
            System.out.println(u);
        }
复制代码

创建方式3:实例工厂创建 

    <!-- 创建方式3:实例工厂创建 
         调用UserFactory对象的createUser2方法创建名为user3的对象.放入容器
     -->
    <bean  name="user3" factory-bean="userFactory" factory-method="createUser2" ></bean>
    <bean  name="userFactory" class="cn.itcast.b_create.UserFactory"   ></bean>

UserFactory类

复制代码
package cn.itcast.b_create;

import cn.itcast.bean.User;

public class UserFactory {
    public  User createUser2(){
        System.out.println("实例工厂创建User");
        return new User();
    }
}
复制代码

测试代码:

复制代码
@Test
public void fun3(){
  //1 创建容器对象
  ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
  //2 向容器"要"user对象
  User u = (User) ac.getBean("user3");
  //3 打印user对象
  System.out.println(u);
}
复制代码

4、spring的分模块配置

 在主配置文件中导入其他路径的配置文件

<!-- 导入其他spring配置文件 -->
<import resource="cn/itcast/b_create/applicationContext.xml"/>

Spring属性注入

set方法注入

复制代码
<!-- set方式注入: -->
    <bean  name="user" class="cn.itcast.bean.User" >
        <!--值类型注入: 为User对象中名为name的属性注入tom作为值 -->
        <property name="name" value="tom" ></property>
        <property name="age"  value="18" ></property>
        <!-- 引用类型注入: 为car属性注入下方配置的car对象 -->
        <property name="car"  ref="car" ></property>
    </bean>
复制代码
<!-- 将car对象配置到容器中 -->
    <bean name="car" class="cn.itcast.bean.Car" >
        <property name="name" value="兰博基尼" ></property>
        <property name="color" value="黄色" ></property>
    </bean>

User类中必须存在SetXXX()方法。

构造方法注入

复制代码
<!-- 构造函数注入 -->
<bean name="user2" class="cn.itcast.bean.User" >
    <!-- name属性: 构造函数的参数名 -->
    <!-- index属性: 构造函数的参数索引 -->
    <!-- type属性: 构造函数的参数类型-->
    <constructor-arg name="name" index="0" type="java.lang.Integer" value="999"  ></constructor-arg>
    <constructor-arg name="car" ref="car" index="1" ></constructor-arg>
</bean>
复制代码

前提,必须有相应的构造函数

    public User(Integer name, Car car) {
        System.out.println("User(Integer name, Car car)!!");
        this.name = name+"";
        this.car = car;
    }

p名称空间注入

复制代码
<!-- p名称空间注入, 走set方法
    1.导入P名称空间  xmlns:p="http://www.springframework.org/schema/p"
    2.使用p:属性完成注入
        |-值类型: p:属性名="值"
        |-对象类型: p:属性名-ref="bean名称"
 -->
    <bean  name="user3" class="cn.itcast.bean.User" p:name="jack" p:age="20" p:car-ref="car"  >
    </bean>
复制代码

spel注入

复制代码
<!-- 
    spel注入: spring Expression Language sping表达式语言
 -->
<bean  name="user4" class="cn.itcast.bean.User" >
        <property name="name" value="#{user.name}" ></property> <!--取对象为user的name值-->
        <property name="age" value="#{user3.age}" ></property> <!--取对象为user3的age值-->
 <property name="car" ref="car" ></property> </bean> 
复制代码

复杂类型注入

复制代码
<!-- 复杂类型注入 -->
<bean name="cb" class="cn.itcast.c_injection.CollectionBean" >
    <!-- 如果数组中只准备注入一个值(对象),直接使用value|ref即可 
    <property name="arr" value="tom" ></property>
    -->
    <!-- array注入,多个元素注入 -->
    <property name="arr">
        <array>
            <value>tom</value>
            <value>jerry</value>
            <ref bean="user4" />
        </array>
    </property>
    
    <!-- 如果List中只准备注入一个值(对象),直接使用value|ref即可 
    <property name="list" value="jack" ></property>-->
    <property name="list"  >
        <list>
            <value>jack</value>
            <value>rose</value>
            <ref bean="user3" />
        </list>
    </property>
    <!-- map类型注入 -->
    <property name="map"  >
        <map>
            <entry key="url" value="jdbc:mysql:///crm" ></entry>
            <entry key="user" value-ref="user4"  ></entry>
            <entry key-ref="user3" value-ref="user2"  ></entry>
        </map> 
    </property>
    <!-- prperties 类型注入 -->
    <property name="prop"  >
        <props>
            <prop key="driverClass">com.jdbc.mysql.Driver</prop>
            <prop key="userName">root</prop>
            <prop key="password">1234</prop>
        </props>
    </property>
</bean>
复制代码
CollectionBean类
复制代码
package cn.itcast.c_injection;

import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Properties;

public class CollectionBean {
    private Object[] arr;//数组类型注入
    private List list;//list/set 类型注入
    private Map map;//map类型注入
    private Properties prop;//properties类型注入
    
    public Object[] getArr() {
        return arr;
    }
    public void setArr(Object[] arr) {
        this.arr = arr;
    }
    public List getList() {
        return list;
    }
    public void setList(List list) {
        this.list = list;
    }
    public Map getMap() {
        return map;
    }
    public void setMap(Map map) {
        this.map = map;
    }
    public Properties getProp() {
        return prop;
    }
    public void setProp(Properties prop) {
        this.prop = prop;
    }
    @Override
    public String toString() {
        return "CollectionBean [arr=" + Arrays.toString(arr) + ", list=" + list + ", map=" + map + ", prop=" + prop
                + "]";
    }
    
    
}
复制代码

Spring结合Struts2的使用

1、需要导包:spring-web-4.2.4.RELEASE.jar

2、在Action中获得容器中的Service对象

  • web.xml中配置容器随项目启动
  <!-- 可以让spring容器随项目的启动而创建,随项目的关闭而销毁 -->
  <listener>
      <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
  • 在Action中获得容器
复制代码
//获得spring容器=>从Application域获得即可

//1 获得servletContext对象
ServletContext sc = ServletActionContext.getServletContext();
//2.从Sc中获得ac容器
WebApplicationContext ac = WebApplicationContextUtils.getWebApplicationContext(sc);
//3.从容器中获得CustomerService
CustomerService cs = (CustomerService) ac.getBean("customerService");

原文链接:https://www.cnblogs.com/2019lgg/p/11177500.html
如有疑问请与原作者联系

标签:

版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有

上一篇:linux maven 安装与配置

下一篇:多线程之创建线程有哪几种方式?