大型项目架构搭建_高内聚、低耦合(1)

软件开发中的项目的架构的终极目标就六个字:

高内聚、低耦合;

架构搭建好了,后期可维护性才能提高,在这基础上,如果还想提高项目的高效性,就只能从算法着手

如果项目中算法都已经做到极致了,还想提高项目的高效性,怎么办?(算法并不一定能处理高并发的问题)

就要用到中间件的技术了

下面我就从这三方面来搭建一个项目

假设我们有个这么需求 我公式和YC公式合作,开发一个项目,这个项目功能是用来测量东西的一个项目,我们公式在存储数据方面做得好,YC公司所以要和我们合作,要我们做一个对数据进行保存的容器,他们公司直接用我公司的容器来保存测量的数据;

对于我公司来说,我们只要管好数据保存方面就好了,对于怎么测量,我们管不着,因为测量的范围和方式太广了,我们公司是做通用平台的,所以我公司只要提供一个测量接口就行了,具体怎么实现,由买我公司项目的YC公司去具体实现。

下面来看具体案例

首先是我公司的容器类,用来保存测量数据的

package com.my;

/***
 * 容器类,用来存测量数据的,并且可以对数据进行一系列的操作
 *
 * @author 汤高
 *
 */
public class Container {

    private Object maxobj;// 最大的测量对象
    private Object minobj;// 最小的测量对象

    private double sum;// 测量数据的和
    private double avg;// 平均值

    private Measuraable measuraable;// 测量工具
    //private Filter filter;

    private Object[] objects;// 用来存放数据的集合

    private int size;// 数组的实际容量

    public static final int NUM = 10;// 数组的初始化容量

    /**
     * 构造函数初始化各属性值
     */
    public Container() {
        this.maxobj = null;
        this.minobj = null;
        this.sum = 0;
        this.avg = 0;
        this.measuraable = null;
        this.size = 0;
        objects = new Object[NUM];
    }

    public double getAvg() {
        return avg;
    }

    public Object getMaxobj() {
        return maxobj;
    }

    public Object getMinobj() {
        return minobj;
    }

    public int getSize() {
        return size;
    }
    //设置测量容器
    public void setMeasuraable(Measuraable measuraable) {
        this.measuraable = measuraable;
    }

    public void add(Object obj) throws Exception {
        // 优雅的设计 第一步 容错处理
        if (obj == null) {
            throw new RuntimeException("要测量的对象不能为空,您传入的对象为:" + obj);// 非受检异常
        }
        if (this.measuraable == null) {
            throw new Exception(" 测量工具不能为空");// 受检异常 要么抛出 要么捕获
        }

        double value = this.measuraable.Measure(obj);
        if (size == 0) { // 刚开始测第一个对象 把最大值最小值都付给第一个对象
            maxobj = obj;
            minobj = obj;
        } else {
            double maxvalue = this.measuraable.Measure(maxobj);
            double minvalue = this.measuraable.Measure(minobj);

            if (value > maxvalue) {
                maxobj = obj;
            }
            if (value < minvalue) {
                minobj = obj;
            }
        }

        // 拓容处理
        enlargeArray();
        objects[size] = obj;// 把测量对象存入数组
        size++;// 实际大小加1
        sum += sum;// 求和
        avg = sum / size;// 计算平均值

    }

    public void enlargeArray() {
        int oldLength = objects.length;
        if (size > objects.length) {
            // 创建新数组 ,长度为原数组的两倍
            Object[] newobjects = new Object[objects.length * 2];
            // 将原数组的数据存到新数组中
            System.arraycopy(objects, 0, newobjects, 0, size);
            // 将newobjects的地址赋值给 objects
            objects = newobjects;
            System.gc();// gc()垃圾回收. 高级->重写虚拟机
        }

    }
    /**
     * 返回所有的测量过的数据 objects默认10个数据,但是不一定存满,只能返回有效数据
     * @return
     */
    public Object[] getAllData(){
        Object[] newobject = new Object[size];
        System.arraycopy(objects, 0, newobject, 0, size);
        return newobject;
    }

    //public void setFilter(Filter filter) {
    //  this.filter=filter;

