使用了继承、多态还有工厂模式和反射,但是还是没有OO的感觉。[已经增加了实现的代码]

简介: 最近项目里遇到了一个问题,为了解决这个问题“动用了”继承、多态还有工厂模式和反射,但是还是没有OO的感觉。呵呵。 先说一下具体情况: 1、使用短信猫来接收短信。简单的说,短信猫收到短信后会往指定的表里面填写数据。
最近项目里遇到了一个问题,为了解决这个问题“动用了” 继承、多态还有工厂模式和反射,但是还是没有OO的感觉。呵呵。

先说一下具体情况:
1、使用短信猫来接收短信。简单的说,短信猫收到短信后会往指定的表里面填写数据。
2、接收到短信,然后根据短信前面的“标志”调用不同的方式来处理。
3、发送确认信息或者是错误提示。
4、想做成一个“通用”的模块,不管是什么项目,都可以使用这个程序来处理接收短信的问题。当然具体的处理方式要能很方便的修改或者扩充。


我的实现方式:
1、定义一个基类,用来处理接收到的短信。
2、由于每一种短信的处理方式都不同,所以需要好多的不同的子类。每一个子类来处理一种短信。
3、调用的时候 如果用 case 的方式的话,每增加一总短信都要修改case 。很烦!而且当应用在一个项目里的时候,case 就得推倒重来,烦!
     最后使用了反射。
4、建立了三个项目,一个是winform的用来检查表里面是否有新的短信,叫做A ;一个是处理短信的项目,叫做B;最后一个就是具体的项目了。

==============
说明:
 具体的项目指的是,可能是A公司的OA,可能是B公司的CRM,也可能是C公司的ERP。
 这里说的是分“项目”,而不是分层。
 处理短信的项目要根据不同的公司的不同的需求来编写,写完了之后编译成DLL,交由 A 来调用。
==============

5、其中 A 是通用的,写好了基本不变。B编译成dll,好让A来调用。
6、A 调用处理短信的类。由于使用了反射,可以“动态”的指定dll名称和 类名。这样就很灵活了,处理短消息的方式有变化的话,只需要更新dll就可以了。
  换成新的项目的时候,换成新的dll就可以了。这样A就不用改了。


现在基本功能是实现了,但是这样就OO了吗?还是没有OO感觉。

感觉还是在用面向过程的思路来写程序,一个子类里面只有一个函数,和面成过程有什么区别呢?

您可能要问了,那我为什么还要用多态呢?其实很简单,这样就可以使用“反射”了,这样我就不用写case了,可以让A不必随不同的项目而修改了。

说白了就是想当变化的时候少改点代码。

正在看面向对象、设计模式了什么的,把自己的想法、做法写出来,请大家批批。


ps:这里好象用“观察者”更好一点,A就是一个发布者,B是一个订阅者,只是我不知道如何让B来订阅A。
也许根本就不适合吧,毕竟有新的短信了,只有一种处理方法是对应的,其他的都不是。

代码补充:

namespace  HBS.SMSReceive
{
    
/**//// <summary>
    
/// 接收短信。基类
    
/// </summary>

    public class MessageReceive
    
{
        
public DataAccessLayer dal ;
        
        
处理短信的函数#region 处理短信的函数
        
/**//// <summary>
        
/// 接收短消息然后作相应的处理
        
/// </summary>
        
/// <param name="Mobile">传入手机号</param>
        
/// <param name="RecvDate">传入收到短信的时间</param>
        
/// <param name="Msg">短信内容</param>

        public virtual string SaveMsg(string Mobile ,string RecvDate,string Msg)
        
{
            
return "";        //表示正常执行。否则表示出错信息。需要把这个信息发给发送者
        }

        
#endregion


        
把短信移动到历史记录里面#region 把短信移动到历史记录里面
        
/**//// <summary>
        
/// 把短消息从inbox 表中移动到历史记录表里面
        
/// </summary>

        public virtual void LogMsg(string InboxID)
        
{
            
//复制记录
            string sql = "insert into InBox_Log (mbno,Msg,ArriveTime,Readed,username,comport) select mbno,Msg,ArriveTime,Readed,username,comport from InBox where id= " + InboxID;
            dal.RunSql(sql);

            
//删除记录
            dal.RunSql("delete from InBox where id=" + InboxID);

        }

        
#endregion

    }


    
子类1:处理没有标志的短信#region 子类1:处理没有标志的短信
    
/**//// <summary>
    
/// 没有标志的短信
    
/// </summary>

