前言:面向对象设计的几大原则?

1》针对接口编程,而不是针对实现编程

2》优先使用对象组合,而不是类继承

1.只根据抽象类中定义的接口来操纵对象有以下两个好处?

1)客户无须知道他们使用对象的特定类型,只需对象有客户所期望的接口

2)客户无须知道他们使用的对象是用什么类来实现的,他们只须知道定义接口的抽象类

这将极大地减少子系统实现之间的相互依赖关系,也产生了可复用的面向对象设计的如下原则:

针对接口编程,而不是针对实现编程

2.继承和组合的比较?

继承:

类继承是在编译时刻静态定义的,且可直接使用,因为程序设计语言支持类继承,可以较方便的改变被复用的实现。

1)继承在编译时刻就定义了,所以无法再运行时刻改变从父类继承的实现,并且父类通常至少定义了部分子类的具体表示。因为继承对子类揭示了其父类的实现细节,常被认为

“破坏了封装性”,子类中的实现与它的父类有如此紧密的依赖关系,以至于父类实现中的任何变化必然导致子类发生变化。

2)如果继承下来的实现不适合解决新的问题,则父类必须重写或被其他更适合的类替换,这种依赖关系限制了灵活性并最终限制复用性。一个方法:只继承抽象类(提供较少的实现)

组合:

1)对象组合是通过获得对其他对象的引用而在运行时刻动态定义的。组合要求对象遵守彼此的接口约定,进而要求更仔细的定义接口,而这些接口并不会妨碍你将一个对象和其他对象

一起使用。对象只能通过接口来访问,并不破坏封装性;只要类型一致,运行时刻还可以用一个对象来替代另一个对象;更进一步,因为对象的实现是基于接口编写的,所以很少依赖

2)优先使用对象组合有助于你保持每个类被封装,并被集中在单个任务上。这样的类和类继承层次会保持较小的规模,另一方面,基于对象组合的设计会有更多的对象(而有较少的类)

且系统的行为将依赖于对象间的关系而不是被定义在某个类中

导出我们的面向对象设计的第二个原则:

优先使用对象组合,而不是类继承

3.模式与框架的区分?

1)设计模式比框架更抽象

框架能够用代码表示,而设计模式只有其实例才能表示为代码。框架的威力在于它们能够使用程序设计语言写出来,它们不仅能被学习,也能被直接执行和复用

2)设计模式是比框架更小的体系结构元素

一个典型的框架包含了多个设计模式,而反之绝非如此

3)框架比设计模式更加特例化

框架总是针对一个特定的应用领域。一个图形编辑器框架可能被用于一个工厂模拟,但它不会被认错为是一个模拟框架

框架变得越来越普遍和重要。它们是面向休息系统获得最大复用的方式。较大的面对对象将会由多层彼此合作的框架组成。应用的大部分设计和代码将来自于他所使用的框架

第二章Java程序设计中最基本的设计模式(23种)

设计模式有什么:描述一个设计模式,通常包含如下四个部分:

模式名称:就是为每一个设计模式取个名字,好记忆交流

坏境和问题:描述在什么场景下,出现什么样的特定的问题

解决方案:描述如何解决问题

效果:描述模式可能带来的问题,或者使用过程中需要权衡的问题

一、单例模式

问题:采用什么方法来控制创建类实例,然后确保在任何给定的时间只创建一个类实例

方案:Singleton模式主要作用是保证在Java程序中,一个类只有一个实例存在。很多时候,比如建立目录,数据库连接都需要这样的单线程操作。

Singleton能够被状态化,这样,多个单态类在一起就可以作为一个状态仓库一样向外提供服务 例子:论坛中的帖子计数器--synchronize的安全自动加一

Singleton也能够被无状态化。好处在于可以节省内存,因为它限制了实例的个数,有利于Java垃圾回收

使用示例:一般Singleton模式通常有两种形式:

one:较安全些

public class Singleton{

private Singleton(){}

//在自己内部定义自己的一个实例,是不是很奇怪?

//注意这是private只供内部调用

private static Singleton instance = new Singleton();

//这里提供了一个供外部访问本class的静态方法,可以直接访问

public static Singleton getInsatnce(){

return instance;

}

}

two:

public class Singleton{

//第一次调用初始Singleton,以后就不用再生成了

private static Singleton instance = null;

public static synchronized Singleton getInstance(){

//这个方法比上面的有所改进,不用每次都进行生成对象,只是第一次

//使用时产生实例,提高了效率

if(instance==null)

instance = new Singleton();

return instance;

}

}