    //}

}

上面就是我公司的核心功能了,就是用来保存数据
对于如何测,我公司只提供一个接口

package com.my;
/***
 * 测量工具的接口
 * @author 汤高
 *
 */
public interface Measuraable {
    /***
     * 测量的方法,提供模板,具体怎么实现 我不管,交给要测量的公司来具体实现
     * @param obj 测量对象
     * @return 返回一个对象的测量结果
     */
    public double Measure(Object obj);

}

具体怎么测试,怎么去实现,由买我们公司的YC去实现

YC公司是用来测BMI值的(人的健康指数)

package com.yc;

import javax.management.RuntimeErrorException;

import com.my.Measuraable;

public class Bmi implements Measuraable {
    @Override  //bmi=weight/(height*height)
    public double Measure(Object object) {
        if(object==null){
            throw new RuntimeException("要测量的对象不能为空");
        }
        if(  !(object instanceof Person ) ){
            throw new RuntimeException("要测量的对象必须是一个Person");
        }
        //将Object 强制类型转换为Person 以取出 height,weight
        Person p=(Person)object;
        double height=p.getHeight();
        double weight=p.getWeight();

        return weight/(height*height);
    }

}

好了,YC公司已经自己做了测量的实现,所以,我们两公司的合作就暂时告一段落了
看测试类;

package test1;

import org.junit.Test;

import com.my.Container;
import com.my.Filter;
import com.yc.Bmi;
import com.yc.BmiDataFilter;
import com.yc.Person;

public class TestVersion1 {

    @Test
    public void test() throws Exception {
        Bmi bmi = new Bmi();

        Filter filter = new BmiDataFilter();
        Container c = new Container();

        c.setMeasuraable(bmi);// 將bmi加载到 container容器
    //  c.setFilter(filter);

        Person p1 = new Person("张三", 70, 2.23);// bmi 最小
        Person p2 = new Person("李四", 250, 1.3);// bmi 最大
        Person p3 = new Person("王五", 100, 1.9);

        Person p4  = new Person("赵六", 500, 2);// 不合理

        Person p5 = new Person("田七", 200, 1.7);

        // 存
        c.add(p1);
        c.add(p2);
        c.add(p3);
        try {
            c.add(p4);
        } catch (Exception e) {
            e.printStackTrace();
        }
        c.add(p5);

        // 取出最大值
        Object max = c.getMaxobj();
        Person maxPerson = (Person) max;

        // 取出最小值
        Object min = c.getMinobj();
        Person minPerson = (Person) min;

        System.out.println("最大值" + maxPerson);
        System.out.println("最小值" + minPerson);
        // 取出所有的值
        System.out.println("==========所有的值============");
        Object[] objs = c.getAllData();
        for (Object o : objs) {
            Person p = (Person) o;
            System.out.println(p);
        }
    }

}

大家看看结果:
最大值Person [name=李四, weight=250.0, height=1.3]
最小值Person [name=张三, weight=70.0, height=2.23]
==========所有的值============
Person [name=张三, weight=70.0, height=2.23]
Person [name=李四, weight=250.0, height=1.3]
Person [name=王五, weight=100.0, height=1.9]
Person [name=赵六, weight=500.0, height=2.0]
Person [name=田七, weight=200.0, height=1.7]

竟然有重量为500斤的,明显不合要求啊!数据不合理啊,所以YC公司就来找我们公司麻烦,要求我们公司把这个问题解决,必须要我们队数据进行过
滤,我们公司老板就不干了
啊,我们是管数据储存的,你竟然还要我们管过滤,那不行,我公司是做通用平台的,做不了具体的过滤实现,你们公司是测人,但是假如别的公司测猪的重量了!
重量肯定是越种越好啊,所以,我们只做过滤数据的接口,具体怎么实现,你们自己去实现,因为我们只做平台。

所以我们公司提供了过滤接口

package com.my;
//过滤数据的接口
public interface Filter {
    public boolean filter(Object object);
}

YC公司听了我们公司老板的话后,觉得是他们不对,具体实现数据过滤应该他们自己做

package com.yc;

import com.my.Filter;

public class BmiDataFilter implements Filter {

