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 条评论
登录 后参与评论

相关文章

来自专栏CaiRui

Python之函数

简介: 定义函数: def xx():     print("xxx") 执行函数 xx() 函数返回值: def emile(): print("发邮...

2099
来自专栏我的技术专栏

Java锁机制(一)synchronized

1464
来自专栏康怀帅的专栏

PHP 面向对象

面向对象三大特性:封装、继承、多态。 官方文档:http://php.net/manual/zh/language.oop5.php 要创建一个类的实例,必须使...

2947
来自专栏尚国

PHP反序列化漏洞

这里你可以看到, 我代码里的类定义为: class F, 这个序列化就是 F, 我定义变量名字是filename, 它这里也是 filename, 我们可以修改...

692
来自专栏用户2442861的专栏

static在C和C++中的用法和区别

http://blog.csdn.net/skyereeee/article/details/8000512

881
来自专栏编程

python的函数(二):作用域

我们在写函数时,时常需要引用全局的变量,或对全局变量赋值。又或者偶尔遇到局部变量与全局变量同名。在处理这些问题时,python语言的游戏规则是怎样的?今天我们就...

1795
来自专栏吴伟祥

logback高级特性使用 原

logback支持类似于占位符的变量替换功能,即如果输出的msg里面带有{}符号且括号中间不带其他字符,那么logback在构造LoggingEvent的时候,...

502
来自专栏九彩拼盘的叨叨叨

JS模式 之通用模式(译)

原文 http://shichuan.github.io/javascript-patterns/

983
来自专栏前端学习心得

闭包不过如此

1292
来自专栏技术之路

实例化和具体化详解

primer Plus在解释具体化和实例化看的有点乱,分解出来备忘 在代码中包含函数模板本身并不会生成函数定义,它只是用于生成函数定义的方案 编译器使用模板为我...

1825

扫码关注云+社区