使用Singleton注意事项:(真正使用好,需要对Java的类线程内存等概念有相当的了解)

1.有时在某些情况下,使用Singleton并不能达到效果,在有多个Singleton对象同时被不同的类装入器装载;在EJB这样的分布式系统中使用也要注意这种情况,

因为EJB是跨服务器的,跨JVM的

分析说明:在EJB中,Singleton模式会失去作用,SUN公司的宠物店源码中ServiceLocator才分为两种,一种是面向WEB的,一种是面向EJB服务的

二、工厂模式---标准的创建对象的方法

问题:定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂模式使一个类的实例化延迟到其子类

方案:工厂模式是一种创建性模式,它定义了一个创建对象的接口,但是却让子类来决定具体实例化哪一个类。当一个类无法预料要创建哪一个类对象或是一个类

需要由子类来指定创建的对象时我们就需要用到工厂模式了。简单来说工厂模式可以根据不同的条件产生不同的实例(相同的类型),工厂模式把这些实例的

具体过程封装起来了,简化了客户端的应用,也改善了程序的扩展性,使得将来可以做最小的改动就可以加入新的待创建的类

为何如此常用?

因为工厂模式就相当于创建实例对象的new,我们经常要根据类class生成实例对象,如A a = new A()工厂模式也是用来创建实例对象的,所以以后new时要多个心眼

是否可以考虑使用工厂模式,虽然这样做,可能多做一些工作,但是会给你的系统带来更大的可扩展性和尽量少的修改量

案例:

我们以类Sample为例,我们要创建Sample的实例对象:Sample sample = new Sample();可是实际情况是通常我们都要在创建sample实例时做点初始化的工作,比如:

赋值,查询数据库等

首先想到的就是使用Sample的构造函数,这样生成实例就写成:Sample sample = new Sample(参数);创建实例化时所作的初始化工作不像赋值那么简单,可能是很长

的代码写入构造函数中,那么你的代码将会很难看(为何?)-----初始化工作如果是很长的一段代码,说明要做的工作很多,将很多工作装入一个方法中,

相当于将很多的鸡蛋放在一个篮子里,是很危险的,这也有悖于Java面向对象的原则,面向对象的封装和分派告诉我们,尽量将长的代码分派切割成每段,

将每段再封装起来(减少段与段之间耦合联系性),以后如果需要修改,只要更改每段,不会再发生牵一动百的事情

工厂模式中有:工厂方法(Factory Method),抽象工厂(Abstract Factory),这两个模式区别在于需要创建对象的复杂程度上

@1.工厂方法:

建立一个专门生产Sample实例的工厂:

public class Factory{

public static Sample creator(int which){

//getClass产生Sample 一般可使用动态类装载入类

if(which==1)

return new SampleA();

else if(which==2)

return new SampleB();

}

}

那么在你的程序中,如果要实例化Sample,就使用 Sample sampleA = Factory.creator(1);

这样在整个涉及到Sample的具体子类,达到封装的效果,也就减少错误修改的机会   通俗来讲:即具体的事情做得越多,越容易犯错

使用工厂方法要注意几个角色,首先你要定义产品接口,如上面的Sample,产品接口下有接口的实现类,如SampleA,其次要有一个factory类,用来生成产品Sample

进一步稍微复杂点,就是在工厂类上进行扩展,工厂类也有继承它的实现类concreateFactory了

@2.抽象工厂

如果我们创建对象的方法变得复杂了,如上面工厂方法中是创建一个对象Sample,如果我们还有新的产品接口Sample2,假设Sample和Sample2均有两个concrete类SampleA(B)

那么,我们就将上例中的Factory扩展成抽象工厂:

public abstract class Factory{

public abstract Sample creator();

public abstract Sample2 creator(String name);

}

public class SimpleFactoy extends Factory{

public Sample creator(){

....................

return new SampleA();

}

public Sample2 creator(String name){

....................

return new Sample2A();

}

}

public class BombFactory extends Factory{

public Sample creator(){

..........

return new SampleB();

}

public Sample2 creator(String name){

....................

return new Sample2B();

}

}

从上面看到两个工厂各自生产出一套Sample和Sample2,也许你会产生疑问,为什么我不可以使用两个工厂方法来分别生产Sample和Sample2?

