diff --git a/README.md b/README.md index 7d9f5f7..783b90f 100644 --- a/README.md +++ b/README.md @@ -45,7 +45,7 @@ A curated collection of idiomatic design & application patterns for Go language. | [State](/behavioral/state.md) | Encapsulates varying behavior for the same object based on its internal state | ✘ | | [Strategy](/behavioral/strategy.md) | Enables an algorithm's behavior to be selected at runtime | ✔ | | [Template](/behavioral/template.md) | Defines a skeleton class which defers some methods to subclasses | ✘ | -| [Visitor](/behavioral/visitor.md) | Separates an algorithm from an object on which it operates | ✘ | +| [Visitor](/behavioral/visitor.md) | Separates an algorithm from an object on which it operates | ✔ | ## Synchronization Patterns diff --git a/behavioral/visitor.md b/behavioral/visitor.md new file mode 100644 index 0000000..b6f3d11 --- /dev/null +++ b/behavioral/visitor.md @@ -0,0 +1,81 @@ +# Visitor Pattern +Visitor behavioral design pattern provides a way to separate an algorithm from an object on which it operates. + +It gives the ability to extend the existing object without modifying the object itself. + +## Implementation +Implementation of a visitor that can add functionality to the shape structures. + +```go +type Visitor interface { + visitCircle(circle Circle) string + visitLine(line Line) string +} + +type Shape interface { + accept(Visitor) string +} + +type Circle struct { + Rad int +} + +func (c Circle) accept(v Visitor) string { + return v.visitCircle(c) +} + +type Line struct { + Len int +} + +func (l Line) accept(v Visitor) string { + return v.visitLine(l) +} +``` + +## Usage +### JSON marshaller +Using visitor to marshal shapes to JSON +```go +type JsonVisitor struct { +} + +func (*JsonVisitor) visitCircle(circle Circle) string { + return fmt.Sprintf(`{"type": "circle", "radius": "%v"}`, circle.Rad) +} + +func (*JsonVisitor) visitLine(line Line) string { + return fmt.Sprintf(`{"type": "line", "length": "%v"}`, line.Len) +} +``` + +```go +circle := Circle{12} +line := Line{42} +jsonVisitor := JsonVisitor{} +fmt.Println(circle.accept(&jsonVisitor)) +fmt.Println(line.accept(&jsonVisitor)) +``` + +### XML marshaller +Using visitor to marshal shapes to XML +```go +type XmlVisitor struct { +} + +func (*XmlVisitor) visitCircle(circle Circle) string { + return fmt.Sprintf(`%d`, circle.Rad) +} + +func (*XmlVisitor) visitLine(line Line) string { + return fmt.Sprintf(`%d`, line.Len) +} +``` + +```go +circle := Circle{12} +line := Line{42} +xmlVisitor := XmlVisitor{} +fmt.Println(circle.accept(&xmlVisitor)) +fmt.Println(line.accept(&xmlVisitor)) +``` diff --git a/behavioral/visitor/main.go b/behavioral/visitor/main.go new file mode 100644 index 0000000..db9aa93 --- /dev/null +++ b/behavioral/visitor/main.go @@ -0,0 +1,64 @@ +package main + +import ( + "fmt" +) + +type Visitor interface { + visitCircle(circle Circle) string + visitLine(line Line) string +} + +type Shape interface { + accept(Visitor) string +} + +type Circle struct { + Rad int +} + +func (c Circle) accept(v Visitor) string { + return v.visitCircle(c) +} + +type Line struct { + Len int +} + +func (l Line) accept(v Visitor) string { + return v.visitLine(l) +} + +type JsonVisitor struct { +} + +func (*JsonVisitor) visitCircle(circle Circle) string { + return fmt.Sprintf(`{"type": "circle", "radius": "%v"}`, circle.Rad) +} + +func (*JsonVisitor) visitLine(line Line) string { + return fmt.Sprintf(`{"type": "line", "length": "%v"}`, line.Len) +} + +type XmlVisitor struct { +} + +func (*XmlVisitor) visitCircle(circle Circle) string { + return fmt.Sprintf(`%d`, circle.Rad) +} + +func (*XmlVisitor) visitLine(line Line) string { + return fmt.Sprintf(`%d`, line.Len) +} + +func main() { + circle := Circle{12} + line := Line{42} + jsonVisitor := JsonVisitor{} + fmt.Println(circle.accept(&jsonVisitor)) + fmt.Println(line.accept(&jsonVisitor)) + + xmlVisitor := XmlVisitor{} + fmt.Println(circle.accept(&xmlVisitor)) + fmt.Println(line.accept(&xmlVisitor)) +}