转:http://www.cnblogs.com/landeanfen/p/4953025.html

C#基础系列——一场风花雪月的邂逅:接口和抽象类

前言:最近一个认识的朋友准备转行做编程,看他自己边看视频边学习,挺有干劲的。那天他问我接口和抽象类这两个东西,他说,既然它们如此相像, 我用抽象类就能解决的问题,又整个接口出来干嘛,这不是误导初学者吗。博主呵呵一笑,回想当初的自己,不也有此种疑惑么。。。今天打算针对他的问题,结合 一个实际的使用场景来说明下抽象类和接口的异同,到底哪些情况需要用接口?又有哪些情况需要用抽象类呢?

C#基础系列目录:

一、业务场景介绍。

博主打算使用原来在华为做外包的时候一个场景:我们针对华为里面的设备做了一个采集设备使用率的程序,设备的类型很多,各种设备的登录和注销方式基本相同,但是每种设备的采集的规则又不太相同。大致的场景就这样,我们来看代码吧。

二、代码示例

根据业务场景,我们简单搭建代码,先来看看代码结构图:

ESTM.Spider:项目的入口程序,只为测试,这里就简单用了一个控制台程序。

ESTM.Spider.Huawei:华为设备采集规则,定义接口抽象实现和具体实现。

ESTM.Utility:解决方案的工具类和接口。

下面来看看具体的实现代码:

1、工具类

namespace ESTM.Utility
{
public class LoginUser
{
public string Username { set; get; } public string Password { set; get; }
} public class Device
{
public string DeviceType { set; get; } public int WaitSecond { set; get; }
}
}

2、接口设计:ISpider.cs

namespace ESTM.Utility
{
//采集接口,定义采集的规则
public interface ISpider
{
bool Login(LoginUser oLoginUser); string Spider(Device oDevice); void LoginOut();
}
}

3、接口抽象实现类:SpiderBase.cs

   /// <summary>
/// 公共的采集基类
/// </summary>
public abstract class SpiderBase : ISpider
{
//华为设备统一采用Telnet方式登录。统一用户名密码都是admin。
public virtual bool Login(LoginUser oLoginUser)
{
Console.WriteLine("华为设备采用Telnet方式登录。"); var bRes = false;
if (oLoginUser.Username == "admin" && oLoginUser.Password == "admin")
{
Console.WriteLine("用户名密码校验正确,登录成功");
bRes = true;
}
else
{
Console.WriteLine("用户名密码校验错误,登录失败");
}
return bRes; } //采集操作和具体的设备类型相关,这里用抽象方法,要求子类必须重写
public abstract string Spider(Device oDevice); //华为设备统一注销
public virtual void LoginOut()
{
Console.WriteLine("华为设备采用Telnet方式注销");
}
}

4、接口具体实现类

  [Export("MML", typeof(ISpider))]
public class SpiderMML:SpiderBase
{
//MML设备采集
public override string Spider(Device oDevice)
{
Console.WriteLine("MML设备开始采集");
return "MML";
}
}
    [Export("TL2", typeof(ISpider))]
public class SpiderTL2:SpiderBase
{
//TL2设备采集
public override string Spider(Device oDevice)
{
Console.WriteLine("TL2设备开始采集");
return "TL2";
}
}

5、在控制台调用

  class Program
{
[Import("MML", typeof(ISpider))]
public ISpider spider { set; get; } static void Main(string[] args)
{
var oProgram = new Program();
RegisterMEF(oProgram); oProgram.spider.Login(new LoginUser() { Username = "admin", Password = "admin" });
oProgram.spider.Spider(new Device() { DeviceType = "HuaweiDevice", WaitSecond = 100 });
oProgram.spider.LoginOut();
} #region 注册MEF
private static void RegisterMEF(object obj)
{
AggregateCatalog aggregateCatalog = new AggregateCatalog();
var thisAssembly = new DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory, "*.dll");
aggregateCatalog.Catalogs.Add(thisAssembly);
var _container = new CompositionContainer(aggregateCatalog, true);
_container.ComposeParts(obj);
}
#endregion
}

6、说明

这是一种比较典型的应用场景。接口定义规则,抽象类定义公共实现或者抽象方法,具体子类实现或者重写抽象类方法。我们重点来看这里的中间桥梁——抽象类。我们知道,抽象类里面既可以有实现的方法,也可以有未实现的抽象方法。

(1)在这里,Login、LoginOut方法由于子类是通用的具有相同逻辑的方法,所以我们需要在抽象类里面去实现这两个方法,如果子类没有特 殊需求,调用的时候直接用父类的方法就好了; 如果子类有特殊需求,可以override父类的方法。这样设计既提高了代码的复用率,也可以灵活复写。