抽象工厂还有另外一个关键要点,是因为SimpleFactory内,生产Sample和生产Sample2的方法之间有一定联系,所以才要将这两个方法捆绑在一个类中。

@三、值对象模式--成为不同层或不同模块之间数据交换的标准方法,体现的是数据的封装,也有利于对象的复用。

问题:应用程序客户端需要与企业Bean之间交换数据,程序间交换数据---基于客户需要与EJB之间大量滴交换数据的情况,具体说来,在J2EE平台中,应用程序通常将服务器端

的程序组件实现为会话bean和实体bean,而这些组件的部分方法则需要将数据返回给客户;这种情况下,通常一个用户会重复调用相关方法多次,直到它得到相关信息

注意的是:多数情况这些方法的调用目的是为了取得单一的信息,例如用户名或者用户地址等

显而易见,在J2EE平台上,这种调用基本上都是远程的,也就是说,用户多次调用相应的方法会给Web带来极大的负担,即使用户和EJB容器加载相同的JVM、OS和计算机

运行EJB程序,由于方法调用被缺省地认为是远程任务,所以这种问题依然存在

方案:值对象(value object)模式通过减少分布式通信的消息而促进数据的交换,通常这里说指的通信是在web层和EJB层之间。在一个远程调用中,一个单一值对象可以被用来

取出一系列相关数据并提供给客户。

当EJB使用值对象的时候,用户可以通过仅仅一次方法调用来取得整个对象,而不是使用多次方法调用以得到对象中每个域的数值;由于值对象是通过值传递来交送给用户

的,所以对于该值的调用或取值都是本地调用,而不是远程方法调用。需注意:这个值对象必须对应于每个属性的访问方法,或者将所有属性都设为公共的

例子:

假设某名为project的业务对象被模拟或者实现为一个实体bean,当客户端调用值对象的getProjectData()方法时,该project实体bean需要通过该值对象向客户端发送数据

public class ProjectVO implements java.io.Serializable{

private String projectId;

private String projectName;

private String managerId;

private String customerId;

private Data startDate;

private Date endDate;

private boolean started;

private boolean completed;

private boolean accepted;

................

//Value object constructors

以下就是具体的每一个get和set方法

}

@四、DAO模式

问题;目前大部分的J2EE应用程序都需要在一定程度上使用可持久的数据,而实现持久性数据的方法因应用程序不同而异,且访问不同存储格式数据的应用程序接口也有显著的差别

当程序组件,即实体bean、会话bean或servlet、jsp等需要访问数据源时,它们会使用正确的应用程序接口来得到连接并管理数据源,但这样也会造成这些组件与数据程序源

之间的依赖关系,从而使得应用程序很难从一个数据存储实体移植到另一个数据存储实体中去;当数据源的物理实现变化的时候,应用程序也相应地加以改变

如何对存储层以外的模块屏蔽这些复杂性,以提供统一的调用存储实现?

方案:

(1)业务对象:表示数据的用户,它需要对于数据的访问,一个业务对象可以用会话bean、实体bean或是其他Java程序来实现

(2)数据访问对象:是这种模式的主题,它提供了底层数据访问的对象,并将其提供给业务对象以使得后者能够透明地访问数据源;同时业务对象也将数据的加载和存储

操作移交给数据访问对象处理。

(3)数据源:这里指的是数据源的物理实现,这个数据源可以是一个数据库,包括关系型数据库、面向对象数据库或文件系统

(4)值对象:指的是数据载体,数据访问对象可以使用值对象来向用户返回数据,而数据访问对象同样可以从用户那里得到值对象来对数据源中的数据进行更新

应用实例:

抽象的DAOFactory类

public abstract class DAOFactory{

public static final int CLOUDSCAPE=1;

public static final int ORACLE = 2;

public static final int SYBASE = 3;

public abstract CustomerDAO getCustomerDAO();

public abstract AccountDAO getAccountAO();

public abstract OrderDAO getOrderDAO();

....................

public static DAOFactory getDAOFactory(int whichFactory){

switch(whichFactory){

case CLOUDSCAPE:

return new CloudscapeDAOFactory();

case ORACLE:

return new OracleDAOFactory();

case SYBASE:

return new SybaseDAOFactory();

......

default:

return null;

}

}

}

具体的DAOFactory的实现:

public class CloudscapeDAOFactory extends DAOFactory{

public abstract CustomerDAO getCustomerDAO(){

return new CloudscapeCustomerDAO();

}

public abstract AccountDAO getAccountAO(){

return new CloudscapeAccountDAO();

}

public abstract OrderDAO getOrderDAO(){

return new CloudscapeOrderDAO();

}

}

