Skip to content

Commit f8099e1

Browse files
author
samuel.casanova
committed
add flyweight design pattern real world example
1 parent bcce5af commit f8099e1

File tree

2 files changed

+180
-0
lines changed

2 files changed

+180
-0
lines changed

src/Flyweight/RealWorld/Output.txt

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
Initially the application takes:
2+
178MB of rss
3+
111MB of heapTotal
4+
86MB of heapUsed
5+
3MB of external
6+
1MB of arrayBuffers
7+
After creating 6500 vehicles the application takes:
8+
201MB of rss
9+
136MB of heapTotal
10+
104MB of heapUsed
11+
3MB of external
12+
1MB of arrayBuffers
13+
Lets create some vehicles flyweights directly to see what happens
14+
After creating 100 flyweights finally the application takes:
15+
1122MB of rss
16+
1055MB of heapTotal
17+
1019MB of heapUsed
18+
3MB of external
19+
1MB of arrayBuffers

src/Flyweight/RealWorld/index.ts

+161
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,161 @@
1+
/**
2+
* EN: Real World Example for the Flyweight Design Pattern
3+
*
4+
* Need: Represent a map of a city with tons of cars and trucks, each with a
5+
* 3D model
6+
*
7+
* Solution: Having a pool of shared 3D vehicle and building models
8+
*/
9+
10+
/**
11+
* EN: The VehicleFlyweight class stores only the shared portion of the state
12+
*/
13+
class VehicleFlyweight {
14+
public shared3DModel: number[];
15+
16+
constructor(protected vehicleType: VehicleType) {
17+
switch (vehicleType) {
18+
case VehicleType.Car:
19+
this.shared3DModel = this.readFile('mediumCar.3d');
20+
break;
21+
case VehicleType.Truck:
22+
this.shared3DModel = this.readFile('largeTruck.3d');
23+
break;
24+
default:
25+
this.shared3DModel = this.readFile('smallMotorbike.3d');
26+
}
27+
}
28+
29+
protected readFile(filename: string): number[] {
30+
if (/^large/.test(filename)) {
31+
return Array.from({ length: 1024 * 1024 }, () => Math.random());
32+
}
33+
if (/^medium/.test(filename)) {
34+
return Array.from({ length: 1024 * 256 }, () => Math.random());
35+
}
36+
return Array.from({ length: 1024 * 16 }, () => Math.random());
37+
}
38+
39+
public render(x: number, y: number, direction: Direction) {
40+
console.log(`Rendered ${this.vehicleType} in position {${x}, ${y}} with direction ${direction}º`);
41+
}
42+
}
43+
enum VehicleType {
44+
Car = 'Car',
45+
Truck = 'Truck',
46+
Motorbike = 'Motorbike',
47+
}
48+
enum Direction {
49+
North = 0,
50+
NorthEast = 45,
51+
East = 90,
52+
SouthEast = 135,
53+
South = 180,
54+
SouthWest = 225,
55+
West = 270,
56+
NorthWest = 315,
57+
}
58+
59+
/**
60+
* EN: The Vehicle class constains the intrinsic state and a reference to the
61+
* shared state
62+
*/
63+
export class Vehicle {
64+
constructor(
65+
public vehicleType: VehicleType,
66+
public x: number,
67+
public y: number,
68+
public direction: Direction,
69+
protected vehicleFlyweight: VehicleFlyweight,
70+
) {}
71+
72+
public render(x: number, y: number, direction: Direction) {
73+
this.vehicleFlyweight.render(x, y, direction);
74+
}
75+
}
76+
/**
77+
* EN: The Vehicle factory internally manages all the Flyweight objects
78+
*/
79+
class VehicleFactory {
80+
private static vehicleFlyweights: Map<VehicleType, VehicleFlyweight> =
81+
new Map<VehicleType, VehicleFlyweight>();
82+
83+
/**
84+
* EN: Checks if the external state exists in the cache, otherwise it
85+
* creates a new one and stores it for reusing in the future
86+
*/
87+
protected static getVehicle(
88+
vehicleType: VehicleType,
89+
x: number,
90+
y: number,
91+
direction: Direction,
92+
): Vehicle {
93+
if (!this.vehicleFlyweights.has(vehicleType)) {
94+
this.vehicleFlyweights.set(vehicleType, new VehicleFlyweight(vehicleType));
95+
}
96+
return new Vehicle(vehicleType, x, y, direction, this.vehicleFlyweights.get(vehicleType));
97+
}
98+
99+
public static getCar(x: number, y: number, direction: Direction): Vehicle {
100+
return this.getVehicle(VehicleType.Car, x, y, direction);
101+
}
102+
103+
public static getTruck(x: number, y: number, direction: Direction): Vehicle {
104+
return this.getVehicle(VehicleType.Truck, x, y, direction);
105+
}
106+
107+
public static getMotorbike(x: number, y: number, direction: Direction): Vehicle {
108+
return this.getVehicle(VehicleType.Motorbike, x, y, direction);
109+
}
110+
}
111+
112+
/**
113+
* EN: The client code is not aware of the internal representation, so no
114+
* reference to Flyweight objects are present.
115+
*/
116+
117+
console.log('Initially the application takes:');
118+
for (const [key, value] of Object.entries(process.memoryUsage())) {
119+
console.log(` ${Math.round(value / (1024 * 1024))}MB of ${key}`);
120+
}
121+
122+
const vehicles: Vehicle[] = [];
123+
124+
for (let i = 0; i < 1000; i += 1) {
125+
const x = Math.random() * 1000;
126+
const y = Math.random() * 1000;
127+
const direction = i % 2 ? Direction.North : Direction.South;
128+
vehicles.push(VehicleFactory.getCar(x, y, direction));
129+
}
130+
131+
for (let i = 0; i < 500; i += 1) {
132+
const x = Math.random() * 1000;
133+
const y = Math.random() * 1000;
134+
const direction = i % 2 ? Direction.East : Direction.West;
135+
vehicles.push(VehicleFactory.getTruck(x, y, direction));
136+
}
137+
138+
for (let i = 0; i < 5000; i += 1) {
139+
const x = Math.random() * 1000;
140+
const y = Math.random() * 1000;
141+
const direction = i % 2 ? Direction.SouthEast : Direction.NorthWest;
142+
vehicles.push(VehicleFactory.getMotorbike(x, y, direction));
143+
}
144+
145+
console.log(`After creating ${vehicles.length} vehicles the application takes:`);
146+
for (const [key, value] of Object.entries(process.memoryUsage())) {
147+
console.log(` ${Math.round(value / (1024 * 1024))}MB of ${key}`);
148+
}
149+
150+
console.log('Lets create some vehicles flyweights directly to see what happens');
151+
152+
const flyweights: VehicleFlyweight[] = [];
153+
154+
for (let i = 0; i < 100; i += 1) {
155+
flyweights.push(new VehicleFlyweight(VehicleType.Truck));
156+
}
157+
158+
console.log(`After creating ${flyweights.length} flyweights finally the application takes:`);
159+
for (const [key, value] of Object.entries(process.memoryUsage())) {
160+
console.log(` ${Math.round(value / (1024 * 1024))}MB of ${key}`);
161+
}

0 commit comments

Comments
 (0)