    public class MessageOperationOther:MessageReceive
    
{
        
public override string SaveMsg(string Mobile ,string RecvDate,string Msg)
        
{
            
return "您发的短信我们无法正确识别,请核对后再次发送,谢谢合作!";
        }

    }

    
#endregion


    
子类2:MessageOperation01 第一种短信的回执#region 子类2:MessageOperation01 第一种短信的回执
    
public class MessageOperation01:MessageReceive
    
{
        
public override string SaveMsg(string Mobile ,string RecvDate,string Msg)
        
{
            
//处理第一种短信,代码略

            
return "";
        }

    }

    
#endregion

}



然后就是A里面的调用的代码

//获取短信内容,放在 DataTable dt 里面
//然后遍历 dt

Assembly.Load("SMS").CreateInstance("SMS.短信" + 短信开头的编号)

代码补充:
处理接收到的短信 #region 处理接收到的短信
        
private void monitorInSMS()
        
{
            
string strSQL = "select * from inbox ";    
            DataTable dt 
= dal.RunSqlDataTable(strSQL);
            
string Mobile = "";            //去掉 86 的手机号
            string[] msg = null;        //短信的类型。
            string re = "";                //处理短信后的结果。""表示正确执行。
            
            
//遍历短信
            foreach (DataRow dr in dt.Rows)
            
{
                Mobile 
= dr["mbno"].ToString();
                
if (Mobile.Length > 11)                //去掉前面的86
                    Mobile = Mobile.Substring(Mobile.Length - 11,11);

                msg 
= dr["Msg"].ToString().Trim().Split(' ');        //获取短信的标志
                
                
//根据标志加载处理短信的实例,在这里省去了 case 。
                msgRecv = (HBS.SMSReceive.MessageReceive)Assembly.Load("HBS.SMSReceive").CreateInstance("HBS.SMSReceive.MessageOperation" + msg[0].Trim());

                
if (msgRecv == null)
                
{
                    
//没有找到对应的分析短消息的类,设置默认选项
                    msgRecv = new HBS.SMSReceive.MessageOperationOther();
                }


                msgRecv.dal 
= dal;        //设置“数据访问层”的实例

                
//处理接收到的短消息
                re = msgRecv.SaveMsg(Mobile,dr["ArriveTime"].ToString(),dr["Msg"].ToString().Trim().Replace("  "," "));

                
if (re.Length == 0)
                
{
                    
//正确执行,发确认短信。
                    msgSend.SendMsg(Mobile,"我们已经收到了您发的短消息,并且保存成功!");
                }

                
else
                
{
                    
//没有正确执行,发送错误信息。
                    msgSend.SendMsg(Mobile,re);
                }

                
                
//移动短信,把短信移动到历史记录里面
                msgRecv.LogMsg(dr["ID"].ToString());
                
            }

            
        }

        
#endregion


大体就是这样了。


反射还是在看了伍迷的小菜系列才会用的,再此表示感谢。
相关文章
|
7月前
面向对象设计领域中的参数多态,包含多态,过载多态和强制多态
面向对象设计领域中的参数多态,包含多态,过载多态和强制多态
84 1
|
2天前
面向对象编程:类、对象、继承与多态
面向对象编程:类、对象、继承与多态
4 0
|
10月前
|
设计模式
设计模式小例子理解封装、继承、多态
设计模式小例子理解封装、继承、多态
50 0
|
5月前
|
编译器 C++
[C++] 面向对象的三大特性:封装、继承和多态
[C++] 面向对象的三大特性:封装、继承和多态
35 0
|
7月前
|
存储 算法 Java
面向对象编程实践:类、对象与继承
面向对象编程实践:类、对象与继承
35 0
|
9月前
|
设计模式
26【软件基础】简单计算器的实现+工厂方法模式应用+封装、继承、多态的体现
工厂方法模式是一种常用的`创建型设计模式`,它提供了一种将对象的创建过程封装起来的方法。在工厂方法模式中,将对象的创建过程交给一个`工厂类`来完成,而不是在代码中直接调用构造函数来创建对象。这样可以使得代码更加灵活,`降低耦合度`,方便后期维护和扩展。
108 0
|
9月前
|
Java
面向对象编程基础:类、对象、封装、继承和多态
面向对象编程基础:类、对象、封装、继承和多态
90 0
|
10月前
|
设计模式 程序员 编译器
【大话设计模式】封装 继承 多态
【大话设计模式】封装 继承 多态
|
10月前
浅谈 面向对象三大特性:封装 继承 多态
浅谈 面向对象三大特性:封装 继承 多态
70 0
Java面向对象篇:封装、继承、多态(三)
Java面向对象篇:封装、继承、多态
Java面向对象篇:封装、继承、多态(三)