Wzorzec budowniczy

Zasada jednej odpowiedzialności

Wzorce projektowe – wzorzec budowniczy.

W dzisiejszych czasach jest wiele reguł, wzorców i zasad programowania, którymi kierują się obecni programiści. Żeby ich kod był zrozumiały i jasny dla innych oraz wydajny posługują się oprócz zasad SOLID, tzw. wzorcami projektowymi. Wzorce projektowe są to uniwersalne, sprawdzone w praktyce rozwiązanie często pojawiających się, powtarzalnych problemów projektowych. Wzorzec pokazuje powiązania i zależności pomiędzy klasami oraz obiektami i ułatwia tworzenie, modyfikację oraz utrzymanie kodu źródłowego. Jest opisem rozwiązania, a nie jego implementacją. Wzorce projektowe stosuje się w projektach wykorzystujących programowanie obiektowe. Jest wiele wzorców projektowych jednym z nich jest wzorzec budowniczy. W tym artykule opisze ten wzorzec oraz przedstawię przykład kodu programu przed zastosowaniem tego wzorca jak i po.

Wstęp

Wzorzec budowniczy z ang. Builder, używamy do hermetyzowania tworzenia produktu i w celu umożliwienia jego wieloetapowego inicjowania oraz rozdzieleniu sposobu tworzenia obiektów od ich reprezentacji. Innymi słowy proces tworzenia obiektu podzielony jest na kilka mniejszych etapów a każdy z tych etapów implementuje się na wiele sposobów. Dzięki takiemu rozwiązaniu możliwe jest tworzenie różnych reprezentacji obiektów w tym samym procesie konstrukcyjnym: sposób tworzenia obiektów zamknięty jest w oddzielnych obiektach zwanych konkretnymi budowniczymi. Wzorzec budowniczy różni się od innych wzorców kreacyjnych tym, że skupia się na sposobie tworzenia obiektów reprezentujących produkty. Wzorzec ten spełnia zasadę odwrócenia zależności oraz zasadę otwarte/zamknięte, otwarte na rozbudowę, zamknięte na modyfikację (łatwo do kodu można dodać nowych budowniczych). Należy do grupy wzorców skatalogowanych przez Gang czworga.

Zastosowanie

Wzorzec budowniczy stosuje się do oddzielenia sposobu tworzenia obiektów od tego jak te obiekty mają wyglądać. Przykładem jest oprogramowanie konwertujące tekst z jednego formatu na drugi. Algorytm odczytujący i interpretujący dane wejściowe oddziel się od algorytmu tworzącego dane wyjściowe. Dzięki takiemu rozwiązaniu możliwe jest zastosowanie jednego obiektu odczytującego dane wejściowe oraz wielu obiektów konwertujących odczytane dane do różnych formatów (ASCII, HTML, RTF, itp.), co zwiększa uniwersalność rozwiązania.

Innym przykładem gdzie można zastosować ten wzorzec jest aplikacja do sprzedaży turnusów wycieczkowych. Klienci mogą wybrać hotel i różne typy biletów, rezerwować miejsca w restauracjach, a także zamówić bilety na dodatkowe imprezy. Program turnusu dla klientów może się różnic w zależności od wyboru opcji w pakiecie. Potrzebna więc nam jest elastyczna struktura danych, która może reprezentować plan pojedynczego klienta w każdej potencjalnej odmianie.

Przykład

Poniżej znajduje się niepoprawny kod bez zastosowania wzorca budowniczy. Występuje tam inicjalizacja obiektu klasy Car. Program jest przeplatany logiką, w pewnych miejscach zostały użyte instrukcje warunkowe a gdzie indziej wzorzec prostej fabryki.

  1. var smallCar = new Car();  
  2. if (budget > 4500)  
  3. {  
  4.     smallCar.Wheels = "Aluminium rims 17 inches";  
  5. }  
  6. else  
  7. {  
  8.     smallCar.Wheels = "Steel rims 15 inches";  
  9. }  
  10. smallCar.Addons = new List<string>();  
  11. smallCar.Addons.Add("CD radio with MP3");  
  12. smallCar.Addons.Add("CD radio");  
  13. smallCar.Engine = EngineFactory.CreateEngine("120 HP engine");

Kod 1. Kod programu bez użycia wzorca

Program jest stosunkowo prosty, jednak gdyby trzeba było zbudować większy samochód, musielibyśmy skopiować cały ten kod i zmienić poszczególne parametry. Byłoby to podejście bardzo imperatywne, raczej mało obiektowe. W dużym projekcie wielu programistów konstruowałoby instancje klasy Car na swój sposób. Chcąc zmienić proces ich budowy, należałoby to zrobić w wielu miejscach.

Poniżej przestawiony jest kod programu z użyciem wzorca budowniczy:

  1. // kierownik - logika inicjalizacji obiektu  
  2. class CarDirector  
  3. {  
  4.     public void ConstructCar(ICarBuilder builder)  
  5.     {  
  6.         builder.BuildWheels();  
  7.         builder.BuildAddons();  
  8.         builder.BuildEngine();  
  9.     }  
  10. }  
  11.   
  12. // budowniczy - interfejs implementacji  
  13. interface ICarBuilder  
  14. {  
  15.     void BuildWheels();  
  16.     void BuildAddons();  
  17.     void BuildEngine();  
  18.     Car GetResult();  
  19. }  
  20.   
  21. // konkretny budowniczy - implementacja   
  22. class SmallCarBuilder : ICarBuilder  
  23. {  
  24.     private Car car = new Car();  
  25.     private EngineFactory engineFactory = new EngineFactory();  
  26.   
  27.     public void BuildWheels()  
  28.     {  
  29.         car.Wheels = "Steel rims 15 inches";  
  30.     }  
  31.   
  32.     public void BuildAddons()  
  33.     {  
  34.         car.Addons = new List();  
  35.         car.Addons.Add("CD radio with MP3");  
  36.         car.Addons.Add("CD radio");  
  37.     }  
  38.   
  39.     public void BuildEngine()  
  40.     {  
  41.         car.Engine = engineFactory.CreateEngine("120 HP engine");  
  42.     }  
  43.   
  44.     public Car GetResult()  
  45.     {  
  46.         return car;  
  47.     }  
  48. }  
  49.   
  50. static void Main(string[] args)  
  51. {  
  52.     CarDirector carDirector = new CarDirector();  
  53.     ICarBuilder smallCarBuilder = new SmallCarBuilder();  
  54.   
  55.     // buduj według logiki CarDirector używając implementacji z SmallCarBuilder  
  56.     carDirector.ConstructCar(smallCarBuilder);  
  57.   
  58.     Car smallCar = smallCarBuilder.GetResult();  

Kod 2. Kod programu z użyciem wzorca

Dzięki użyciu wzorca główna metoda programu znacząco się uprościła. Uzyskaliśmy jednolity interfejs polimorficzny służący do tworzenia instancji samochodu.

Podsumowanie

Podsumowując, wzorzec ten stosuje się głównie do budowania struktur kompozytowych. Posiada wiele zalet takich jak: hermetyzuje operacje niezbędne do stworzenia złożonego obiektu oraz umożliwia tworzenie obiektów w procedurze wielokrokowej i nie narzuca tej procedury. Ukrywa wewnętrzną reprezentację produktu przed klientem. Implementacje produktów można wymieniać, ponieważ klient korzysta wyłącznie z abstrakcyjnego interfejsu. Wadą jaką posiada ten wzorzec jest to, że tworzenie obiektów wymaga od klientów większej wiedzy z zakresu dziedziny problemu niż w przypadku stosowania wzorca Fabryka.