Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

creational/abstract-factory: adds implementation #97

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ A curated collection of idiomatic design & application patterns for Go language.

| Pattern | Description | Status |
|:-------:|:----------- |:------:|
| [Abstract Factory](/creational/abstract_factory.md) | Provides an interface for creating families of releated objects | ✘ |
| [Abstract Factory](/creational/abstract-factory.md) | Provides an interface for creating families of releated objects | ✘ |
| [Builder](/creational/builder.md) | Builds a complex object using simple objects | ✔ |
| [Factory Method](/creational/factory.md) | Defers instantiation of an object to a specialized function for creating instances | ✔ |
| [Object Pool](/creational/object-pool.md) | Instantiates and maintains a group of objects instances of the same type | ✔ |
Expand Down
2 changes: 1 addition & 1 deletion SUMMARY.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@

* [Go Patterns](/README.md)
* [Creational Patterns](/README.md#creational-patterns)
* [Abstract Factory](/creational/abstract_factory.md)
* [Abstract Factory](/creational/abstract-factory.md)
* [Builder](/creational/builder.md)
* [Factory Method](/creational/factory.md)
* [Object Pool](/creational/object-pool.md)
Expand Down
128 changes: 128 additions & 0 deletions creational/abstract-factory.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,128 @@
# Abstract Factory Method Pattern

Abstract Factory is a creational design pattern that allows you to create a _collection of related objects_.
It provides an extra level of indirection over the [factory design pattern](factory.md).
Using this pattern you can decouple the factory from your client code.

## Implementation

The example implementation shows how to create a collection (top & bottom) of summer clothes or winter clothes.

### Types

We begin with the simplest interfaces to represent a top or bottom article of clothing. Our items of clothing are [Stringer](https://tour.golang.org/methods/17) types.

```go
package main

import "fmt"

type Top interface {
String() string
}

type Bottom interface {
String() string
}
```

Next we introduce a couple of tops and bottoms into our system.

```go
type TankTop struct {
}

func (top TankTop) String() string {
return "Tank Top"
}

type Shorts struct {
}

func (bottom Shorts) String() string {
return "Shorts"
}

type Sweater struct {
}

func (top Sweater) String() string {
return "Sweater"
}

type WoolenPant struct {
}

func (bottom WoolenPant) String() string {
return "Woolen Pant"
}

```

Next we create an interface to manage the _relatedness_ of clothes (summery or wintery)

```go
type ClothesFactory interface {
GetTop() Top
GetBottom() Bottom
}
```

We give concrete implementations to this interface and bring it all together.

```go
type SummerClothesFactory struct {
}

func (summer SummerClothesFactory) GetTop() Top {
return TankTop{}
}

func (summer SummerClothesFactory) GetBottom() Bottom {
return Shorts{}
}

type WinterClothesFactory struct {
}

func (winter WinterClothesFactory) GetTop() Top {
return Sweater{}
}

func (winter WinterClothesFactory) GetBottom() Bottom {
return WoolenPant{}
}
```

## Usage

```go
type Wardrobe struct {
clothesFactory ClothesFactory
}

func NewWardrobe(factory ClothesFactory) *Wardrobe {
return &Wardrobe{
clothesFactory: factory,
}
}

func (wardrobe Wardrobe) GetTop() Top {
return wardrobe.clothesFactory.GetTop()
}

func (wardrobe Wardrobe) GetBottom() Bottom {
return wardrobe.clothesFactory.GetBottom()
}


func main() {
summerWardrobe := NewWardrobe(SummerClothesFactory{})
fmt.Printf("Summer clothes are %s & %s", summerWardrobe.GetTop(), summerWardrobe.GetBottom())

fmt.Println()
winterWardrobe := NewWardrobe(WinterClothesFactory{})
fmt.Printf("Winter clothes are %s & %s", winterWardrobe.GetTop(), winterWardrobe.GetBottom())

}
```