JavaWeb学习-MVC


 一、MVC概念

1. MVC : Model(模型)、View(视图)、Controller(控制器)

视图层:用于做数据展示以及和用户交互的一个界面
控制层:能够接受客户端的请求,具体的业务功能还是需要借助于模型组件来完成
模型层:模型分为很多种:有比较简单的pojo/vo(value object),有业务模型组件,有数据访问层组件。

 1)pojo/vo : 值对象模型                 2)DAO : 数据访问对象模型
 3)BO : 业务对象模型                   4)DTO:数据传输对象模型

2. 区分业务对象和数据访问对象

  1. DAO中的方法都是单精度方法或者称之为细粒度方法。什么叫单精度?一个方法只考虑一个操作,比如添加,那就是insert操作、查询那就是select操作…
  2. BO中的方法属于业务方法或者称之为粗粒度方法,而实际的业务是比较复杂的,因此业务方法的粒度是比较粗的。

例如:

注册这个功能属于业务功能,也就是说注册这个方法属于业务方法。
那么这个业务方法中包含了多个DAO方法。也就是说注册这个业务功能需要通过多个DAO方法的组合调用,从而完成注册功能的开发。
注册举例:

  1)检查用户名是否已经被注册 – DAO中的select操作
  2)向用户表新增一条新用户记录 – DAO中的insert操作
  3)向用户积分表新增一条记录(新用户默认初始化积分100分) – DAO中的insert操作
  4)向系统消息表新增一条记录(某某某新用户注册了,需要根据通讯录信息向他的联系人推送消息) – DAO中的insert操作
  5)向系统日志表新增一条记录(某用户在某IP在某年某月某日某时某分某秒某毫秒注册) – DAO中的insert操作
  6)…

  3. 在系统中添加业务层组件。(FruitService 调用FruitDao)图示:

JavaWeb学习-MVC

public class FruitServiceImpl implements FruitService {
    private FruitDAO fruitDAO = new FruitDAOImpl();
    @Override
    public List<Fruit> getFruitList() {
        return fruitDAO.getFruitList();
    }

    @Override
    public Fruit getFruitById(int fid) {
        return fruitDAO.getFruitById(fid);
    }
}
public class FruitController {
    private FruitService fruitService = new FruitServiceImpl();
    //调用改为 fruitService.XXX
}

二、MVC-IOC

1. 耦合/依赖:

  在软件系统中,层与层之间是存在依赖的。我们也称之为耦合。

      我们系统架构或者是设计的一个原则是: 高内聚低耦合。层内部的组成应该是高度聚合的,而层与层之间的关系应该是低耦合的,最理想的情况0耦合(就是没有耦合)

实现下层改动不会影响上层(比如:删掉serviceImpl不会出错)

代码实现

1)配置文件

<beans>
    <!--1. 先描述需要什么组件
        2. 再描述组件与组件之间的依赖关系-->
    <bean id ="fruitDAO" class="com.fruits.dao.Impl.FruitDAOImpl">

    </bean>
    <bean id ="fruitService" class="com.fruits.ser.impl.FruitServiceImpl">
        <!-- property 标签用来表示属性:name表示属性名,ref表示引用其他bean的id值 -->
        <!-- FruitServiceImpl需要DAO: private FruitDAO fruitDAO = new FruitDAOImpl();
        所以  name 就是FruitServiceImpl中需要的属性名  而 ref 就引用到 某个bean的id:fruitDAO
        -->
        <property name="fruitDAO" ref="fruitDAO"></property>
    </bean>
    <bean id="fruit" class="com.fruits.controller.FruitController">
        <property name="fruitService" ref="fruitService"></property>
    </bean>
</beans>

2)创建一个Bean工厂,用来实现根据bean中的id获得class对象

package com.fruits.io;

public interface BeanFactory {
    Object getBean(String id); //根据bean中的id 获得class对象
}

实现类

public class ClassPathXmlApplicationContext implements BeanFactory{
    private Map<String,Object> beanMap = new HashMap<>();

