2015-08-13 23:54:40 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								---
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								layout: pattern
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								title: Specification
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								folder: specification
							 
						 
					
						
							
								
									
										
										
										
											2015-08-15 18:03:05 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								permalink: /patterns/specification/
							 
						 
					
						
							
								
									
										
										
										
											2015-08-20 21:40:07 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								categories: Behavioral
							 
						 
					
						
							
								
									
										
										
										
											2015-12-28 15:52:44 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								tags:
							 
						 
					
						
							
								
									
										
										
										
											2019-12-13 21:09:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								 -  Data access
							 
						 
					
						
							
								
									
										
										
										
											2015-08-13 23:54:40 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								---
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2016-07-07 08:46:56 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								## Also known as
  
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2016-07-07 08:46:56 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Filter, Criteria
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2016-01-03 21:14:30 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								## Intent
  
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Specification pattern separates the statement of how to match a candidate, from the candidate object 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								that it is matched against. As well as its usefulness in selection, it is also valuable for 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								validation and for building to order.
							 
						 
					
						
							
								
									
										
										
										
											2015-08-13 23:54:40 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								## Explanation
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Real world example
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								>  There is a pool of different creatures and we often need to select some subset of them. We can 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								>  write our search specification such as "creatures that can fly", "creatures heavier than 500 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								>  kilograms", or as a combination of other search specifications, and then give it to the party that 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								>  will perform the filtering.
  
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								In Plain Words
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								>  Specification pattern allows us to separate the search criteria from the object that performs the 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								>  search.
  
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Wikipedia says
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								>  In computer programming, the specification pattern is a particular software design pattern, 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								>  whereby business rules can be recombined by chaining the business rules together using boolean 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								>  logic.
  
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								**Programmatic Example**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								If we look at our creature pool example from above, we have a set of creatures with certain 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								properties. Those properties can be part of a pre-defined, limited set (represented here by the 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								enums Size, Movement and Color); but they can also be continuous values (e.g. the mass of a 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Creature). In this case, it is more appropriate to use what we call "parameterized specification", 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								where the property value can be given as an argument when the Creature is instantiated, allowing for 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								more flexibility. A third option is to combine pre-defined and/or parameterized properties using 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								boolean logic, allowing for near-endless selection possibilities (this is called "composite 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								specification", see below). The pros and cons of each approach are detailed in the table at the end 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								of this document.
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public interface Creature {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  String getName();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  Size getSize();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  Movement getMovement();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  Color getColor();
							 
						 
					
						
							
								
									
										
										
										
											2019-11-16 13:24:46 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								  Mass getMass();
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								And `Dragon`  implementation looks like this.
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class Dragon extends AbstractCreature {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public Dragon() {
							 
						 
					
						
							
								
									
										
										
										
											2019-11-16 13:24:46 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    super("Dragon", Size.LARGE, Movement.FLYING, Color.RED, new Mass(39300.0));
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Now that we want to select some subset of them, we use selectors. To select creatures that fly, we 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								should use `MovementSelector` .
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								public class MovementSelector extends AbstractSelector< Creature >  {
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  private final Movement movement;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public MovementSelector(Movement m) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    this.movement = m;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  @Override 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public boolean test(Creature t) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return t.getMovement().equals(movement);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								On the other hand, when selecting creatures heavier than a chosen amount, we use 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								`MassGreaterThanSelector` . 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-16 13:24:46 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								public class MassGreaterThanSelector extends AbstractSelector< Creature >  {
							 
						 
					
						
							
								
									
										
										
										
											2019-11-16 13:24:46 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  private final Mass mass;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public MassGreaterThanSelector(double mass) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    this.mass = new Mass(mass);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  @Override 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public boolean test(Creature t) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return t.getMass().greaterThan(mass);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								With these building blocks in place, we can perform a search for red creatures as follows:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
									
										
										
										
											2020-01-04 22:06:08 +05:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    var redCreatures = creatures.stream().filter(new ColorSelector(Color.RED))
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								      .collect(Collectors.toList());
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								But we could also use our parameterized selector like this:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
									
										
										
										
											2020-01-04 22:06:08 +05:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    var heavyCreatures = creatures.stream().filter(new MassGreaterThanSelector(500.0)
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								      .collect(Collectors.toList());
							 
						 
					
						
							
								
									
										
										
										
											2019-11-02 14:45:50 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
									
										
										
										
											2015-09-03 18:17:07 +05:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Our third option is to combine multiple selectors together. Performing a search for special 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								creatures (defined as red, flying, and not small) could be done as follows:
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
									
										
										
										
											2020-01-04 22:06:08 +05:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    var specialCreaturesSelector = 
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								      new ColorSelector(Color.RED).and(new MovementSelector(Movement.FLYING)).and(new SizeSelector(Size.SMALL).not());
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-04 22:06:08 +05:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    var specialCreatures = creatures.stream().filter(specialCreaturesSelector)
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								      .collect(Collectors.toList());
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								**More on Composite Specification**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								In Composite Specification, we will create custom instances of `AbstractSelector`  by combining 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								other selectors (called "leaves") using the three basic logical operators. These are implemented in 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								`ConjunctionSelector` , `DisjunctionSelector`  and `NegationSelector` . 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public abstract class AbstractSelector< T >  implements Predicate< T >  {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public AbstractSelector< T >  and(AbstractSelector< T >  other) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return new ConjunctionSelector< >(this, other);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public AbstractSelector< T >  or(AbstractSelector< T >  other) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return new DisjunctionSelector< >(this, other);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
									
										
										
										
											2019-11-16 13:24:46 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								  public AbstractSelector< T >  not() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return new NegationSelector< >(this);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 15:20:42 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-11-16 13:24:46 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```java
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								public class ConjunctionSelector< T >  extends AbstractSelector< T >  {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-07-30 20:28:47 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								  private final List< AbstractSelector < T > > leafComponents;
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  @SafeVarargs 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ConjunctionSelector(AbstractSelector< T > ... selectors) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    this.leafComponents = List.of(selectors);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								   *  Tests if *all*  selectors pass the test.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								   */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  @Override 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  public boolean test(T t) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return leafComponents.stream().allMatch(comp -> (comp.test(t)));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
									
										
										
										
											2019-11-16 13:24:46 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								All that is left to do is now to create leaf selectors (be it hard-coded or parameterized ones) that 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								are as generic as possible, and we will be able to instantiate the ``AbstractSelector` ` class by 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								combining any amount of selectors, as exemplified above. We should be careful though, as it is easy 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								to make a mistake when combining many logical operators; in particular, we should pay attention to 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								the priority of the operations. In general, Composite Specification is a great way to write more 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								reusable code, as there is no need to create a Selector class for each filtering operation. Instead, 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								we just create an instance of ``AbstractSelector` ` "on the spot", using tour generic "leaf" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								selectors and some basic boolean logic.
							 
						 
					
						
							
								
									
										
										
										
											2019-11-17 13:25:32 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								**Comparison of the different approaches**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								| Pattern | Usage | Pros | Cons |
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								|---|---|---|---|
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								| Hard-Coded Specification | Selection criteria are few and known in advance | + Easy to implement | - Inflexible |
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								| | | + Expressive |
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								| Parameterized Specification | Selection criteria are a large range of values (e.g. mass, speed,...) | + Some flexibility | - Still requires special-purpose classes |
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								| Composite Specification | There are a lot of selection criteria that can be combined in multiple ways, hence it is not feasible to create a class for each selector | + Very flexible, without requiring many specialized classes | - Somewhat more difficult to comprehend |
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								| | | + Supports logical operations | - You still need to create the base classes used as leaves |
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-12-07 20:01:13 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								## Class diagram
  
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-12-07 20:01:13 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								## Applicability
  
						 
					
						
							
								
									
										
										
										
											2020-09-13 18:04:30 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-12-07 20:01:13 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Use the Specification pattern when
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  You need to select a subset of objects based on some criteria, and to refresh the selection at various times. 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  You need to check that only suitable objects are used for a certain role (validation). 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2016-07-21 09:27:48 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								## Related patterns
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  Repository 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2016-01-03 21:14:30 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								## Credits
  
						 
					
						
							
								
									
										
										
										
											2015-09-03 18:17:07 +05:30 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  [Martin Fowler - Specifications ](http://martinfowler.com/apsupp/spec.pdf )