4.3:Customer的基本DAO接口

public interface CustomerDAO{

public int insertCustomer(.....);

public boolean deleteCustomer(....);

public Customer findCustomer(....);

........

}

4.4.Customer的Cloudscape DAO实现

public class CloudscapeCustomerDAO implements CustomerDAO{

public CloudscapeCustomerDAO(){初始化;}

public int insertCustomer(.....){....};

public boolean deleteCustomer(....){.....};

public Customer findCustomer(....){......};

}

4.5:Customer值对象

public class Customer implements java.io.Serializable{

int CustomerNumber;

Stirng name;

String streetAddress;

String city;

.......//getter and setter methods

}

4.6:使用DAO和DAO工厂-----客户端代码

//创建一个DAO工厂

DAOFactory cloudscapeFactory = DAOFactory.getDAOFactory(DAOFactory.DAOCLOUDSCAPE);

//创建一个DAO

CustomerDAO custDAO = cloudscapeFactory.getCustomerDAO();

//创建一个用户

int newCustNo = custDAO.insertCustomer(.....);

//查找到一个用户

Customer cust = custDAO.findCustomer(.....);

//设置参数

cust.setAddress(...);

cust.setEmial(...);

//更新

custDAO.updateCustomer(cust);

..................等等其他操作

第三章 Java程序设计和模式应用

前面我们讲到了两个面向对象的设计原则:

一是针对接口编程,而不是针对实现编程

二是优先使用对象组合,而不是类继承

下面我们讲学习其他的常见的面向对象的设计原则:

1.开放-封闭法则(OCP)即对新增开放,对修改封闭。而且应尽量做到不用修改模块的源代码,就能更改模块的行为。

认为我们应该设计出永远也不需要改变的模块,我们可以添加新代码来扩展系统的行为。我们不能对已有的代码进行修改

符合OCP的模块需满足两个标准:

1)可扩展,即“对扩展是开放的”,---模块的行为可以被扩展,以满足新的需求

2)不可更改,即“对更改是封闭的”----模块的源代码是不允许进行改动的

如何去做呢?

1)抽象 2)多态   3)继承  4)接口

一个软件系统的所有模块不可能都满足OCP,但是我们应该努力最小化这些不满足OCP的模块数量

开放-封闭法则是OO设计的真正核心

符合该法则便意味着最高等级的复用性和可维护性

2.依赖性倒置原则:

就是依赖抽象而不要依赖具体的实现

3.接口隔离原则:

就是不要使用通用的接口,而是为不同的用户使用不同的接口

4.替换原则:

即子类应当可以替换父类并出现在父类能够出现的任何地方

类设计的基本经验:

类要单一

加强内聚,松散耦合

好的封装性

类的粒度要合理

实现类不能依赖它的使用类

应考虑灵活性,也就是可配置、可维护

要考虑性能,考虑可伸缩性

要考虑今后可能的变化,也就是可扩展性

要考虑合理的复用

要合理的考虑接口和抽象类的使用

尽量减少类与协作类的交互次数和交互信息的量

父类不应知道子类的信息,子类必须知道父类的信息

更多的使用类的组合,而不是继承

访问对象必须通过接口,不能绕过接口直接访问

分层:最典型的三层架构,表现层--》逻辑层--》数据层

表现层功能:展示数据、人机交互、收集参数调用逻辑层

逻辑层功能:进行数据的逻辑校验、进行逻辑判断、实现业务功能、处理相关功能、处理后续流程、组织数据返回给表现层

数据层功能:实现数据持久化、实现对象和持久化数据的双向映射

层间交互的基本原则:

1.表现层调用逻辑层,逻辑层调用数据层,不可反过来!!

2.层间交互也应该通过接口进行调用,以确保各层的实现独立变化