    public ClassPathXmlApplicationContext() {
        try {
            InputStream inputStream = getClass().getClassLoader().getResourceAsStream("applicationContext.xml");
            //1.创建DocumentBuilderFactory
            DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
            //2.创建DocumentBuilder对象
            DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
            //3.创建Document对象
            Document document = documentBuilder.parse(inputStream);

            //4.获取所有的bean节点
            NodeList beanNodeList = document.getElementsByTagName("bean");
            for (int i = 0; i < beanNodeList.getLength(); i++) {
                Node beanNode = beanNodeList.item(i);
                if (beanNode.getNodeType() == Node.ELEMENT_NODE) {
                    Element beanElement = (Element) beanNode;
                    String beanId = beanElement.getAttribute("id");
                    String className = beanElement.getAttribute("class");
                    Class beanClass = Class.forName(className);
                    //创建bean实例
                    Object beanObj = beanClass.newInstance();
                    //将bean实例对象保存再map容器中
                    beanMap.put(beanId, beanObj);
                }
            }
            //5. 组装bean之间的依赖关系
            for (int i = 0; i < beanNodeList.getLength(); i++) {
                Node beanNode = beanNodeList.item(i);
                if (beanNode.getNodeType() == Node.ELEMENT_NODE) {
                    Element beanElement = (Element) beanNode;
                    String beanId = beanElement.getAttribute("id"); //fruitService
                    NodeList beanChildNodesList = beanElement.getChildNodes();
                    for (int j = 0; j < beanChildNodesList.getLength(); j++) {
                        Node beanChildNode  = beanChildNodesList.item(j);
                        if(beanChildNode.getNodeType()==Node.ELEMENT_NODE && "property".equals(beanChildNode.getNodeName())){
                            Element propertyElement = (Element)beanChildNode;
                            String propertyName = propertyElement.getAttribute("name");  //  propertyName=fruitDAO
                            String propertyRef = propertyElement.getAttribute("ref");  // propertyRef=fruitDAO
                            //1)找到propertyRef对应的实例
                            Object refObj = beanMap.get(propertyRef);  //refObj = com.fruits.dao.Impl.FruitDAOImpl
                            //2)将refObj设置到当前bean对应的实例的property属性上去
                            Object beanObj = beanMap.get(beanId);  // beanObj = com.fruits.ser.impl.FruitServiceImpl
                            Class beanClass = beanObj.getClass(); // beanClass = FruitServiceImpl
                            Field propertyField = beanClass.getDeclaredField(propertyName); //获取beanClass 中与propertyName有关的属性 propertyField =  private FruitDAO fruitDAO
                            propertyField.setAccessible(true);
                            propertyField.set(beanObj,refObj);  //将refObj设置到beanObj对应的propertyField中去
                        }                          //refObj = FruitDAOImpl  beanObj =  FruitServiceImpl  propertyField = fruit
                    }
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (ParserConfigurationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (SAXException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        }
    }
    @Override
    public Object getBean(String id) {
        return beanMap.get(id);
    }
}

3) FruitServiceImpl

public class FruitServiceImpl implements FruitService {
    private FruitDAO fruitDAO = null;
    @Override
    public List<Fruit> getFruitList() {
        return fruitDAO.getFruitList();
    }
}

4)FruitController

public class FruitController {
    private FruitService fruitService = null;
}

2. IOC-控制反转 / DI – 依赖注入

 控制反转:

    1) 之前在Servlet中,我们创建service对象 , FruitService fruitService = new FruitServiceImpl();
       这句话如果出现在servlet中的某个方法内部,那么这个fruitService的作用域(生命周期)应该就是这个方法级别;
       如果这句话出现在servlet的类中,也就是说fruitService是一个成员变量,那么这个fruitService的作用域(生命周期)应该就是这个servlet实例级别

    2) 之后我们在applicationContext.xml中定义了这个fruitService。然后通过解析XML,产生fruitService实例,存放在beanMap中,这个beanMap在一个BeanFactory中
       因此,我们转移(改变)了之前的service实例、dao实例等等他们的生命周期。控制权从程序员转移到BeanFactory。这个现象我们称之为控制反转

    依赖注入:

    1) 之前我们在控制层出现代码:FruitService fruitService = new FruitServiceImpl();
       那么,控制层和service层存在耦合。

    2) 之后,我们将代码修改成FruitService fruitService = null ;
       然后,在配置文件中配置:
       <bean id=”fruit” class=”FruitController”>
            <property name=”fruitService” ref=”fruitService”/>
       </bean>

原创文章,作者:carmelaweatherly,如若转载,请注明出处:https://blog.ytso.com/276261.html

(0)
上一篇 2022年7月22日
下一篇 2022年7月22日

相关推荐

发表回复

登录后才能评论