(2)另一方面,抽象类里面也定义了抽象方法,这个抽象方法在这里的作用就很好体现了:如果子类不重写父类的抽象方法,编译通不过,直接报错。这样就要求我们子类必须要重写抽象方法。从这点来说,抽象方法和接口的方法申明区别不大。

(3)如果这里不用抽象类,就用一个普通的类来代替行不行?博主的答案是:行!但不好!如果你非要说,我用一个普通的类,将public abstract string Spider(Device oDevice);这个方法写成

public virtual string Spider(Device oDevice)
{
return "";
}

貌似也没问题,反正子类要重写的。确实,这样设计没问题,但是如果你不慎子类忘了override呢?程序还是会跑起来,运行的时候可能会报错。微软既然给我们提供了abstract这么一个东西,我们为什么不用呢。

三、代码扩展

以上我们抽象类使用的必要性和使用方法是介绍完了。那么接下来新的问题来了,可能就有人问了,你上面说了叭叭叭说了这么多,无非就是说了抽象类的必 要性,那么既然抽象类这么有用,我们直接用抽象类就好了,你干嘛还要弄一个接口呢。谈到这里,就要说到面向接口编程。其实,面向接口编程和面向对象编程并 不是平级的,它并不是比面向对象编程更先进的一种独立的编程思想,而是附属于面向对象思想体系,属于其一部分。或者说,它是面向对象编程体系中的思想精髓 之一。而之前博主的文章就分享过面向接口编程的意义所在:依赖倒置,松耦合。那么这里是否可以不要接口,直接用抽象类代替呢?答案还是行!但不好!

比如我们现在又来了新的需求,中兴也要用我们的采集系统,但是它的设备类型、登录注销方式和华为设备区别非常大。那么这个时候我们接口的意义就体现 了,如果我们使用接口,我们只需要再重写一个ESTM.Spider.Huawei这个项目就好了,我们暂且命名叫 ESTM.Spider.Zhongxing。我们来看看:

代码如下:

namespace ESTM.Spider.Zhongxing
{
/// <summary>
/// 中兴设备采集基类
/// </summary>
public abstract class SpiderBase:ISpider
{
//中兴设备通用登录方法
public virtual bool Login(LoginUser oLoginUser)
{
Console.WriteLine("中兴设备登录前多了一个数据校验:.......");
Console.WriteLine("中兴设备采用WMI方式登录。"); var bRes = false;
if (oLoginUser.Username == "root" && oLoginUser.Password == "root")
{
Console.WriteLine("用户名密码校验正确,登录成功");
bRes = true;
}
else
{
Console.WriteLine("用户名密码校验错误,登录失败");
}
return bRes;
} //定义抽象方法,要求子类必须重写
public abstract string Spider(Device oDevice); //中兴设备通用注销
public virtual void LoginOut()
{
Console.WriteLine("中兴设备采用WMI方式注销");
}
}
}
namespace ESTM.Spider.Zhongxing
{
[Export("ZXGC", typeof(ISpider))]
public class SpiderZXGC:SpiderBase
{
public override string Spider(Utility.Device oDevice)
{
Console.WriteLine("中兴ZXGC设备开始采集");
return "ZXGC";
}
}
}
namespace ESTM.Spider.Zhongxing
{
[Export("ZXGY", typeof(ISpider))]
public class SpiderZXGY:SpiderBase
{
public override string Spider(Utility.Device oDevice)
{
Console.WriteLine("中兴ZXGY设备开始采集");
return "ZXGY";
}
}
}

由于这里采用了接口,我们将ESTM.Spider.Zhongxing这个项目开发完成后生成dll,将dll放到控制台程序中,直接通过MEF 导入不同的子类对象就可以使用,不需要更改控制台里面的大部分东西。如果不用接口,而是直接用抽象类代替,那么控制台里面大部分的代码都得改,并且控制台 程序依赖多个dll,对设计的松耦合也不利。博主这里为了简单,用了MEF来简单导入,其实正式项目中,应该是用工厂采用反射直接创建出具体的实例。

四、总结

1、接口是一组规则的集合,它主要定义的是事物的规则,体现了是这种类型,你就必须有这些规则的概念。它的目的主要是依赖倒置和松耦合,从这点来说,接口不能省掉或者用抽象类代替。总而言之,接口和抽象类不可同日而语。

2、抽象类主要用于公共实现和约束子类必须重写。以上面的例子说明,Login、Loginout用于公共实现,提高了代码复用,Spider用于抽象,约束子类必须要重写Spider方法。这也就是这里不能用普通类的原因。

3、用一句话概括接口和抽象类的区别:使用抽象类是为了代码的复用,而使用接口的动机是为了实现多态性(依赖倒置)。至于使用的时候到底是用接口还是抽象类,看具体的情况。

