-
Tìm các mẫu thiết kế trong mã nguồn của các dự án có trên 1000 sao trên github.
-
So sánh các mẫu tìm được với mẫu chuẩn của 23 mẫu thiết kế.
-
Viết báo cáo ở dạng file .md và nộp bằng pull request đến repo của giáo viên.
-
REPO_PATTERNLOCKVIEW Repo của tác giả aritraroy với 2700 sao trên github
-
REPO_CPPDESIGNPATTERN Repo của tác giả liu-jianhao với với 1000 sao trên github
-
20021395 Nguyễn Quang Minh
-
20021387 Nguyễn Phúc Long
-
20021334 Đào Văn Đức
https://github.com/minh16022002/NguyenQuangMinh
Các thành viên nộp vào repo của nhóm, sau đó nhóm trưởng tổng hợp lại và gửi lên repo của giáo viên.
Sau khi tham khảo 2 repo trên, nhóm em tiến hành so sánh với các mẫu thiết kế:
Những Design pattern loại này cung cấp một giải pháp để tạo ra các object và che giấu được logic của việc tạo ra nó, thay vì tạo ra object một cách
trực tiếp bằng cách sử dụng method new. Điều này giúp cho chương trình trở nên mềm dẻo hơn trong việc quyết định object nào cần được tạo ra trong những tình huống được đưa ra.
Trong package com.andrognito.rxpatternlockview.events
của repo số 1:
Abstract class BasePatternLockEvent
được kế thừa bởi các class sau:
class PatternLockCompleteEvent
class PatternLockCompoundEvent
class PatternLockProgressEvent
Trong package com.andrognito.rxpatternlockview.observables
của repo số 1:
Abstract class BasePatternLockViewObservable
được kế thừa bởi các class sau:
class PatternLockViewCompleteObservable
class PatternLockViewCompoundObservable
class PatternLockViewProgressObservable
Việc tạo ra những Super-factory này dùng để tạo ra các Factory tiếp theo rất giống với mẫu thiết kế Abstract Factory.
Trong Abstract Factory pattern, một interface có nhiệm vụ tạo ra một Factory của các object có liên quan tới nhau mà không cần phải chỉ ra trực tiếp các class của object. Mỗi Factory được tạo ra có thể tạo ra các object bằng phương pháp giống như Factory pattern.
Trong package com.andrognito.patternlockview.utils
của repo số 1:
Mô tả: đảm bảo chỉ duy nhất một thể hiện (instance) được tạo ra và nó sẽ cung cấp cho bạn một method để có thể truy xuất được thể hiện duy nhất đó mọi lúc mọi nơi trong chương trình.
Các lớp dưới đây đều chỉ có một thể hiện và chỉ cung cấp một method có thể truy xuất được thể hiện duy nhất đó
class PatternLockUtils
private PatternLockUtils() {
throw new AssertionError("You can not instantiate this class. Use its static utility " +
"methods instead");
}
class RandomUtils
private RandomUtils() {
throw new AssertionError("You can not instantiate this class. Use its static utility " +
"methods instead");
}
class ResourceUtils
private ResourceUtils() {
throw new AssertionError("You can not instantiate this class. Use its static utility " +
"methods instead");
}
Khác so với Singleton: Các class vẫn tồn tại những hàm ngoài luồng, dùng để xử lý ngoại lệ, đôi khi khó nhận biết
Mô tả: Tách rời việc xây dựng (construction) một đối tượng phức tạp khỏi biểu diễn của nó sao cho cùng một tiến trình xây dựng có thể tạo được các biểu diễn khác nhau.
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Builder
Đối tượng House được biểu diễn bằng cách tách thành nhiều biểu diễn nhỏ và ít phức tạp hơn như Builder, Director:
class House
class HouseBuilder
class HouseDirector
class StoneHouse
class StoneHouseBuilder
Mẫu thiết kế này khá giống với mẫu thiết kế chuẩn.
Mô tả: Prototype cho phép sao chép các đối tượng hiện có mà không làm cho code của bạn phụ thuộc vào các lớp của chúng.
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Prototype
Thể hiện việc cho phép các đối tượng hiện có được sao chép mà không làm cho code bị phụ thuộc:
class BinarySplitter
class TxtSplitter
class PictureSplitter
class VideoSplitter
class ISplitter{
public:
virtual void split()=0;
virtual ISplitter* clone()=0;
virtual ~ISplitter(){}
};
Những Design pattern loại này liên quan tới class và các thành phần của object. Nó dùng để thiết lập, định nghĩa quan hệ giữa các đối tượng.
Mô tả: Các interface không liên quan tới nhau có thể làm việc cùng nhau.
Adapter được thể hiện trong repo số 1:
Nhóm các class dưới đây tuy mang những chức năng khác nhau nhưng vẫn làm việc cùng nhau, kế thừa từ class io.reactivex.Observable
class PatternLockViewCompleteObservable
class PatternLockViewCompoundObservable
class PatternLockViewProgressObservable
Sau khi so sánh với mẫu thiết kế Adapter, nhóm mình thấy việc sử dụng mẫu thiết kế này khá phổ biến và có nhiều biến thể của chúng, chẳng hạn như việc các lớp có thể tuy cùng kế thừa từ một class, có mục đích sử dụng khác nhau, nhưng vẫn hoạt động và làm việc cùng nhau bình thường.
Mô tả: Tách tính trừu tượng ra khỏi tính hiện thực của nó. Từ đó có thể dễ dàng chỉnh sửa hoặc thay thế mà không làm ảnh hưởng đến những nơi có sử dụng lớp ban đầu -> cả hai bộ phận có thể thay đổi độc lập nhau. Nó giúp giảm sự phụ thuộc giữa trừu tượng và hiện thực, giảm số lượng những lớp con không cần thiết, code gọn gàng, dễ bảo trì, mở rộng về sau.
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Bridge
Đoạn code minh hoạ cho việc tách tính trừu tượng ra khỏi tính hiện thực trong class Messager
class Messager{
public:
virtual void Login(string username, string password)=0;
virtual void SendMessage(string message)=0;
virtual void SendPicture(Image image)=0;
virtual void PlaySound()=0;
virtual void DrawShape()=0;
virtual void WriteText()=0;
virtual void Connect()=0;
virtual ~Messager(){}
};
class PCMessagerBase
class MobileMessagerBase
class PCMessagerLite
class PCMessagerPerfect
class MobileMessagerLite
class MobileMessagerPerfect
Mô tả:Tổ chức các đối tượng theo cấu trúc phân cấp dạng cây. Tất cả các đối tượng trong cấu trúc được thao tác theo một cách thuần nhất như nhau.
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Composite
Minh hoạ:
class Component {
public:
virtual void process() = 0;
virtual ~Component(){}
};
class Composite
class Leaf : public Component{
string name;
public:
Leaf(string s) : name(s) {}
void process(){
//process current node
}
};
Mô tả: Gán thêm trách nhiệm cho đối tượng (mở rộng chức năng) vào lúc chạy (dynamically) mà không làm thay đổi cấu trúc lớp của nó. Decorator thể hiện như là đóng gói của một lớp đã tồn tại.
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Decorator-Pattern
Minh hoạ cho Decorator Pattern:
class Stream{
public:
virtual char Read(int number)=0;
virtual void Seek(int position)=0;
virtual void Write(char data)=0;
virtual ~Stream(){}
};
class FileStream: public Stream{
public:
virtual char Read(int number){
//读文件流
}
virtual void Seek(int position){
//定位文件流
}
virtual void Write(char data){
//写文件流
}
};
class NetworkStream :public Stream{
public:
virtual char Read(int number){
}
virtual void Seek(int position){
}
virtual void Write(char data){
}
};
class MemoryStream :public Stream{
public:
virtual char Read(int number){
}
virtual void Seek(int position){
}
virtual void Write(char data){
}
};
class CryptoFileStream
class CryptoNetworkStream
class CryptoMemoryStream
class BufferedFileStream
class BufferedNetworkStream
class BufferedMemoryStream
class CryptoBufferedFileStream
Từ các class minh hoạ ta thấy mẫu thiết kế này khá giống với mẫu thiết kế chuẩn.
Mô tả: Phương pháp trả về các đối tượng được lưu trong bộ nhớ đệm thay vì tạo mới.
class Font {
private:
string key;
public:
Font(const string& key){
//...
}
};
class FontFactory{
private:
map<string,Font* > fontPool;
public:
Font* GetFont(const string& key){
map<string,Font*>::iterator item=fontPool.find(key);
if(item!=footPool.end()){
return fontPool[key];
}
else{
Font* font = new Font(key);
fontPool[key]= font;
return font;
}
}
void clear(){
//...
}
};
Mô tả: Ủy quyền tất cả công việc thực sự cho một số đối tượng khác và mỗi phương thức proxy phải tham chiếu đến một đối tượng dịch vụ trừ khi proxy là một lớp con của một dịch vụ.
Code minh hoạ:
class ISubject{
public:
virtual void process();
};
class SubjectProxy: public ISubject{
public:
virtual void process(){
}
};
class ClientApp{
ISubject* subject;
public:
ClientApp(){
subject=new SubjectProxy();
}
void DoTask(){
subject->process();
}
};
Nhóm pattern này liên quan đến thuật toán và sự phân công công việc giữa các nhóm đối tượng.
Mô tả: Chain of Responsibility cho phép chuyển các yêu câu dọc theo chuỗi xử lý. Khi nhận được yêu cầu, mỗi trình xử lý sẽ quyết định xử lý yêu cầu hoặc chuyển cho trình xử lý khác. Pattern này hoạt động như một danh sách liên kết với việc đệ quy duyệt qua các phần tử, sử dụng khi chương trình cần xử lý nhiều loại yêu cầu khác nhau, nhưng chưa chưa xác định được loại yêu cầu và cách sắp xếp tuần tự của nó. Code minh họa:
class Reqest
{
string description;
RequestType reqType;
public:
Reqest(const string & desc, RequestType type) : description(desc), reqType(type) {}
RequestType getReqType() const { return reqType; }
const string& getDescription() const { return description; }
};
class ChainHandler{
ChainHandler *nextChain;
void sendReqestToNextHandler(const Reqest & req)
{
if (nextChain != nullptr)
nextChain->handle(req);
}
protected:
virtual bool canHandleRequest(const Reqest & req) = 0;
virtual void processRequest(const Reqest & req) = 0;
public:
ChainHandler() { nextChain = nullptr; }
void setNextChain(ChainHandler *next) { nextChain = next; }
void handle(const Reqest & req)
{
if (canHandleRequest(req))
processRequest(req);
else
sendReqestToNextHandler(req);
}
};
class Handler1 : public ChainHandler{
protected:
bool canHandleRequest(const Reqest & req) override
{
return req.getReqType() == RequestType::REQ_HANDLER1;
}
void processRequest(const Reqest & req) override
{
cout << "Handler1 is handle reqest: " << req.getDescription() << endl;
}
};
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Chain%20of%20Resposibility
Mô tả: Command Pattern cho phép biến một yêu cầu thành một đối tượng độc lập và chứa đầy đủ thông tin về yêu cầu đó. Chuyển đổi này giúp truyền các yêu cầu dưới dạng tham số của phương thức, trì hoãn hoặc chờ đợi việc thực thi một yêu cầu hay hỗ trợ các hoạt động hoàn tác. Pattern này được sử dụng khi cần tham số hóa các đối tượng theo một hành động thực hiện hoặc khi cần tạo và thực thi các yêu cầu vào các thời điểm khác nhau. Code minh họa:
class Command
{
public:
virtual void execute() = 0;
};
class ConcreteCommand1 : public Command
{
string arg;
public:
ConcreteCommand1(const string & a) : arg(a) {}
void execute() override
{
cout<< "#1 process..."<<arg<<endl;
}
};
class ConcreteCommand2 : public Command
{
string arg;
public:
ConcreteCommand2(const string & a) : arg(a) {}
void execute() override
{
cout<< "#2 process..."<<arg<<endl;
}
};
class MacroCommand : public Command
{
vector<Command*> commands;
public:
void addCommand(Command *c) { commands.push_back(c); }
void execute() override
{
for (auto &c : commands)
{
c->execute();
}
}
};
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Command
Mô tả: Mẫu này được sử dụng để truy cập vào các phần tử của 1 collection (array, map, set, tree...) một cách tuần tự mà không cần quan tâm tới cấu trúc của nó. Iterator được sử dụng để giảm thiểu các mã trung lập khi duyệt phần tử. Code minh họa:
template<typename T>
class Iterator
{
public:
virtual void first() = 0;
virtual void next() = 0;
virtual bool isDone() const = 0;
virtual T& current() = 0;
};
template<typename T>
class MyCollection{
public:
Iterator<T> GetIterator(){
//...
}
};
template<typename T>
class CollectionIterator : public Iterator<T>{
MyCollection<T> mc;
public:
CollectionIterator(const MyCollection<T> & c): mc(c){ }
void first() override {
}
void next() override {
}
bool isDone() const override{
}
T& current() override{
}
};
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/blob/master/Iterator/Iterator.cpp
Mô tả: Mediator Patern (mẫu hình trung gian) được sử dụng để giảm sự phức tạp trong “giao tiếp” giữa các lớp và các đối tượng. Mô hình này cung cấp một lớp trung gian có nhiệm vụ xử lý thông tin liên lạc giữa các tầng lớp, hỗ trợ bảo trì mã code dễ dàng bằng cách khớp nối lỏng lẻo. Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Mediator
Mô tả: Memento là mẫu thiết kế có thể lưu lại trạng thái của một đối tượng để khôi phục lại (Undo) sau này mà không vi phạm nguyên tắc đóng gói. Dữ liệu trạng thái đã lưu trong đối tượng memento không thể truy cập bên ngoài đối tượng được lưu và khôi phục. Điều này bảo vệ tính toàn vẹn của dữ liệu trạng thái đã lưu. Code minh họa:
class Memento
{
string state;
//..
public:
Memento(const string & s) : state(s) {}
string getState() const { return state; }
void setState(const string & s) { state = s; }
};
class Originator
{
string state;
//....
public:
Originator() {}
Memento createMomento() {
Memento m(state);
return m;
}
void setMomento(const Memento & m) {
state = m.getState();
}
};
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/blob/master/Memento/memento.cpp
Mô tả: Mẫu Observer tạo mối liên hệ one-to-many giữa subject và các observer với nhau(chẳng hạn 1 subject sẽ có thuộc tính là một mảng bao gồm nhiều observer) nên khi trạng thái của subject thay đổi, tất cả các observer liên kết với subject này sẽ được thông báo và tự động cập nhật. Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Observer
Mô tả: Mẫu State cho phép một đối tượng thay đổi hành vi của nó khi trạng thái nội bộ của nó thay đổi. Đối tượng sẽ xuất hiện để thay đổi lớp của nó. Mẫu này được sử dụng khi hành vi của đối tượng phụ thuộc vào trạng thái của nó và nó phải có khả năng thay đổi hành vi của nó lúc run-time theo trạng thái mới. Code minh họa:
type MobileAlertState interface {
alert()
}
type AlertStateContext struct {
currentState MobileAlertState
}
func NewAlertStateContext() *AlertStateContext {
return &AlertStateContext{currentState: &Vibration{}}
}
func (ctx *AlertStateContext) SetState(state MobileAlertState) {
ctx.currentState = state
}
func (ctx *AlertStateContext) Alert() {
ctx.currentState.alert()
}
type Vibration struct{}
func (v *Vibration) alert() {
fmt.Println("vibrating....")
}
type Silence struct{}
func (s *Silence) alert() {
fmt.Println("silent ....")
}
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/State
Mô tả: Là mẫu thiết kế cho phép chọn thuật toán trong 1 nhóm các thuật toán liên quan đến nhau ngay tại lúc chương trình đang chạy để thực hiện một hoạt động nào đó. Code minh họa:
class SalesOrder{
TaxBase tax;
public:
double CalculateTax(){
//...
if (tax == CN_Tax){
//CN***********
}
else if (tax == US_Tax){
//US***********
}
else if (tax == DE_Tax){
//DE***********
}
else if (tax == FR_Tax){ //¸ü¸Ä
//...
}
//....
}
};
class TaxStrategy{
public:
virtual double Calculate(const Context& context)=0;
virtual ~TaxStrategy(){}
};
class CNTax : public TaxStrategy{
public:
virtual double Calculate(const Context& context){
//***********
}
};
class USTax : public TaxStrategy{
public:
virtual double Calculate(const Context& context){
//***********
}
};
class DETax : public TaxStrategy{
public:
virtual double Calculate(const Context& context){
//***********
}
};
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Strategy-Pattern
Mô tả: Mẫu Template Method cho phép lớp con định nghĩa lại cách thực hiện của một thuật toán, mà không phải thay đổi cấu trúc thuật toán”. Điều này có nghĩa là Template method giúp cho chúng ta tạo nên một bộ khung (template) cho một vấn đề đang cần giải quyết. Trong đó các đối tượng cụ thể sẽ có cùng các bước thực hiện, nhưng trong mỗi bước thực hiện đó có thể khác nhau. Code minh họa:
class Library
{
public:
void Step1()
{
cout << "Step1" << endl;
}
void Step3()
{
cout << "Step3" << endl;
}
void Step5()
{
cout << "Step5" << endl;
}
};
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Template%20Method
Mô tả: Mẫu Visitor cho phép định nghĩa các thao tác (operations) trên một tập hợp các đối tượng (objects) không đồng nhất (về kiểu) mà không làm thay đổi định nghĩa về lớp (classes) của các đối tượng đó. Code minh họa:
class Visitor;
class Element
{
public:
virtual void accept(Visitor& visitor) = 0; //第一次多态辨析
virtual ~Element(){}
};
class ElementA : public Element
{
public:
void accept(Visitor &visitor) override {
visitor.visitElementA(*this);
}
};
class ElementB : public Element
{
public:
void accept(Visitor &visitor) override {
visitor.visitElementB(*this); //第二次多态辨析
}
};
class Visitor{
public:
virtual void visitElementA(ElementA& element) = 0;
virtual void visitElementB(ElementB& element) = 0;
virtual ~Visitor(){}
};
Đường dẫn: https://github.com/liu-jianhao/Cpp-Design-Patterns/tree/master/Visitor
Sau khi tìm hiểu các mẫu thiết kế trong hai repo trên, nhóm chúng mình xin đưa ra những kết luận chung của bài báo cáo:
-
Đa phần các mẫu thiết kế được sử dụng khá giống các mẫu thiết kế chuẩn, có sửa đổi linh hoạt để phù hợp với dự án.
-
Lợi ích của việc vận dụng được các mẫu thiết kế vào dự án:
- Đơn giản hoá mã nguồn.
- Giảm thời gian và công sức suy nghĩ các cách phân bố và cấu trúc chương trình.
- Dễ dàng kiểm tra, nâng cấp mã nguồn.
Cảm ơn thầy và các bạn đã đọc và nhận xét. Phần báo cáo của nhóm mình còn nhiều thiếu sót, mong thầy và các bạn thông cảm.