Flutter Design Patterns: 3 — Template Method

An overview of the Template Method design pattern and its implementation in Dart and Flutter

Previously in the series, I have analysed one of the most common and most useful design patterns available — Adapter. In this article, I would like to analyse and implement one of the behavioural design patterns — Template Method.


Table of Contents

  • Analysis
  • Implementation
  • Other articles in this series
  • Your contribution

What is the Template Method design pattern?

Define the skeleton of an algorithm in an operation, deferring some steps to sub­classes. Template Method lets subclasses redefine certain steps of an algorithm without changing the algorithm’s structure.

The Template method is a fundamental technique for code reuse. Let’s say you have an algorithm which consists of multiple steps:

  1. Reading the data from some kind of data source;
  2. Processing it;
  3. Providing the calculated results.

In one case, you want to get the data from a third-party API, process it and provide the results in the console window. In another case, you would like to read the data from a file in your local disk drive, process it and send the formatted result document via e-mail. Despite the differences between these two cases, these algorithms share the same structure — reading, processing the data and providing the results. That’s where the Template Method design pattern is useful — when you need to allow changing the details of each step or certain steps of the algorithm while enforcing the structure and order of the steps themselves. The question is, how to define this kind of template?


Analysis

Class Diagram — Template Method
  • AbstractClass — contains a templateMethod() operation defining the skeleton of an algorithm. The template method calls primitive operations as well as operations defined in AbstractClass or those of other objects.
  • ConcreteClass — relies on AbstractClass to implement the invariant steps of the algorithm.

Template Method operation types

  • Primitive operations — abstract operations that must be implemented by subclasses; concrete operations that provide a default implementation and can be redefined by subclasses if necessary.
  • Final operations — concrete operations that can not be overridden by subclasses.
  • Hook operations — concrete operations which provide default behaviour that subclasses can extend if necessary. A hook operation often does nothing by default.
  • Template method itself can be declared as final so that it could not be overridden by subclasses.

Note: Template methods need to specify which operations are hooks and which are the abstract operations. To do that, you can identify the operations that should be overridden by adding a prefix to their names e.g. the MacApp framework prefixes template method names with “Do-”.

The Hollywood Principle

The Hollywood Principle (source)

The Hollywood Principle is simply stated as:

Don’t call us, we’ll call you!

How this statement is related to the Template Method? The base class of the Template Method is considered a high-level component — clients or concrete implementations of the algorithm should depend on it. The subclasses of the template are the low-level components — they don’t call anything themselves and are only called by the high-level template method. This describes the relationship between high-level and low-level components — abstractions should not depend on details, but the details should depend on abstractions (Dependency Inversion principle, the letter D in SOLID principles). Thus, the Template Method design pattern could state that low-level components don’t call us, we call them!

Applicability


Implementation

Let’s say, in the Flutter application we want to create an algorithm which calculates the BMI (body mass index) of students. The algorithm retrieves students’ data, applies students filtering (if necessary), calculates their BMI and provides the results. Also, we want to retrieve students information from different data sources and we want to have a separate algorithm which includes only teenage students for the calculations. As you can see, the structure of the algorithm is general, but the implementation details (algorithm steps) vary among different implementations. To implement this algorithm, we would use the Template method design pattern.

Class diagram

Class Diagram — Implementation of the Template Method design pattern

The main class in the diagram is StudentsBmiCalculator. Its primary purpose is to define a template of the BMI calculation algorithm which returns a list of Student objects (with the calculated BMI for each student) as a result via calculateBmiAndReturnStudentList() method. This abstract class is used as a template (base class) for the concrete implementations of the students’ BMI calculation algorithm — StudentsXmlBmiCalculator, StudentsJsonBmiCalculator and TeenageStudentsJsonBmiCalculator. StudentsXmlBmiCalculator uses the XmlStudentsApi to retrieve students information as an XML string and returns it as a list of Student objects via the overridden getStudentsData() method. Both of the other two implementations (StudentsJsonBmiCalculator and TeenageStudentsJsonBmiCalculator) uses the JsonStudentsApi to retrieve students information in JSON format and returns the parsed data via the overridden getStudentsData() method. However, TeenageStudentsJsonBmiCalculator additionally reimplements (overrides) the doStudentsFiltering() hook method to filter out not teenage students before calculating the BMI values. StudentsSection UI widget uses the StudentsBmiCalculator abstraction to retrieve and represent the calculated results in TemplateMethodExample widget.

Note: it would make more sense to extract the XML or JSON parsing logic from the getStudentsData() method in the concrete calculation classes to a separate class using the Adapter design pattern, but for the demonstration purposes of the Template Method pattern and to show that the implementation of getStudentsData() may differ among the derived classes of the StudentsXmlBmiCalculator, I have decided to leave logic as it is now.

StudentsBmiCalculator

The algorithm consists of several steps:

  1. Retrieve students data — getStudentsData();
  2. Do students filtering (if needed) — doStudentsFiltering();
  3. Calculate the BMI for each student — _calculateStudentsBmi();
  4. Return students data — return studentList.

The first step is mandatory and should be implemented in each concrete implementation of the students BMI calculator — that is, the method getStudentsData() is abstract and must be overridden in the derived class (primitive operation). Students filtering step is optional, yet it could be overridden in the derived class. For this reason, doStudentsFiltering() method has a default implementation which does not change the workflow of the algorithm by default (hook operation). Other steps are defined in the algorithm’s template itself, are common for all implementations and could not be changed (final operations).

StudentsXmlBmiCalculator

StudentsJsonBmiCalculator

TeenageStudentsJsonBmiCalculator

Additionally, the doStudentsFiltering() hook method is overridden to filter out not teenage students.

Student

JsonStudentsApi

XmlStudentsApi

Example

The example itself uses StudentsSection component which requires a specific BMI calculator of type StudentsBmiCalculator to be provided via a constructor. For this example, we inject three different implementations of BMI calculator (StudentsXmlBmiCalculator, StudentsJsonBmiCalculator and TeenageStudentsJsonBmiCalculator) which extend the same template (base class) — StudentsBmiCalculator — to three different StudentsSection widgets.

StudentsSection uses the injected BMI calculator of type StudentsBmiCalculator. The widget does not care about the specific implementation of the BMI calculator as long as it uses (extends) the same template (base class). This lets us provide different students’ BMI calculation algorithms/implementations without making any changes to the UI code.

The final result of the Template Method’s implementation looks like this:

All of the code changes for the Template Method design pattern and its example implementation could be found here.



Your contribution

Flutter Community

Articles and Stories from the Flutter Community

Mangirdas Kazlauskas

Written by

Software Engineer | Flutter Enthusiast https://www.linkedin.com/in/mangirdas-kazlauskas/

Flutter Community

Articles and Stories from the Flutter Community

Welcome to a place where words matter. On Medium, smart voices and original ideas take center stage - with no ads in sight. Watch
Follow all the topics you care about, and we’ll deliver the best stories for you to your homepage and inbox. Explore
Get unlimited access to the best stories on Medium — and support writers while you’re at it. Just $5/month. Upgrade