24/05/2018, 19:06

Mẫu kiến tạo (Creational Pattern)

Những mẫu này hỗ trợ cho một trong những nhiệm vụ của lập trình hướng đối tượng – khởi tạo đối tượng trong hệ thống. Hầu hết các hệ thống hướng đối tượng phức tạp yêu cầu nhiều đối tượng được thể hiện theo thời gian, và các mẫu này hỗ trợ cho ...

Những mẫu này hỗ trợ cho một trong những nhiệm vụ của lập trình hướng đối tượng – khởi tạo đối tượng trong hệ thống. Hầu hết các hệ thống hướng đối tượng phức tạp yêu cầu nhiều đối tượng được thể hiện theo thời gian, và các mẫu này hỗ trợ cho việc tạo các tiến trình bằng việc cung cấp các khả năng:

- Sự thể hiện chung – Điều này cho phép các đối tượng được tạo ra trong hệ thống không cần phải định nghĩa một đặc tả kiểu lớp trong mã nguồn

- Đơn giản – Một vài mẫu làm cho việc khởi tạo đối tượng trở nên dễ dàng, vì vậy lớp “gọi” khởi tạo đối tượng không phải viết mã nhiều cũng như phức tạp.

Abstract Factory Method Pattern- Ý nghĩa

Đóng gói một nhóm những lớp đóng vai trò “sản xuất” (Factory) trong ứng dụng, đây là những lớp được dùng để tạo lập các đối tượng. Các lớp sản xuất này có chung một giao diện lập trình được kế thừa từ một lớp cha thuần ảo gọi là “lớp sản xuất ảo”.

- Cấu trúc mẫu

Trong đó:

+ AbstractFactory: là lớp trừu tượng, tạo ra các đối tượng thuộc 2 lớp trừu tượng là: AbstractProductA và AbstractProductB

+ ConcreteFactoryX: là lớp kế thừa từ AbstractFatory, lớp này sẽ tạo ra một đối tượng cụ thể.

+ AbstractProduct: là các lớp trừu tượng, các đối tượng cụ thể sẽ là các thể hiện của các lớp dẫn xuất từ lớp này.- Tình huống áp dụng

+ Phía trình khách sẽ không phụ thuộc vào việc những sản phẩm được tạo ra như thế nào.

+ Ứng dụng sẽ được cấu hình với một hoặc nhiều họ sản phẩm.

+ Các đối tượng cần phải được tạo ra như một tập hợp để có thể tương thích với nhau.

+ Chúng ta muốn cung cấp một tập các lớp và chúng ta muốn thể hiện các ràng buộc, các mối quan hệ giữa chúng mà không phải là các thực thi của chúng(interface).Ví dụ 6.1

Giả sử ta cần viết một ứng dụng quản lý địa chỉ và số điện thoại cho các quốc gia trên thế giới. Điạ chỉ và số địa thoại của mỗi quốc gia sẽ có 1 số điểm giống nhau và 1 số điểm khác nhau. Ta xây dựng sơ đồ lớp như sau:

Ta sẽ xây dựng các phương thức tạo Address, và PhoneNumber cụ thể trong các lớp USAAddressPhoneFactory, FrechAddressPhoneFactory.

Với phương thực createProductAddress() của lớp AAddressPhoneFactory sẽ trả về đối tượng của lớp USAAddress.

Với phương thực createProductAddress() của lớp FrechAddressPhoneFactory sẽ trả về đối tượng của lớp FrechAddressTương tự với PhoneNumber.

AddressFactory.java

public interface AddressFactory {

public Address createAddress();

public PhoneNumber createPhoneNumber();

}

Address.java

public abstract class Address {

private String street;

private String city;

private String region;

private String postalCode;

public static final String EOL_STRING =

System.getProperty("line.separator");

public static final String SPACE = " ";

public String getStreet() {

return street; }

public String getCity() {

return city; }

public String getPostalCode() {

return postalCode; }

public String getRegion() {

return region; }

public abstract String getCountry();

public String getFullAddress() {

return street + EOL_STRING + city + SPACE + postalCode + EOL_STRING; }

public void setStreet(String newStreet) {

street = newStreet; }

public void setCity(String newCity) {

city = newCity; }

public void setRegion(String newRegion) {

region = newRegion; }

public void setPostalCode(String newPostalCode) {

postalCode = newPostalCode; }

}

