동일하거나 유사한 객체들 사이에 가능한 많은 데이터를 서로 공유하여 사용하도록 하여 메모리 사용량을 최소화 하는 패턴
모바일 기기나 임베디드 시스템 등 저용량 메모리를 지원하는 기기에서 메모리를 관리하는 것이 아주 중요합니다.
다음의 경우, 플라이웨이트를 적용하기에 적합합니다.
싱글톤 패턴은 클래스 자체가 오직 1개의 인스턴스만을 가지도록 제한하지만, 플라이웨이트 패턴은 팩토리가 제어하는 것입니다.
즉, 인스턴스 생성을 누가 제어하느냐의 차이입니다.
여러 도형을 생성하는 예제를 플라이웨이트 패턴으로 작성해보았습니다.
다음 블로그의 예시를 참조했습니다.
import java.awt.Graphics; | |
import java.awt.Color; | |
interface Shape{ | |
public void draw(Graphics g, int x, int y, int width, int height, Color color) ; | |
} |
class Line implements Shape { | |
public Line() { | |
System.out.println("Create Line Object"); | |
} | |
@Override | |
public void draw(Graphics line, int x1, int y1, int x2, int y2, Color color) { | |
line.setColor(color); | |
line.drawLine(x1, y1, x2, y2); | |
} | |
} | |
class Oval implements Shape { | |
private boolean fillOption; | |
public Oval(boolean b) { | |
this.fillOption = b; | |
System.out.println("Create Oval Object(Fill Option: "+b+")"); | |
} | |
@Override | |
public void draw(Graphics circle, int x, int y, int width, int height, Color color) { | |
circle.setColor(color); | |
// 원을 그린다. | |
circle.drawOval(x, y, width, height); | |
if (fillOption) { | |
// 그린 원을 채운다. | |
circle.fillOval(x, y, width, height); | |
} | |
} | |
} |
생성자에서 객체 생성시, 즉 new 키워드를 통해 생성시 출력을 통해 확인할 수 있습니다.
class ShapeFactory { | |
private static HashMap<ShapeType, Shape> shapes = new HashMap<ShapeType, Shape>(); | |
public static Shape getShape(ShapeType type) { | |
Shape shapeTmp = shapes.get(type); | |
if (shapeTmp == null) { | |
if (type == ShapeType.OVAL_FILL) { | |
shapeTmp = new Oval(true); | |
} else if (type == ShapeType.OVAL_NOTFILL) { | |
shapeTmp = new Oval(false); | |
} else if (type == ShapeType.LINE) { | |
shapeTmp = new Line(); | |
} | |
shapes.put(type, shapeTmp); | |
} | |
return shapeTmp; | |
} | |
public static enum ShapeType { | |
OVAL_FILL, OVAL_NOTFILL, LINE; | |
} | |
} |
Factory 내의 static Hashmap을 통해 이전에 생성된 객체들을 관리할 수 있도록 하였습니다.
도형의 타입 별로 하나의 객체를 생성하도록 제한합니다.
import java.awt.BorderLayout; | |
import java.awt.Container; | |
import java.awt.event.ActionEvent; | |
import java.awt.event.ActionListener; | |
import java.util.HashMap; | |
import javax.swing.JButton; | |
import javax.swing.JFrame; | |
import javax.swing.JPanel; | |
public class Main extends JFrame { | |
private final int WIDTH; | |
private final int HEIGHT; | |
private static final ShapeFactory.ShapeType shapes[] = { ShapeFactory.ShapeType.LINE, ShapeFactory.ShapeType.OVAL_FILL, ShapeFactory.ShapeType.OVAL_NOTFILL }; | |
private static final Color colors[] = { Color.RED, Color.GREEN, Color.YELLOW }; | |
public Main(int width, int height) { | |
this.WIDTH = width; | |
this.HEIGHT = height; | |
Container contentPane = getContentPane(); | |
JButton startButton = new JButton("Draw"); | |
final JPanel panel = new JPanel(); | |
contentPane.add(panel, BorderLayout.CENTER); | |
contentPane.add(startButton, BorderLayout.SOUTH); | |
setTitle("Flyweight Pattern Example"); | |
setSize(WIDTH, HEIGHT); | |
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); | |
setVisible(true); | |
startButton.addActionListener(new ActionListener() { | |
public void actionPerformed(ActionEvent event) { | |
Graphics g = panel.getGraphics(); | |
for (int i = 0; i < 20; ++i) { | |
Shape shape = ShapeFactory.getShape(getRandomShape()); | |
shape.draw(g, getRandomX(), getRandomY(), getRandomWidth(), | |
getRandomHeight(), getRandomColor()); | |
} | |
} | |
}); | |
} | |
private ShapeFactory.ShapeType getRandomShape() { | |
return shapes[(int) (Math.random() * shapes.length)]; | |
} | |
private int getRandomX() { | |
return (int) (Math.random() * WIDTH); | |
} | |
private int getRandomY() { | |
return (int) (Math.random() * HEIGHT); | |
} | |
private int getRandomWidth() { | |
return (int) (Math.random() * (WIDTH / 10)); | |
} | |
private int getRandomHeight() { | |
return (int) (Math.random() * (HEIGHT / 10)); | |
} | |
private Color getRandomColor() { | |
return colors[(int) (Math.random() * colors.length)]; | |
} | |
public static void main(String[] args) { | |
Main drawing = new Main(500, 600); | |
} | |
} |
JFrame으로 화면에 창을 생성하고, 버튼을 부착합니다.
버튼에 EventListoner를 부착해 버튼이 클릭되면, Random으로 좌표, 크기, 도형 모양, 색상 값을 생성해 20개의 도형을 화면에 draw 합니다.
실행하면 다음처럼 창이 뜹니다.
버튼을 클릭할 때마다 20개의 도형이 그려집니다.
그려진 도형은 20개이지만 출력된 내용은 다음처럼 도형당 한 번씩 입니다.
Create Oval Object(Fill Option: true)
Create Oval Object(Fill Option: false)
Create Line Object
즉 도형당 한 번씩 객체를 생성하고 이후에는 공유해서 사용하는 것을 알 수 있습니다.
이 상태에서 버튼을 다시 누른다해도 출력은 발생하지 않고 이전에 생성된 객체를 공유합니다.
Java 래퍼 클래스의 valueOf() 래퍼 클래스의 valueOf()는 플라이웨이트 패턴을 이용하고 있습니다. 따라서 new를 이용해 객체를 매번 생성하는 것 보다, valueOf를 통해 생성하는 것이 더 효율적입니다.
Object ob1 = String.valueOf("abc");
Object ob2 = new String("abc");
자바의 String Pool Java에서는 String pool을 별도로 두어 같은 문자열이 다시 사용될 때에 새로운 메모리를 할당하는 것이 아니라 String pool에 있는지 검사해 있으면 가져오고 없으면 새로 메모리를 할당하여 String pool에 등록한 후 사용하도록 하고 있습니다.
참조
감사합니다.
Text by Chaelin. Photographs by Chaelin, Unsplash.