c#基础系列(转)的更多相关文章

  1. 【Basics of Entity Framework】【EF基础系列1】

    EF自己包括看视频,看MSDN零零散散的学了一点皮毛,这次打算系统学习一下EF.我将会使用VS2012来学习这个EF基础系列. 现在看看EF的历史吧: EF版本 相关版本特性介绍 EF3.5 基于数据 ...

  2. C#基础系列——Attribute特性使用

    前言:上篇 C#基础系列——反射笔记 总结了下反射得基础用法,这章我们来看看C#的另一个基础技术——特性. 1.什么是特性:就博主的理解,特性就是在类的类名称.属性.方法等上面加一个标记,使这些类.属 ...

  3. C#基础系列——小话泛型

    前言:前面两章介绍了C#的两个常用技术:C#基础系列——反射笔记 和 C#基础系列——Attribute特性使用 .这一章来总结下C#泛型技术的使用.据博主的使用经历,觉得泛型也是为了重用而生的,并且 ...

  4. C#基础系列——委托实现简单设计模式

    前言:上一篇介绍了下多线程的相关知识:C#基础系列——多线程的常见用法详解,里面就提到了委托变量.这篇简单介绍下委托的使用.当然啦,园子里面很多介绍委托的文章都会说道:委托和事件的概念就像一道坎,过了 ...

  5. C#基础系列——委托和设计模式(二)

    前言:前篇 C#基础系列——委托实现简单设计模式 简单介绍了下委托的定义及简单用法.这篇打算从设计模式的角度去解析下委托的使用.我们知道使用委托可以实现对象行为(方法)的动态绑定,从而提高设计的灵活性 ...

  6. C#基础系列——再也不用担心面试官问我“事件”了

    前言:作为.Net攻城狮,你面试过程中是否遇到过这样的问题呢:什么是事件?事件和委托的区别?既然事件作为一种特殊的委托,那么它的优势如何体现?诸如此类...你是否也曾经被问到过?你又是否都答出来了呢? ...

  7. C#基础系列——异步编程初探:async和await

    前言:前面有篇从应用层面上面介绍了下多线程的几种用法,有博友就说到了async, await等新语法.确实,没有异步的多线程是单调的.乏味的,async和await是出现在C#5.0之后,它的出现给了 ...

  8. C#基础系列——一场风花雪月的邂逅:接口和抽象类

    前言:最近一个认识的朋友准备转行做编程,看他自己边看视频边学习,挺有干劲的.那天他问我接口和抽象类这两个东西,他说,既然它们如此相像, 我用抽象类就能解决的问题,又整个接口出来干嘛,这不是误导初学者吗 ...

  9. 2015年12月28日 Java基础系列(六)流

    2015年12月28日 Java基础系列(六)流2015年12月28日 Java基础系列(六)流2015年12月28日 Java基础系列(六)流

随机推荐

  1. Django~queries

    API queries create, retrieve, update and delete

  2. ABAP 通过视图取数到内表函数

    CALL FUNCTION 'VIEW_GET_DATA'     EXPORTING       view_name = 'V_TVBUR'     TABLES       data      = ...

  3. Match:DNA repair(POJ 3691)

    基因修复 题目大意:给定一些坏串,再给你一个字符串,要你修复这个字符串(AGTC随便换),使之不含任何坏串,求修复所需要的最小步数. 这一题也是和之前的那个1625的思想是一样的,通过特殊的trie树 ...

  4. codeforces 493B.Vasya and Wrestling 解题报告

    题目链接:http://codeforces.com/problemset/problem/493/B 题目意思:给出 n 个 techniques,每个 technique 的值为 ai. ai & ...

  5. Android状态栏微技巧,带你真正意义上的沉浸式

    记得之前有朋友在留言里让我写一篇关于沉浸式状态栏的文章,正巧我确实有这个打算,那么本篇就给大家带来一次沉浸式状态栏的微技巧讲解. 其实说到沉浸式状态栏这个名字我也是感到很无奈,真不知道这种叫法是谁先发 ...

  6. php命令行运行出现错误Call to undefined function curl_init()

    在windows命令行窗口下运行php命令,需要将php.exe所在的路径添加到Path环境变量,例如C:\wamp\bin\php\php5.4.3 启动Apache服务 在命令行中输入php te ...

  7. oracle dataguard (一)

    一.什么是data guard及data guard的工作原理 Data Guard 是一个集合,由一个primary数据库(生产数据库)及一个或多个standby数据库(最多9个)组成.组成Data ...

  8. openURL的使用方法:

    openURL的使用方法: view plaincopy toclipboardprint?        [[UIApplication sharedApplication] openURL:[NS ...

  9. gitlab 用户头像不能显示的问题

    [root@GitLab assets]# cat /etc/gitlab/gitlab.rb # Change the external_url to the address your users ...

  10. SQLServer多表连接查询

    双表内部连接查询 select wName,dName from DepartMent,Worker where DepartMent.dID=Worker.did select wName,dNam ...