    @Override
    public boolean filter(Object object) {
        if (object == null) {
            throw new RuntimeException("要过滤的对象不能为空");
        }
        if (!(object instanceof Person)) {
            throw new RuntimeException("要过滤的对象必须是一个Person");
        }

        // 将Object 强制类型转换为Person 以取出 height,weight
        Person p = (Person) object;
        double height = p.getHeight();
        double weight = p.getWeight();

        if(height<1||height>2.5){
            //throw new RuntimeException("身高数据不合理"+height);
            return false;
        }
        if(weight<40||weight>400){
            return false;
        }
        return true;
    }

}

好了,过滤数据的功能做好了,下面再看看测试,
首先先在容器类里面添加一个过滤器

package com.my;

/***
 * 容器类,用来存测量数据的,并且可以对数据进行一系列的操作
 *
 * @author 汤高
 *
 */
public class Container {

    private Object maxobj;// 最大的测量对象
    private Object minobj;// 最小的测量对象

    private double sum;// 测量数据的和
    private double avg;// 平均值

    private Measuraable measuraable;// 测量工具
    private Filter filter;

    private Object[] objects;// 用来存放数据的集合

    private int size;// 数组的实际容量

    public static final int NUM = 10;// 数组的初始化容量

    /**
     * 构造函数初始化各属性值
     */
    public Container() {
        this.maxobj = null;
        this.minobj = null;
        this.sum = 0;
        this.avg = 0;
        this.measuraable = null;
        this.size = 0;
        objects = new Object[NUM];
    }

    // 添加对数据过滤的过滤器
    public void setFilter(Filter filter) {
        this.filter = filter;

    }

    public double getAvg() {
        return avg;
    }

    public Object getMaxobj() {
        return maxobj;
    }

    public Object getMinobj() {
        return minobj;
    }

    public int getSize() {
        return size;
    }

    // 设置测量容器
    public void setMeasuraable(Measuraable measuraable) {
        this.measuraable = measuraable;
    }

    public void add(Object obj) throws Exception {
        // 优雅的设计 第一步 容错处理
        if (obj == null) {
            throw new RuntimeException("要测量的对象不能为空,您传入的对象为:" + obj);// 非受检异常
        }
        if (this.measuraable == null) {
            throw new Exception(" 测量工具不能为空");// 受检异常 要么抛出 要么捕获
        }

        // 判断是否有过滤器,如果有,则调用过滤器的过滤功能进行过滤
        if (this.filter != null) {
            if (this.filter.filter(obj) == false) {
                throw new RuntimeException("要过滤的对象不是有效对象,数据错误,请检查" + obj);
            }
        }

        double value = this.measuraable.Measure(obj);
        if (size == 0) { // 刚开始测第一个对象 把最大值最小值都付给第一个对象
            maxobj = obj;
            minobj = obj;
        } else {
            double maxvalue = this.measuraable.Measure(maxobj);
            double minvalue = this.measuraable.Measure(minobj);

            if (value > maxvalue) {
                maxobj = obj;
            }
            if (value < minvalue) {
                minobj = obj;
            }
        }

        // 拓容处理
        enlargeArray();
        objects[size] = obj;// 把测量对象存入数组
        size++;// 实际大小加1
        sum += sum;// 求和
        avg = sum / size;// 计算平均值

    }

    public void enlargeArray() {
        int oldLength = objects.length;
        if (size > objects.length) {
            // 创建新数组 ,长度为原数组的两倍
            Object[] newobjects = new Object[objects.length * 2];
            // 将原数组的数据存到新数组中
            System.arraycopy(objects, 0, newobjects, 0, size);
            // 将newobjects的地址赋值给 objects
            objects = newobjects;
            System.gc();// gc()垃圾回收. 高级->重写虚拟机
        }

    }

    /**
     * 返回所有的测量过的数据 objects默认10个数据,但是不一定存满,只能返回有效数据
     *
     * @return
     */
    public Object[] getAllData() {
        Object[] newobject = new Object[size];
        System.arraycopy(objects, 0, newobject, 0, size);
        return newobject;
    }

}

下面再测试

package test1;

import org.junit.Test;

