java设计模式(23种Java设计模式)

圈圈笔记 95

一、设计模式概念

定义

设计模式是一套对代码设计经验的总结,被人们反复利用,多人熟知的代码设计方式;Java一共包含23种设计模式,其目的为了提高代码的可读性,可扩展性以及代码的复用性,为了解决在写代码过程中遇到的代码设计问题。

设计模式的六大原则

1.开闭原则

2.里氏替换原则

3.依赖倒转原则

4.接口隔离原则

5.最少知道原则

6.合成复用原则

设计模式的分类

创建型模式:对象实例化的模式,创建型模式用于解耦对象的实例化过程。

结构型模式:把类或对象结合在一起形成一个更大的结构。

行为型模式:类和对象如何交互,及划分责任和算法。

二、常用的几种设计模式

1、单例模式

单例模式是创建对象的一种特殊方式,程序从始至终都只创建一个对象叫单例(单实例)

1.1、懒汉式单例

publicclassSingleTest{//懒汉式单例,在使用时创建对象//1、私有静态变量privatestaticSingleTest singleTest=null;//2、将构造器私有化privateSingleTest(){ }//3、提供一个静态方法,并返回该类的对象publicstaticSingleTestgetInstance(){if(singleTest==null){//第一次访问singleTest=newSingleTest();; }returnsingleTest; } }

1.2、饿汉式单例

publicclassSingle{//1、 饿汉式单例模式, 在类加载时创建一个对象privatestaticSingle single =newSingle();// 2 构造器私有化privateSingle(){ }// 3 提供返回类对象的静态方法publicstaticSinglegetInstance(){if(single !=null){returnsingle; }returnnull; = } }

2、工厂方法模式

就是建立一个工厂类,对实现了同一接口的一些类进行实例的创建。首先看下关系图:

举例如下:(我们举一个发送邮件和短信的案列)

首先,创建二者的共同接口:

publicinterfaceSender{publicvoidSend(); }

其次,创建实现类:

publicclassMailSenderimplementsSender{@OverridepublicvoidSend(){ System.out.println("this is mailsender!"); } }publicclassSmsSenderimplementsSender{@OverridepublicvoidSend(){ System.out.println("this is sms sender!"); } }

最后,建工厂类:

publicclassSendFactory{publicSenderproduce(String type){if("mail".equals(type)) {returnnewMailSender(); }elseif("sms".equals(type)) {returnnewSmsSender(); }else{ System.out.println("请输入正确的类型!");returnnull; } } }

我们来测试下:

publicclassFactoryTest{publicstaticvoidmain(String[] args){ SendFactory factory =newSendFactory(); Sender sender = factory.produce("sms"); sender.Send(); } }

输出:this is sms sender!

(2)多个工厂方法模式

该模式是对普通工厂方法模式的改进,在普通工厂方法模式中,如果传递的字符串出错,则不能正确创建对象,而多个工厂方法模式是提供多个工厂方法,分别创建对象。关系图:

将上面的代码做下修改,改动下SendFactory类就行,如下:

publicSenderproduceMail(){returnnewMailSender(); }publicSenderproduceSms(){returnnewSmsSender(); } }

测试类如下:

publicclassFactoryTest{publicstaticvoidmain(String[] args){ SendFactory factory =newSendFactory(); Sender sender = factory.produceMail(); sender.Send(); } }

输出:this is mailsender!

(3)静态工厂方法模式

将上面的多个工厂方法模式里的方法置为静态的,不需要创建实例,直接调用即可。

publicclassSendFactory{publicstaticSenderproduceMail(){returnnewMailSender(); }publicstaticSenderproduceSms(){returnnewSmsSender(); } }publicclassFactoryTest{publicstaticvoidmain(String[] args){ Sender sender = SendFactory.produceMail(); sender.Send(); } }

输出:this is mailsender!

总体来说,工厂模式适合:凡是出现了大量的产品需要创建,并且具有共同的接口时,可以通过工厂方法模式进行创建。在以上的三种模式中,第一种如果传入的字符串有误,不能正确创建对象,第三种相对于第二种,不需要实例化工厂类,所以,大多数情况下,我们会选用第三种——静态工厂方法模式。

2、抽象工厂模式(Abstract Factory)

工厂方法模式有一个问题就是,类的创建依赖工厂类,也就是说,如果想要拓展程序,必须对工厂类进行修改,这违背了闭包原则,所以,从设计角度考虑,有一定的问题,如何解决?就用到抽象工厂模式,创建多个工厂类,这样一旦需要增加新的功能,直接增加新的工厂类就可以了,不需要修改之前的代码。因为抽象工厂不太好理解,我们先看看图,然后就和代码,就比较容易理解。

请看案列:

publicinterfaceSender{publicvoidSend(); }

两个实现类:

publicclassMailSenderimplementsSender{@OverridepublicvoidSend(){ System.out.println("this is mailsender!"); } }publicclassSmsSenderimplementsSender{@OverridepublicvoidSend(){ System.out.println("this is sms sender!"); } }

两个工厂类:

publicclassSendMailFactoryimplementsProvider{@OverridepublicSenderproduce(){returnnewMailSender(); } }publicclassSendSmsFactoryimplementsProvider{@OverridepublicSenderproduce(){returnnewSmsSender(); } }

在提供一个接口:

publicinterfaceProvider{publicSenderproduce(); }

测试类:

publicclassTest{publicstaticvoidmain(String[] args){ Provider provider =newSendMailFactory(); Sender sender = provider.produce(); sender.Send(); } }

其实这个模式的好处就是,如果你现在想增加一个功能:发及时信息,则只需做一个实现类,实现Sender接口,同时做一个工厂类,实现Provider接口,就OK了,无需去改动现成的代码。这样做,拓展性较好!

模板方法

定义:

模板方法是一种行为模式,父类的一个方法定义完成这个方法的步骤,但不具体实现具体细节,由子类完成各个步骤的实现,在创建子类对象时,最终实现过程是子类的方法。

模板方法的准备:

1、继承关系

2、父类是抽象类:抽象类实现了模板方法,定义了算法的估计

3、子类继承抽象类:实现抽象方法,完成完整的算法

publicabstractclassAbstractPerson{/** * 定义一个模板方法,用于实现这个方法的基本骨架 * 每一步骤的具体实现由子类完成 */publicvoidpreparedSchool(){ getUp(); dressing(); eat(); }//起床publicabstractvoidgetUp();//穿衣服publicabstractvoiddressing();//吃早餐publicabstractvoideat(); }publicclassStudentextendsAbstractPerson{@OverridepublicvoidgetUp(){ System.out.println("学生起床,起不来,闹钟响3次"); }@Overridepublicvoiddressing(){ System.out.println("学生穿衣服,找不到衣服"); }@Overridepublicvoideat(){ System.out.println("学生吃早餐,来不及吃早餐"); } }publicclassTeacherextendsAbstractPerson{@OverridepublicvoidgetUp(){ System.out.println("老师起床,7点半起床"); }@Overridepublicvoiddressing(){ System.out.println("老师要衣服得体,穿工装"); }@Overridepublicvoideat(){ System.out.println("老师吃早餐。"); } }publicclassTest1{publicstaticvoidmain(String[] args){ Student stu =newStudent(); stu.preparedSchool(); Teacher teacher =newTeacher(); teacher.preparedSchool(); } }

上一篇:

下一篇:

  推荐阅读

分享