Adapter Pattern

What is Adapter Pattern?

Adapter pattern is a structural pattern that converts the interface of a class into another interface the clients expect. Adapter lets classes work together that couldn’t otherwise because of incompatible interfaces.

When to use it?

  • Use Adapter pattern when you want to use legacy code or third-party libraries but its interface is incompatible with other part of your application.

Problem

We’ve been using an old printer but now new free software is invented which is integrated with modern printer. The problem is we can’t afford to buy modern printer but still want to use the software. Is there a way of using new software but with old printer which is incompatible with the software? Yes, here Adapter pattern comes in handy.

Solution

  1. NewSoftware
    This is our Client. This only accepts ModernPrinter interface. NewSoftware thinks it’s dealing with ModernPrinter.

  2. ModernPrinter
    Provides compatible interface for Client.

  3. PrinterAdapter
    Receives a method call modernPrint() from ModernPrinter, then translates it into a format the

  4. OldPrinter
    This is what we want to use but its interface is incompatible with our NewSoftware. oldPrint() gets executed when ModernPrinter calls modernPrint().

Structure

Remember an adapter can be used with any subclass of the adaptee.

Implementation in Java

public class OldPrinter {

    public void oldPrint(String document) {
        System.out.println(document + " by old printer");
    }
}

Enter fullscreen mode Exit fullscreen mode

public interface ModernPrinter {

    void modernPrint(String document);
}

Enter fullscreen mode Exit fullscreen mode

public class PrinterAdapter implements ModernPrinter {

    private OldPrinter oldPrinter;

    public PrinterAdapter(OldPrinter oldPrinter) {
        this.oldPrinter = oldPrinter;
    }

    @Override
    public void modernPrint(String document) {
        oldPrinter.oldPrint(document);
    }
}

Enter fullscreen mode Exit fullscreen mode

public class NewSoftware {

    public static void main(String[] args) {
        OldPrinter oldPrinter = new OldPrinter();
        // Because PrinterAdapter implements ModernPrinter, it can be seen
        // as a ModernPrinter which is integrated with NewSoftware
        ModernPrinter adapter = new PrinterAdapter(oldPrinter);
        adapter.modernPrint("Hello world");
    }
}

Enter fullscreen mode Exit fullscreen mode

In our assumption, you NewSoftware is integrated with ModernPrinter, so it doesn’t make sense to code like oldPrinter.oldPrint("Hello world"); in NewSoftware class (even though it will be compiled without errors).

Output:

Hello world by old printer

Enter fullscreen mode Exit fullscreen mode

Object and Class Adapter

What we’ve seen is called an Object Adapter, which adapts the adaptee using composition. There is another type of adapter called a Class Adapter, which uses inheritance to achieve the intent of the Adapter pattern.

Java doesn’t support multiple inheritance. So we can’t use class adapter in Java, but we will study it anyway as you might encounter class adapter in other language.

Because object adapter uses composition, Adapter can hold references to one or more Adaptee objects. Thus, object adapter offers more flexibility. While class adapter is tightly coupled with one and only one adaptee which leads to less flexibility.

Pitfalls

  • An adapter that does a lot of work besides simple interface translation results in adding/altering adaptee behavior, which is far from the intent of Adapter pattern.

Comparison with Decorator Pattern

  • Decorator pattern wraps object to add small behavior, while Adapter pattern wraps object to adapt to the interface that is required by Client.

You can check all the design pattern implementations here.
GitHub Repository

原文链接:Adapter Pattern

© 版权声明
THE END
喜欢就支持一下吧
点赞14 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容