import com.my.Container;
import com.my.Filter;
import com.yc.Bmi;
import com.yc.BmiDataFilter;
import com.yc.Person;

public class TestVersion1 {

    @Test
    public void test() throws Exception {
        Bmi bmi = new Bmi();

        Filter filter = new BmiDataFilter();
        Container c = new Container();

        c.setMeasuraable(bmi);// 將bmi加载到 container容器
        c.setFilter(filter);

        Person p1 = new Person("张三", 70, 2.23);// bmi 最小
        Person p2 = new Person("李四", 250, 1.3);// bmi 最大
        Person p3 = new Person("王五", 100, 1.9);

        Person p4  = new Person("赵六", 500, 2);// 不合理

        Person p5 = new Person("田七", 200, 1.7);

        // 存
        c.add(p1);
        c.add(p2);
        c.add(p3);
        try {
            c.add(p4);
        } catch (Exception e) {
            e.printStackTrace();
        }
        c.add(p5);

        // 取出最大值
        Object max = c.getMaxobj();
        Person maxPerson = (Person) max;

        // 取出最小值
        Object min = c.getMinobj();
        Person minPerson = (Person) min;

        System.out.println("最大值" + maxPerson);
        System.out.println("最小值" + minPerson);
        // 取出所有的值
        System.out.println("==========所有的值============");
        Object[] objs = c.getAllData();
        for (Object o : objs) {
            Person p = (Person) o;
            System.out.println(p);
        }
    }

}

大家看看结果
java.lang.RuntimeException: 要过滤的对象不是有效对象,数据错误,请检查Person [name=赵六, weight=500.0, height=2.0]
at com.my.Container.add(Container.java:78)
at test1.TestVersion1.test(TestVersion1.java:36)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at org.junit.runners.model.FrameworkMethod1.runReflectiveCall(FrameworkMethod.java:50)atorg.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)atorg.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)atorg.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)atorg.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)atorg.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)atorg.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)atorg.junit.runners.ParentRunner3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner1.schedule(ParentRunner.java:71)atorg.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)atorg.junit.runners.ParentRunner.access000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:86)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:675)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)
最大值Person [name=李四, weight=250.0, height=1.3]
最小值Person [name=张三, weight=70.0, height=2.23]
==========所有的值============
Person [name=张三, weight=70.0, height=2.23]
Person [name=李四, weight=250.0, height=1.3]
Person [name=王五, weight=100.0, height=1.9]
Person [name=田七, weight=200.0, height=1.7]

数据为500的就被过滤掉了

从上面这个案例就可以看出,做项目架构的重要性了,真正实现了高内聚、低耦合,大家可以仔细去体会体会,欢迎大家提出问题,大家一起进步!

下面是我项目的源代码,大家可以下载

后面我还会更新 怎样用算法来提高项目的效率。

时间: 2024-08-25 01:11:40

大型项目架构搭建_高内聚、低耦合(1)的相关文章

框架节构设计(高内聚低耦合)

内聚是从功能角度来度量模块内的联系,一个好的内聚模块应当恰好做一件事.它描述的是模块内的功能联系: 耦合是软件结构中各模块之间相互连接的一种度量,耦合强弱取决于模块间接口的复杂程度.进入或访问一个模块的点以及通过接口的数据. 定义 高内聚低耦合,是软件工程中的概念,是判断设计好坏的标准,主要是面向对象的设计,主要是看类的内聚性是否高,耦合度是否低.  概念 内聚就是一个模块内各个元素彼此结合的紧密程度,高内聚就是一个模块内各个元素彼此结合的紧密程度高. 所谓高内聚是指一个软件模块是由相关性很强的

连载:面向对象葵花宝典:思想、技巧与实践(29) - 高内聚低耦合

高内聚低耦合,可以说是每个程序猿,甚至是编过程序,或者仅仅只是在大学里面学过计算机,都知道的一个简单的设计原则. 虽然如此流行和人所众知,但其实真正理解的人并不多,很多时候都是人云亦云. =============================================================== 要想真正理解"高内聚低耦合",需要回答两个问题: 1)为什么要高内聚低耦合? 2)高内聚低耦合是否意味内聚越高越好,耦合越低越好?   第一个问题:为什么要高内聚低耦合? 经

