【树莓派系列】设计模式:工厂模式的实现与创建(C语言实现)

设计模式:工厂模式的实现与创建(C语言实现)

一,设计模式概念引入

1. 什么是设计模式

设计模式通常被面向对象的软件开发人员所采用,是软件开发人员在软件开发过程中面临问题的解决方案。也是众多软件开发人员经过长时间的试验和错误总结出来的23种设计模式。虽然很多人认为C语言只是面向过程的开发,但鄙人认为此说法并不全面,难道面向对象的语言就不面向过程了吗?同理,在C语言的使用过程中一样允许面向对象。在平时代码编写中,有时会遇到修改一处BUG而影响整个代码的运行,设计模式的出现正好就解决了这一问题。

● 设计模式是建筑设计领域引入到计算机科学中来的;

● 设计模式是对面向对象设计中反复出现的问题的解决方案;

● 算法不是设计模式,因为算法致力于解决问题而非设计问题;

● 设计模式通常描述了一组相互紧密作用的类与对象;

● 设计模式使代码更稳定,更安全,拓展性更强,同时也方便代码重构。

2.什么是类和对象

面向过程和面向对象只是代码的一种编程方式和编程思想,不应该和编程语言绑定起来

C语言面向过程也可以面向对象,还能用其设计模式,只是不太友好的面向对象
Java面向对象也可以面向过程,只是不太友好的面向过程

:类是面对对象程序设计实现信息封装的基础。类是一种用户定义的引用数据类型,也称类类型。比如C语言中结构体,就是用户自己定义的一种数据类型

struct Animal{
      int age;
      int sex;         //成员属性
      void (*peat)();  //函数指针
      void (*pbeat)(); //成员方法
};

对象:类的一种具象。每个类包含数据说明和一组操作数据或传递消息的函数。类的实例就称为对象。

● dog是类Animal的一种具体对象;

● cat是类Animal的一种具体对象;

● person是类Animal的一种具体对象;

struct Animal dog;
struct Animal cat;
struct Animal person;

3.什么是工厂模式

● 工厂模式是最常用的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的(最佳)方式;

● 在工厂模式中,我们在创建对象时不会对客户端暴露创建逻辑,并且是通过使用一个共同的接口来指向新创建的对象;

● 共同的接口:我们自己创建共同接口API,从工厂里面获得需要的对象放入main函数中去运行。

image-20240905215718968

每一个功能模块都是要创建的对象,而每一个对象都可以单独创建一个代码文件,需要那个功能对象,就将相应的文件链入共同的接口内

二,C语言工厂模式的实现

1.普通类和对象的代码实现

#include <stdio.h>
 
/*类:struct Animal*/
/*对象:struct Animal dog/cat/person*/
 
struct Animal{
	char name[128];
    int age;
    int sex;             //成员属性
    void (*peat)();
    void (*pbeat)();	 //成员方法
};
   
void dogeat()
{
	printf("狗吃屎\n");
}
void cateat()
{
	printf("猫吃鱼\n");
}
void personeat()
{
	printf("人吃米\n");
}
 
void dogbeat()
{
	printf("咬\n");
}
void catbeat()
{
	printf("挠\n");
}
void personbeat()
{
	printf("揍\n");
}
 
int main()
{
	struct Animal dog = {
		.peat = dogeat,
		.pbeat = dogbeat
	};
	
	struct Animal cat = {
		.peat = cateat,
		.pbeat = catbeat
	};
	
	struct Animal person = {          
		.peat = personeat,    //对象,事务的具象
		.pbeat = personbeat
	}; 
	  
	dog.peat();//同样是吃饭,对象不一样
	cat.peat();
	person.peat();
	
	dog.pbeat();//同样是打架。对象不一样
	cat.pbeat();
	person.pbeat();
	
	return 0;
}

2.工厂模式代码实现

● 工厂模式不会暴露创建对象的代码逻辑,上面对象代码逻辑全部暴露无遗;

