Java Spring MVC入门(六)依赖注入之实践部分

  海军情结                                            

  最近在玩一款海战游戏——空中网出的《战舰世界》,游戏内可以控制一艘军舰参加海战,可以凭借主炮、鱼雷击沉“敌方”军舰,虽然是游戏,游戏内的军舰都是历史上曾经服役过的,排水量、造型、大小都很历史上一模一样。里面有德国著名战舰“俾斯麦”号(击沉英国皇家海军旗舰“胡德”号),美国的“衣阿华” (日本投降签字就在衣阿华级三号舰“密苏里”号上),还有世界上最大的战舰——日本的“大和”号。在这款游戏中可以让你生临其境体验海战的精彩与残酷。

    除了这款游戏之外空中网还有同系列的两款游戏——《坦克世界》和《战机世界》,分别是陆战和空战游戏,那么为何偏偏玩海战呢? 这里我说下我对海军的一点情节。

  海军是什么? 文明和野蛮的交织地,说他文明是因为海军需要最高精尖的技术和资金的投入,没有技术和钱就别想打造海军,所以海军的军官一般都是由贵族子弟担任。说他野蛮是因为海战是残酷的,轻则战舰爆炸着火,重则沉没。许多军官坚守与军舰共存亡的精神,战败后与军舰一起沉没决不偷生,从这点看来海军是高贵的。

     海军是什么? 是国家命运的守护者,每一场海战的胜败都将决定一国的命运,一旦战败,国家的大门将被敲开,命运由此改变,甲午战争就是这样。

     希望中国海军能够强大,保家卫国。

  依赖注入的种类                                    

  上次说到依赖注入的理论部分,那么具体代码该怎么写呢?下面我们将一一介绍,依赖注入有好几种实现方法,分别是Set注入、构造函数注入和注解注入。

      先简单的介绍下三种注入方式:

    Set注入:Spring XML文件里声明依赖关系,代码里必须有set方法设置依赖注入

    构造函数注入:Spring XML文件里声明依赖关系,代码里必须有构造方法设置依赖注入

    注解注入:无须申明依赖关系,代码里使用注解即可自动注入。

  

  Set注入                                            

1、首先Spring的XML文件中声明依赖关系

 <bean id = "IocDemoOne" class="com.game.demo.scofi.IocDemoOne">
        <property name="axe" ref="axe"></property>
 </bean>
 <bean id ="axe" class="com.game.demo.scofi.Axe"></bean>

其中name表示代码中set方法的名称,Spring会取出name的值这里是axe,之后把首字母大写,再加上字符串“set” , 最后字符串为"setAxe",以此为方法名,系统会调用代码中的这个方法,当然代码里必须要有此方法。

ref表示需要注入的bean对象ID ,会把此对象作为setAxe的参数,传入到调用者。

2、编写java代码

IocDemoOne

package com.game.demo.scofi;

import java.io.IOException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;


/**
 * 通过XML文件配置依赖注入
 * SET方法注入demo
 * @author wangyongguo
 *
 */
@Controller
public class IocDemoOne {
    private Axe myAxe;
    //通过@RequestMapping为控制器指定哪些需要的请求
    @RequestMapping("/iocDemoOne")
    public void index(HttpServletRequest request, HttpServletResponse response) throws IOException {
            
        response.setHeader("content-type", "text/html;charset=UTF-8");
        String c = myAxe.useAxe();
        response.getWriter().append("I am Demo one!");
        response.getWriter().append(c);
    }
    
    public void setAxe(Axe axe){
        this.myAxe = axe;
    }
}

 Axe.java

package com.game.demo.scofi;
public class Axe{
    
    public String useAxe(){
        return "I am use AXE";
    }
}

setAxe方法中把系统注入进来axe对象 赋值给myAxe这样调用者就获得了被依赖对象axe ,之后就可以使用他的方法了。

最后执行结果:

I am Demo one!I am use AXE

  构造函数注入                                          

1、首先Spring的XML文件中声明依赖关系

<bean id ="axe" class="com.game.demo.scofi.Axe"></bean>
    
    <bean id = "IocDemoTwo" class="com.game.demo.scofi.IocDemoTwo">
        <constructor-arg ref="axe" />
    </bean>
constructor-arg 标签表示构造函数注入,ref值的含义和set注入一样,会把axe对象作为构造函数的参数注入到调用者对象内。