java阅读笔记的更多相关文章

  1. Effective Java阅读笔记——引言

    “我很希望10年前就拥有这本书.可能有人认为我不需要任何Java方面的书籍,但是我需要这本书.” ——Java之父 James Gosling 在图书馆找到这本java著作时,首先看到了这句话.   ...

  2. Effective Java 阅读笔记——并发

    66:同步访问共享的可变数据 synchronized:1互斥,阻止线程看到的对象处于不一致的状态:2保证线程在进入同步区时能看到变量的被各个线程的所有修改 Java中,除了long或者double, ...

  3. Effective Java 阅读笔记——枚举和注解

    30:用enum代替int常量 当需要一组固定常量的时候,应该使用enum代替int常量,除了对于手机登资源有限的设备应该酌情考虑enum的性能弱势之外. 31:用实例域代替序数 应该给enum添加i ...

  4. Effective Java 阅读笔记——方法

    38:检查参数的有效性 每当编写方法或者构造器的时候,应该考虑它的参数有哪些限制,在方法的开头处对参数进行检查,并且把这些限制写入文档. 注意: 对于公有方法,应该使用@throws标签在文档中说明违 ...

  5. 瘋耔java语言笔记

    一◐ java概述                                                                                        1.1 ...

  6. 《Thinking In Java》阅读笔记

    <Thinking In Java>阅读笔记 前四章:对象导论. 一切都是对象. 操作符. 控制执行流程 public在一个文件中只能有一个,可以是一个类class或者一个接口interf ...

  7. 《Java编程思想》阅读笔记二

    Java编程思想 这是一个通过对<Java编程思想>(Think in java)进行阅读同时对java内容查漏补缺的系列.一些基础的知识不会被罗列出来,这里只会列出一些程序员经常会忽略或 ...

  8. Java 接口与继承 道至简第六章发表阅读笔记

    一.继承条件下的构造方法调用 class Grandparent { public Grandparent() { System.out.println("GrandParent Creat ...

  9. 云时代架构阅读笔记六——Java内存模型详解(二)

    承接上文:云时代架构阅读笔记五——Java内存模型详解(一) 原子性.可见性.有序性 Java内存模型围绕着并发过程中如何处理原子性.可见性和有序性这三个特征来建立的,来逐个看一下: 1.原子性(At ...

随机推荐

  1. Poj2482 Stars in Your Window(扫描线)

    题面 Poj 题解 下面内容引用自"李煜东 <算法竞赛进阶指南>"(对原文略有缩减,侵删): 因为矩形的大小固定,所以矩形可以由它的任意一个顶点唯一确定.我们可以考虑把 ...

  2. angularjs学习笔记3-directive中scope的绑定修饰符

    在angularjs中,一个directive返回一个对象,对象存在很多属性,并且可以在directive中自定义自己的scope,而使用自己的scope是为了防止一个directive被使用在多个地 ...

  3. VB查询数据库之结账——机房收费系统总结(五)

    对于机房收费的结账,我感觉是所有窗体中,最难的一个.这个窗体我真的做了好多天.它的难度系数我感觉是最高的. 首先,你要理清上机时间和收费标准的关系,在预备时间中,是不收费的. 其次,在超过预备时间,一 ...

  4. [Arc074E] RGB Sequence

    [Arc074E] RGB Sequence Description 今天也在愉快地玩Minecraft!现在MM有一块1?N的空地,每个格子按照顺序标记为1到N.MM想要在这块空地上铺上红石块.绿宝 ...

  5. HDU 6057 Kanade's convolution(FWT)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=6057 [题目大意] 有 C[k]=∑_(i&j=k)A[i^j]*B[i|j] 求 Ans ...

  6. 【推导】Codeforces Round #432 (Div. 2, based on IndiaHacks Final Round 2017) B. Arpa and an exam about geometry

    题意:给你平面上3个不同的点A,B,C,问你能否通过找到一个旋转中心,使得平面绕该点旋转任意角度后,A到原先B的位置,B到原先C的位置. 只要A,B,C构成等腰三角形,且B为上顶点.那么其外接圆圆心即 ...

  7. 20162312 2016-2017-2《Java程序设计》课程总结

    一.每周作业链接汇总 预备作业01 写的是有关老师和学生的关系: 预备作业02 如何做中学: 预备作业03 实验楼学习linux环境: 第一周作业 java入门,虚拟机等课前准备: 第二周作业 掌握J ...

  8. 分页器与form表单

    分页器 分页器对象 from django.core.paginator import Paginator Paginator对象: paginator = Paginator(user_list, ...

  9. 修改request的parameter的几种方式(转载)

    转载地址:https://blog.csdn.net/xieyuooo/article/details/8447301

  10. win7 64位 php环境开启curl服务Call to undefined function

    无法使用curl_init(),一般情况问题可能出在没有去加载php的扩展文件php_curl.dll(windows操作系统),但是检查了一下系统配置,发现,环境下已经将php.ini文件里 ;ex ...