年终知识分享——大型项目架构

                                    关于这两个page的内容,请看我的博客:http://www.cnblogs.com/liulun/p/3625941.html                                                                   这个博文参考了很多资料,就不在一一例举了   陀螺   在田野上转 在清风里转 在飘着香的鲜花上转 在沉默里转 在孤独里转 在结着冰的湖面上转 在欢笑里转 在泪水里转

对高内聚,低耦合的理解

内聚:一个模块内各个元素彼此结合的紧密程度 耦合:一个软件结构内不同模块之间互连程度的度量     (一)这是判断设计好坏的标准,主要是面向OO的设计,主要是看类的内聚性是否高,偶合度是否低.         高内聚:类与类之间的关系而定.高,意思是他们之间的关系要简单,明了,不要有很强的关系,不然,运行起来就会出问题.一个类的运行影响到其他的类.     低耦合:类内部的方法而言.把程序的功能尽量分散,别在一个类里只写一个或很好的方法,因为那样会给你的调试等带来很多问题.出了错你都不知道在什么

二、大型网站架构模式

网站架构模式     分层 横向维度,每个部分负责单一的职责.上层对下层依赖和调用. 应用层,服务层和数据层     分割 网站越大,不同功能和服务分割出来形成高内聚低耦合模块单元.     分布式 分布式部署,即将不同模块部署在不同的服务器上,通过远程调用协同工作. 数据在分布式环境中保持数据一致性非常难,分布式难以保证. 分布式应用和服务.分布是静态资源.分布式数据和存储.分布式计算.并发和协同的分布式锁. Zookeeper 的典型的应用场景(配置文件的管理.集群管理.同步锁.Leader

大型网站架构不得不考虑的几点问题

前言:这两天机器坏了,正在送修中,写个系列的大型http://www.aliyun.com/zixun/aggregation/11116.html">网站架构的文章,希望对有志在互联网做出一番事业的站长朋友们一些帮助. 注意:这里的大型网站架构只包括高互动性高交互性的数据型大型网站,基于大家众所周知的原因,我们就不谈新闻类和一些依靠HTML静态化就可以实现的架构了,我们以高负载高数据交换高数据流动性的网站为例,比如海内,开心网等类似的web2.0系列架构.我们这里不讨论是PHP还是JSP

作为大型网站架构必须考虑的十大问题

这里的大型网站架构只包括高互动性高交互性的数据型大型网站,基于大家众所周知的原因,我们就不谈新闻类和一些依靠HTML静态化就可以实现的架构了,我们以高负载高数据交换高数据流动性的网站为例 比如海内,开心网等类似的web2.0系列架构.我们这里不讨论是PHP还是JSP或者.NET环境,我们从架构的方面去看问题,实现语言方面并不是问题,语言的优势在于实现而不是好坏,不论你选择任何语言,架构都是必须要面对的. 这里讨论一下大型网站需要注意和考虑的问题 1.海量数据的处理 众所周知,对于一些相对小的站点

大型网站架构不得不考虑的10个问题,互联网营销

这里的大型网站架构只包括高互动性高交互性的数据型大型网站,基于大家众所周知的原因,我们就不谈新闻类和一些依靠HTML静态化就可以实现的架构了,我们以高负载高数据交换高数据流动性的网站为例,比如海内,开心网等类似的web2.0系列架构.我们这里不讨论是PHP还是JSP或者.NET环境,我们从架构的方面去看问题,实现语言方面并不是问题,语言的优势在于实现而不是好坏,不论你选择任何语言,架构都是必须要面对的. 这里讨论一下大型网站需要注意和考虑的问题. 1.海量数据的处理 众所周知,对于一些相对小的站

[讨论]请用一个词来形容 高内聚,低耦合

问题描述 如果说原来没有采用低耦合的代码可以形容为[牵一发而动全身],多么的不好,那么高内聚该如何用一词形象的形容 解决方案 解决方案二:哎解决方案三:藕断丝连解决方案四:耦合到底是什么意思啊???解决方案五:高内聚,本人认为就是代码简洁...