码迷,mamicode.com
首页 > Windows程序 > 详细

《C#图解教程》读书笔记之六:接口和转换

时间:2017-05-22 23:15:49      阅读:310      评论:0      收藏:0      [点我收藏+]

标签:引用类型   注意   ++   抛出异常   类型转换   invalid   bsp   读书   匹配   

本篇已收录至《C#图解教程》读书笔记目录贴,点击访问该目录可获取更多内容。

一、接口那点事儿

  (1)什么是接口?

  一组函数成员而未实现的引用类型。只有类和结构能实现接口。

  (2)从IComparable接口看接口实例:

  假设有如下一段代码,它使用Array类的一个静态方法Sort对一个未排序的int类型数组进行排序,并输出排序后的结果。

技术分享
using System;

class Program
{
   static void Main()
   {
      var myInt = new[] { 20, 4, 16, 9, 2 };       // Create an array of ints.

      Array.Sort( myInt );                         // Sort elements by magnitude.

      foreach ( var i in myInt )                   // Print them out.
         Console.Write( "{0} ", i );
   }
}
技术分享

  Sort方法在int类型数组的排序工作上做的很好,但是如果我们尝试在自定义的类上使用就会发生异常,例如下面的MyClass类。

class MyClass
{
   public int TheValue;
}

  Sort为何不能对MyClass进行排序,原因在于:它不知道如何比较自定义对象及如何进行排序。Array类的Sort方法其实依赖于一个IComparable的接口,它声明在BCL中,包含唯一的CompareTo方法。它接收一个object类型的参数,可以匹配任何引用类型。

public interface IComparable
{
    int CompareTo(object obj);
}

  这下,我们知道了int类型默认实现了IComparable接口,而我们的MyClass则没有。因此,我们需要将MyClass实现这个IComparable接口。

技术分享
class MyClass : IComparable                          
{
   public int TheValue;
   public int CompareTo( object obj )                 
   {
      MyClass mc = (MyClass) obj;

      if ( this.TheValue < mc.TheValue )
         return -1;

      if ( this.TheValue > mc.TheValue )
         return 1;

      return 0;
   }
}
技术分享

  现在,MyClass类实现了IComparable接口,它可以用于Sort方法了。

技术分享
class Program
{
   static void PrintOut( string s, MyClass[] mc )
   {
      Console.Write( s );
      foreach ( var m in mc )
         Console.Write( "{0} ", m.TheValue );

      Console.WriteLine( "" );
   }

   static void Main()
   {
      var myInt = new[] { 20, 4, 16, 9, 2 };

      MyClass[] mcArr = new MyClass[5];               
      for ( int i = 0; i < 5; i++ )                   
      {
         mcArr[i] = new MyClass();
         mcArr[i].TheValue = myInt[i];
      }

      PrintOut( "Initial Order: ", mcArr );           

      Array.Sort( mcArr );                            
      PrintOut( "Sorted Order:  ", mcArr );           
   }
}
技术分享

  现在,一个完整的接口实例已经完毕。

技术分享

  (3)使用接口注意事项:

    ①声明接口时:不能包含:数据成员、静态成员;只能声明:方法、属性、事件、索引器;

TIP:接口允许有任何的访问修饰符,但是接口成员是隐式public的,不允许有任何的访问修饰符,包括public。

技术分享

    ②实现接口时:在基类列表中包括接口名称;为每一个接口的成员实现接口;

  (4)接口是一种引用类型:我们不能直接通过类或对象的成员访问接口,然而,我们可以通过把类对象转换成接口类型来获取指向接口的引用。一旦有了接口的引用,我们就可以使用点号来调用接口的方法。

技术分享
using System;

interface IIfc1
{
   void PrintOut( string s );
}

class MyClass : IIfc1
{
   public void PrintOut( string s )
   {
      Console.WriteLine( "Calling through:  {0}", s );
   }
}

class Program
{
   static void Main()
   {
      MyClass mc = new MyClass();         
      mc.PrintOut( "object" );           

      IIfc1 ifc = (IIfc1) mc;            
      ifc.PrintOut( "interface" );       
   }
}
技术分享

  下面我们看看上面的代码在内存中的分配:

技术分享

  (5)接口和as运算符=>天生一对

  在以往使用接口引用时,我们往往会使用强制类型转换,但强制类型转换会抛出异常(异常是指代码中的意外错误,它会严重降低代码速度)。如何避免这个问题,我们可以使用as运算符,在类对象未实现接口时不会抛出异常,只会返回null。

技术分享

二、看我72变:转换

  (1)本质:接受一个类型的值并使用它作为另一个类型的等价值的过程

技术分享

  (2)转换分类:

  ①预定义的转换:数字、装箱/拆箱、引用转换;

  数字类型的转换详见下图:

技术分享

  装箱/拆箱是一个比较重要的点,现在我们来看看:

  装箱(boxing)是值类型->引用类型,本质其实是创建副本。装箱是一种隐式转换,它接收值类型的值,根据这个值在在堆上创建一个完整的引用类型对象并返回对象引用。

技术分享

  拆箱(unboxing)是引用类型->值类型,本质把装箱后的对象转换回值类型。拆箱是显示转换。

技术分享

  ②用户自定义的转换:隐式和显示的自定义转换;

技术分享 View Code

  (3)is运算符:

  在转换过程中,有些转换是不成功的,并且会在运行时抛出一个InvalidCastException异常。我们可以使用is运算符来检查转换是否会成功,从而避免盲目地尝试转换

技术分享

本章思维导图

技术分享

附件

  思维导图(jpg、pdf以及mmap源文件)下载:http://pan.baidu.com/s/1qWNOGGW

 

作者:周旭龙

出处:http://www.cnblogs.com/edisonchou/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。

《C#图解教程》读书笔记之六:接口和转换

标签:引用类型   注意   ++   抛出异常   类型转换   invalid   bsp   读书   匹配   

原文地址:http://www.cnblogs.com/luozhijun/p/6891669.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!