2、编写代码

IocDemoTwo.java

package com.game.demo.scofi;

import java.io.IOException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;


/**
 * 通过XML文件配置依赖注入
 * 构造器注入DEMO
 * @author wangyongguo
 *
 */
@Controller
public class IocDemoTwo {
    
    private Axe myAxe;
    //构造函数注入
    public IocDemoTwo(Axe axe){
        this.myAxe = axe;
    }
    
    //通过@RequestMapping为控制器指定哪些需要的请求
    @RequestMapping("/iocDemoTwo")
    public void index(HttpServletRequest request, HttpServletResponse response) throws IOException {
            
        response.setHeader("content-type", "text/html;charset=UTF-8");
        String c = myAxe.useAxe();
        response.getWriter().append("I am Demo two!");
        response.getWriter().append(c);
    }

}

Axe.java 同上

运行结果:

I am Demo two!I am use AXE

  注解注入                                         

上面两种方法都要在java代码里写方法来实现依赖注入,这样非常麻烦,为了解决这个问题,注解注入应用而生,不再需要写方法来实现,而是通过注解即可。

具体看代码 IocDemoThree.java

代码位置:

package com.game.demo.iocTwo;

import java.io.IOException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

/**
 * 使用@Autowired注解实现依赖注入
 * spring配置文件里必须包含 <bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" />
 * spring才能理解@Autowired,否则将不会生效,或者使用 <context:component-scan base-package="xxx" /> 会自动扫描所有注解
 * 
 * @author wangyongguo
 *
 */
@Controller
public class IocDemoThree {
    //如果Spring找不到Axe类型的bean,会报出异常。
    //如果不想报出异常就需要加上(required = false)
    //@Autowired(required = false) 
    @Autowired
    private Axe myAxe;
    
    @RequestMapping("/iocDemoThree")
    public void index(HttpServletRequest request, HttpServletResponse response) throws IOException {
        
        response.setHeader("content-type", "text/html;charset=UTF-8");
        String c = myAxe.useAxe();
        response.getWriter().append("I am Demo Three 11!");
        response.getWriter().append(c);
    }
}

 对应的配置文件代码:

<context:component-scan base-package="com.game.demo.iocTwo" >
    </context:component-scan>

接口注入: 

上面介绍的是注入简单的Java类,如果有一个接口,有多个实现类,这样就会存在有同一个类型多个Bean的情况存在,那么这种情况该如何处理?

假设我们有一个Car接口 ,有两个类Benz和BMW 实现这个接口

Car.java

package com.game.demo.iocThree;

public interface Car {
    
    public String getCarName();
}

Benz.java

package com.game.demo.iocThree;

import org.springframework.stereotype.Service;

@Service
public class Benz implements Car {
    
    public String getCarName(){
        return "I am Benz GLK";
    }
}

BMW.java

package com.game.demo.iocThree;

import org.springframework.stereotype.Service;

@Service
public class Bmw implements  Car{
    
    public String getCarName(){
        return "I am BMW 740";
    }
}

IocDemoFour.java

package com.game.demo.iocThree;

import java.io.IOException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

/**
 * @Autowired 接口注入
 * 如果@Autowired 属性的类型是接口,这个接口被多个类实现,就会有一个问题,到底改注入哪个类呢?
 * @author wangyongguo
 *
 */
@Controller
public class IocDemoFour {
    
    //必须加上import org.springframework.beans.factory.annotation.Qualifier;
    //@Qualifier注解括号里面的应当是Car接口实现类的类名,但必须小写,否则出错
    @Autowired
    @Qualifier("bmw")
    private Car myCar;
    
    @RequestMapping("/iocDemoFour")
    public void index(HttpServletRequest request, HttpServletResponse response) throws IOException {
        response.setHeader("content-type", "text/html;charset=UTF-8");
        String c = myCar.getCarName();
        response.getWriter().append("I am Demo Four!");
        response.getWriter().append(c);
    }
}

根据上面代码我们可以看到,当存在同一类型有多个bean情况存在时,我们使用 @Qualifier 注解,qualifier的意思是合格者,通过这个标示,表明了哪个实现类才是我们所需要的,我们修改调用代码,添加@Qualifier注解,括号里面的内容是需要注入小写类名。

 

posted @ 2017-09-14 18:30  scofi  阅读(4067)  评论(2编辑  收藏  举报