● 以上面的代码为例,以工厂模式的方式将dog,cat,person这几个对象拆分成单独的代码文件,并在每个文件中提供用来链接的函数,方便main函数的调用;

image-20240906215040103

● cat.c
#include "animal.h"  //自己定义的头文件用双引号

void cateat()
{
    printf("猫吃鱼\n");
}
void catbeat()
{
    printf("挠\n");
}

struct Animal cat = {
    .name = "Tom",
    .peat = cateat,
    .pbeat = catbeat
};

struct Animal* putCatInLink(struct Animal *phead) //头插法,会改变头,所以用到指针
{                                  //链表头传过来
    if(phead == NULL){  //如果头为空,那就返回cat为链表头           
        return &cat;
    }else{           
        cat.next = phead; //否者cat的下一个为链头
        phead = &cat;     //再把链头指向猫
        return phead;
    }
};		
● dog.c
#include "animal.h"

void dogeat()
{
    printf("狗吃屎\n");
}

void dogbeat()
{
    printf("咬\n");
}

struct Animal dog = {
    .name = "huang",
    .peat = dogeat,
    .pbeat = dogbeat
};

struct Animal* putDogInLink(struct Animal *phead)
{
    if(phead == NULL){
        return &dog;
    }else{
        dog.next = phead;
        phead = &dog;
        return phead;
    }
};		
● person.c
#include "animal.h"

void personeat()
{
    printf("人吃米\n");
}

void personbeat()
{
    printf("揍\n");
}


struct Animal person = {  
    .name = "xiaomin",
    .peat = personeat,    //对象,事务的具象
    .pbeat = personbeat
};


struct Animal* putPersonInLink(struct Animal *phead)
{
    if(phead == NULL){
        return &person;
    }else{
        person.next = phead;
        phead = &person;
        return phead;
    }
};		
● animal.h
#include <stdio.h>

struct Animal{
    char name[128];
    int age;
    int sex;             //成员属性
    void (*peat)();
    void (*pbeat)();	 //成员方法

    struct Animal *next; //涉及到链表,添加链表节点
};

struct Animal* putCatInLink(struct Animal *phead); //cat.c文件中的该函数将cat相关全局的结构体加到链表中 
struct Animal* putDogInLink(struct Animal *phead);//dog.c文件中的该函数将dog相关全局的结构体加到链表中
struct Animal* putPersonInLink(struct Animal *phead);//person.c文件中的该函数将person相关全局的结构体加到链表中
● mainpro.c
#include "animal.h"
 
int main()
{
	struct Animal *phead = NULL;//初始化
 
	phead = putCatInLink(phead);
	phead = putDogInLink(phead);
	phead = putPersonInLink(phead);
 
	return 0;
}

编译说明: 多个.c文件同时编译:*gcc .c, 如果编译过程中没有报错,但是没有输出内容,是因为在mainpro.c的代码中没有输出代码,但是链接成功

● 完善mainpro.c
#include "animal.h"
#include <string.h>

struct Animal *findUtilByName(char *str,struct Animal *phead){
    struct Animal *tmp = phead;

    if(phead == NULL){
        printf("空\n");
        return NULL;
    }else{
        while(tmp != NULL){  //遍历链表
            if(strcmp(tmp->name,str) == 0){ //tmp的值和输入的名字是相等的,说明找到了
                return tmp;
            } 
            tmp = tmp->next; //没找到的话就继续遍历链表
        }
        return NULL;//一直没找到就返回NULL
    }
}
int main()
{
    char buf[128] = {'\0'};
    struct Animal *phead = NULL;//初始化
    struct Animal *ptmp;

    phead = putCatInLink(phead);
    phead = putDogInLink(phead);
    phead = putPersonInLink(phead);

    while(1){
        printf("请输入:Tom,haung,xiaomin\n");
        scanf("%s",buf);
        ptmp = findUtilByName(buf,phead);//根据用户输入的名字,找到相应的结构体
        if(ptmp != NULL){
            ptmp->pbeat();
            ptmp->peat();
        }

        memset(buf,'\0',sizeof(buf)); 
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

橘猫.exe

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值