java设计模式(三)--抽象工厂模式

转载:http://zz563143188.iteye.com/blog/1847029

前面的工厂方法模式虽然清晰,但还是感觉有些繁琐,通常使用的还是抽象工厂模式。

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

3.1创建发送接口

/**
 * 发送接口
 * Created by mrf on 2016/2/25.
 */
public interface Sender {
    public String send();
}

3.2创建两个实现类

/**
 * 邮件发送
 * Created by mrf on 2016/2/25.
 */
public class MailSender implements Sender {
    @Override
    public String send() {
        System.out.println("This is emailSender!");
        return "email";
    }
}
/**
 * 短信发送
 * Created by mrf on 2016/2/25.
 */
public class SmsSender implements Sender {
    @Override
    public String send() {
        System.out.println("This is SmsSender!!");
        return "sms";
    }
}

3.3创建抽象工厂

/**
 * 抽象工厂模式
 * 消息发送提供者
 * Created by mrf on 2016/2/26.
 */
public interface Provider {
    public Sender produce();
}

3.4创建两个工厂实现类

/**
 * Created by mrf on 2016/2/26.
 */
public class MailSendFactory implements Provider {
    @Override
    public Sender produce() {
        return new MailSender();
    }
}
/**
 * Created by mrf on 2016/2/26.
 */
public class SmsSendFactory implements Provider {
    @Override
    public Sender produce() {
        return new SmsSender();
    }
}

3.5测试

/**
 * Created by mrf on 2016/2/26.
 */
public class ProviderTest {

    private Provider provider;

    @Test
    public void testMailProduce() throws Exception {
        provider = new MailSendFactory();
        Sender sender = provider.produce();
        assertEquals("email", sender.send());
    }
    @Test
    public void testSmsProduce() throws Exception {
        provider = new SmsSendFactory();
        Sender sender = provider.produce();
        assertEquals("sms", sender.send());
    }
}

 分析:

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

这里主要利用的原则是:上转型对象。就是父类可以调用子类继承或重写的方法。provider接口可以调用实现了这个接口的类的对应的方法,如果实现类定义了其他方法则无法调用。这里涉及java的基础知识。下面再分析下上转型对象。

3.6上转型对象

3.6.1定义

A为基类(可以为接口),B为子类或实现类。A a;a = new B();a就是上转型对象。

3.6.2特点

  1. 上转型对象不能操作子类新增加的成员变量,不能使用子类新增的方法。即为较子类B失去一些属性和功能,这些属性和功能是新增的。
  2. 上转型对象可以操作子类继承或隐藏的成员变量,也可以使用子类继承的或重写的方法。即为上转型对象可以操纵父类原有的属性和功能,无论这些方法是否被重写。
  3. 上转型对象调用方法时,就是调用子类继承和重写过的方法。而不会是新增的方法,也不是父类原有的方法。
  4. 可以将对象的上转型对象再强制转换到一个子类对象,强制转换过的对象具有子类所有属性和功能。

3.6.3注意

  • 接口定义的权限都是public的,因为继承体系权限不可以缩小
  • 将子类或实现类赋值给基类,基类只能操作基类本身和子类重写的东西

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏牛肉圆粉不加葱

[6] - 类和对象之进阶(二)

Scala 中的可见性非常灵活且复杂,这篇文章希望通过大量的示例来说清楚各种情况下的可见性是怎么样的。

612
来自专栏xingoo, 一个梦想做发明家的程序员

20120918-双向链表类定义《数据结构与算法分析》

将新的节点插入双向链表的时候: iterator insert(iterator itr,const Object & x)//向双向链表中插入一个x节点 { ...

1896
来自专栏java工会

java内部类的全面总结

1766
来自专栏有趣的Python

2-Java基础语法-常量与变量

_hello 合法; Void 合法(区分大小写,和void不一样); abc$123 合法; abc 123 非法(包含空格)

461
来自专栏菜鸟前端工程师

JavaScript学习笔记003-函数0属性0模板字符串

592
来自专栏along的开发之旅

Sevlet的生命周期

Servlet的生命周期是指: servlet类对象什么时候创建,什么时候调用什么方法,什么时候销毁。

802
来自专栏苍云横渡学习笔记

【JavaSE(五)】Java面向对象(下)

893
来自专栏Java技术栈

JDK8新特性之接口默认方法与静态方法

接口默认方法与静态方法 有这样一些场景,如果一个接口要添加一个方法,那所有的接口实现类都要去实现,而某些实现类根本就不需要实现这个方法也要写一个空实现,所以接口...

3199
来自专栏GreenLeaves

C# 装箱和拆箱

C#的值类型可以分为在栈上分配内存的值类型和在托管堆上分配内存的引用类型。 1、那么值类型和引用类型能否相互转换呢? 答案是肯定的,C#通过装箱和拆箱来实现两者...

16910
来自专栏Linux驱动

23.C++- 继承的多种方式、显示调用父类构造函数、父子之间的同名函数、virtual虚函数  上章链接: 22.C++- 继承与组合,protected访问级别

继承方式 继承方式位于定义子类的”:”后面,比如: class Line : public Object //继承方式是public {...

3209

扫描关注云+社区