Proxy design pattern allows you to create a wrapper class over real object. Wrapper class which is proxy, controls access to real object so in turn you can add extra functionalities to real object without changing real object's code.
As described by GoF:
"Provide a surrogate or placeholder for another object to control access over it."
Real life example may be proxy server used in IT companies for internet access.It blocks access to social networking sites like facebook,twitter and other contents which are not work related.
When To Use It
Proxy is required whenever there is need for more sophisticated or versatile reference to an object than a simple pointer. Here are some situations when proxy pattern is applicable.
- A remote proxy provides a local representative for an object in a different address space. Providing interface for remote resources such as web service or REST resources.
- A virtual proxy creates expensive object on demand.
- A protection proxy controls access to the original object. Protection proxies are useful when objects should have different access rights.
- A smart reference is a replacement for a bare pointer that performs additional actions when an object is accessed.
- Adding a thread-safe feature to an existing class without changing the existing class's code.
UML diagram for Proxy Design Pattern
Elements
- Proxy
- maintains a reference that lets the proxy access the real subject.Proxy may refer to a subject if the RealSubject and Subject interface are the same.
- provides an interface identical to Subject's so that proxy can be substituted for the real subject.
- controls access to the real subject and may be responsible for creating and deleting it.
- Subject
- defines the common interface for RealSubject and proxy so that a Proxy can be used anywhere a RealSubject is expected.
- RealSubject
- defines the real object that proxy represents.
WorkFlow
Proxy forwards request to RealSubject when appropriate, depending on the kind of proxy.
Example
Let's see an example. I am taking example of protection proxy. In our example, we have a folder in which you can perform various operation such as copy, paste file or subfolder. In OOP terms, we have IFolder interface and Folder class which provides performOperatons() method, and these are the existing class and interface that we cannot change. We want to further specify that only user with authorization can access it and perform operations such as cut or copy files and sub folder.
UML diagram for example:
Comparing with above generic elements, we have
- FolderProxy(Proxy)
- Folder(RealSubject)
- IFolder(Subject)
Java codes for above classes:
IFolder.java (Subject):
package org.arpit.javapostsforlearning.designpatterns;
public interface IFolder {
public void performOperations();
}
Following class is our realSubject class. Let's say we can not change it but we want to provide authorization on it.
Folder.java (RealSubject)
package org.arpit.javapostsforlearning.designpatterns;
public class Folder implements IFolder{
public void performOperations()
{
System.out.println("Performing operation on folder");
}
}
Following class provides authorization to Folder class. It checks for userName and password and if matched then only it gives access to folder.
FolderProxy.java(Proxy)
package org.arpit.javapostsforlearning.designpatterns;
public class FolderProxy implements IFolder{
Folder folder;
User user;
public FolderProxy(User user) {
this.user = user;
}
public void performOperations() {
if(user.getUserName().equalsIgnoreCase("arpit") && user.getPassword().equalsIgnoreCase("arpit"))
{
folder=new Folder();
folder.performOperations();
}
else
{
System.out.println("You don't have access to this folder");
}
}
}
User.java
package org.arpit.javapostsforlearning.designpatterns;
public class User {
String userName;
String password;
public User(String userName, String password) {
this.userName = userName;
this.password = password;
}
public String getUserName() {
return userName;
}
public String getPassword() {
return password;
}
}
ProxyDesignPatternMain.java
package org.arpit.javapostsforlearning.designpatterns;
public class ProxyDesignPatternMain {
public static void main(String[] args) {
User user=new User("arpit","arpit");
FolderProxy folderProxy=new FolderProxy(user);
System.out.println("When userName and password are correct:");
folderProxy.performOperations();
System.out.println("**************************************");
User userWrong=new User("abc","abc");
FolderProxy folderProxyWrong=new FolderProxy(userWrong);
System.out.println("When userName and password are incorrect:");
folderProxyWrong.performOperations();
}
}
Output
When userName and password are correct:
Performing operation on folder
**************************************
When userName and password are incorrect:
You don't have access to this folder
So without changing Folder class, we have provide authentication to folder with the help of FolderProxy class.