USAddressFactory.java

public class USAddressFactory implements AddressFactory{

public Address createAddress(){

return new USAddress(); }

public PhoneNumber createPhoneNumber(){

return new USPhoneNumber();

}

}

USAddress.java

public class USAddress extends Address{

private static final String COUNTRY = "UNITED STATES";

private static final String COMMA = ",";

public String getCountry(){

return COUNTRY;

}

public String getFullAddress(){

return getStreet() + EOL_STRING + getCity() + COMMA + SPACE + getRegion() + SPACE + getPostalCode() + EOL_STRING + COUNTRY + EOL_STRING;

}

}

Tương tự cho lớp PhoneNumber và USAPhoneNumber

Builder Pattern- Ý nghĩa

Phân tách những khởi tạo các thành phần của một đối tượng phức hợp, để có thể cùng một khởi tạo mà có thể tạo nên nhiều định dạng khác nhau.

- Cấu trúc mẫu

Trong đó:

+ Director: là lớp điều khiển tạo ra một đối tượng Product.

+ Builder: là lớp trừu tượng cho phép tạo ra đối tượng Product từ các phương thức nhỏ khởi tạo từng thành phần của Product

+ ConcreteBuilder: là lớp dẫn xuất của Builder, khởi tạo từng đối tượng cụ thể, lớp này sẽ khởi tạo đối tượng.

- Tình huống áp dụng

+ Có cấu trúc bên trong phức tạp (đặc biệt là một biến là một tập các đối tượng liên quan với nhau).

+ Có các thuộc tính phụ thuộc vào các thuộc tính kháco Sử dụng các đối tượng khác trong hệ thống mà có thể khó khởi tạo hoặc khởi tạo phức tạp.

- Ví dụ 6.2

Ta lại xét đối tượng Address, có các thành phần sau: Street, City và Region. Ta phân tách việc khởi tạo 1 đối tượng Address thành các phần : buildStreet, buildCity và buildRegion.

Trong đó:

+ AddressDirector: là lớp tạo ra đối tượng Address.

+ AddressBuilder: là lớp trừu tượng cho phép tạo ra 1 đối tượng Address bằng các phương thức khởi tạo từng thành phần của Addresso USAddressBuilder: là lớp tạo ra các Address. USAddressBuilder sẽ tạo ra địa chỉ theo chuẩn của USA.

Address.java

class Address(){

private String street;

private String city;

private String region;

/**

* @return the city

*/

public String getCity() {

return city;

}

/**

* @param city the city to set

*/

public void setCity(String city) {

this.city = city;

}

/**

* @return the region

*/

public String getRegion() {

return region;

}

/**

* @param region the region to set

*/

public void setRegion(String region) {

this.region = region;

}

/**

* @return the street

*/

public String getStreet() {

return street;

}

/** @param street the street to set **/

public void setStreet(String street) {

this.street = street;

}

}

AddressBuilder.java

abstract class AddressBuilder{

abstract public void buildStreet(String street){}

abstract public void buildCity(String city){}

abstract public void buildRegion(String region){}

}

USAddressBuilder.java

class USAddressBuilder extends AddressBuilder {

private Address add;

public void buildStreet(String street){

add.setStreet(street);

}

public void buildCity(String city){

add.setCity(city);

}

public void buildRegion(String region){

add.setRegion(region);

}

public Address getAddress(){

return add;

} }

AddressDirector.java

class AddressDirector{

public void Contruct(AddressBuilder builder, String street, String city, String region){

builder.buildStreet(street);

builder.buildCity(city);

builder.buildRegion(region);

}

}

Client.java

class Client{

AddressDirector director = new AddressDirector();

USAddressBuilder b = new USAddressBuilder();

director.Contruct(b. “Street 01”, “City 01”, “Region 01”);

Address add = b.getAddress();

}

0