C#之语言详述

	上一篇写到的是C#的核心语言,属于泛泛而谈。这一篇继续C#的学习,开始对C#语言的详述,本篇内容主要包括委托、事件和异常处理。
一. 委托
	要理解委托的概念,必须清楚什么是函数指针。
	函数指针是对函数的间接引用,支持通过变量来调用函数。通过函数指针,我们可以将函数作为一个参数或作为一个返回值进行传递。函数指针可以使应用程序更灵活、可扩展性和可伸缩性更好。但函数指针不是类型安全的。
	从生活角度去理解,我觉得函数指针就像是一个快递点。在这里,当我们需要的时候,我们可以通过它来取件寄件。
	一个“委托”是对一个或多个函数指针的一种抽象。委托是一个引用类型,具有一个签名和返回类型。事件是最实用的委托应用程序之一。事件在后面还会有详细阐述。
	下面是对委托的一般过程进行了解:一般包括定义、创建和调用这三个步骤。
	(1)delegate关键字是用于定义新委托的。以下代码定义了一个新的委托,名为DelegateClass的类:
一个委托可以在命名空间中、在类中进行定义,但不能在一个方法中作为一个类字段或一个局部变量进行定义。

public delegate int DelegateClass(string info); //定义一个委托


(2)new关键字是用于为一个委托创建一个实例。委托的构造函数不能被重载。这里包含三种不同的情况,以下代码是对类进行初始化的三种情况:

a) 对于一个实例方法,采用object.method格式

b) 对于一个静态方法,采取class.method格式 

c) 对于方法和委托包含在同一类中,则既不需要对象名,也不需要类名。

public delegate void DelegateClass();
public class constructors
        {
            public static void Main()
            {
                DelegateClass del1 = new DelegateClass(constructors.MethodA);
                //静态方法下的实例化
                DelegateClass del2 = new DelegateClass(MethodA);
                //指定的方法和委托包含在同一类中下的实例化
                ZClass obj = new ZClass();
                DelegateClass del3 = new DelegateClass(obj.MethodB);
                //实例方法下的实例化
            }
            public static void MethodA()
            {
            }
        }
        public class ZClass
        {
            public  void  MethodB()
            {
            }
        }
	(3)invoke方法对委托进行调用。如果委托包含一个函数指针,函数的返回就成为委托的返回。当多个函数指针被存储到指定委托的时候,最后一个函数的返回就成为委托的返回。
二.事件
	在学习VB的时候,就了解到的一句话是:以事件驱动程序。在编程时,我们也是想要实现某项功能,就在相应的事件下去编写代码。比如:点击“登录”按钮,就进入到机房收费系统的主界面。我们就可以在Command按钮下的Click事件下编写代码,从而驱动程序。
	在这本书C#中写道,一个事件是指一个对象或类想要通知其他人有事发生(这里指的其他人是另一个类,而不是另一个对象)。这就是在加入面向对象思想后的一个变化,所有的一切都是在类中打交道。
	任何对一个事件感兴趣的对象或类都可以订阅该事件。订阅者通过提交一个委托为一个事件进行注册。该委托必须是单角色的,并且只包含一个单独的函数指针,这个函数是订阅者对该事件的响应。当指定的事件被引发的时候,发布者调用该函数,给订阅者提供一个对事件进行响应的机会。该函数被称为一个事件处理程序。事件可以有多个订阅者,而没有订阅者的事件不会被引发。
	上面说了很多专业的语言,我觉得简单理解来看,其实就是一个事件可以被多次访问,从而做出相应的处理程序;也可以是无人访问,不过这样的事件就是不会被引发罢了。
	(1)event关键字用于定义一个事件

<pre name="code" class="csharp">accessibility event delegatename eventname
	(2)public或protected关键字一般是对一个事件的可访问性设置

public delegate void DelegateClass();
<pre name="code" class="csharp">public event DelegateClass MyEvent;

	(3)add和remove方法用于订阅和取消订阅
	(4)引发事件是用调用运算符“()”,添加调用运算符到事件就会引发该事件。

public void SomeMethod()
{
      if (anEvent!=null)
      {
          anEvent(null,null);
      }
}
三.异常处理

什么是异常?异常是应用程序发生异常事件或错误情况,分为系统异常和应用程序异常。

系统异常是公共语言运行时(CLR)引发的,包括空引用、内存泄漏、被零除和堆栈溢出异常。应用程序异常,被看做是自定义异常,是由应用程序引发的。

(1)异常举例

常见的一个异常就是被零除,也就是零被作为除数。如下面代码,发生了被零除异常,终止了程序。

public static void Main()
{
       int var1=5,var2=0;
       var1/=var2;//exception occurs
}

把可能产生异常的代码放在一个try块中,因为try块中的代码是受异常保护的,这样就可以捕获异常。catch块处理异常,显示堆栈跟踪。

public static void Main()
        {
            try
            {
                int var1 = 5, var2 = 0;
                var1 /= var2;//exception occurs
            }
            catch (DivideByZeroException except)
            {
                Console.WriteLine("Exception " + except.StackTrace);
            }
        }

这样运行结果就是会提醒程序具体哪一行出错:


(2)结构化的异常处理

也就是一个专门负责异常处理的工具名称。它通过评估堆栈决定何时代码受保护和何处捕捉和处理一个异常。

a)try语句:监视器,监视保护代码中的异常。

下面是一个在超过数组边界时发生的边界溢出处理错误。

public static void Main()
        {
            try
            {
                MethodA();
            }
            catch (Exception except)
            {
                Console.WriteLine( except.Message);
            }
        }
        public static void MethodA()
        {
            int[] values = { 1, 2, 3, 4 };
            for (int count = 0; count <= values.Length; ++count)
            {
                Console.WriteLine(values[count]);
            }
        }

结果是在未保护的MethodA方法中的一个异常,Main调用MethodA,Main的范围包括MethodA,因此,Main中的try块扩展保护到MethodA。异常在Main中被捕捉。

b)catch语句:筛选和处理异常。其中筛选器可以帮助我们捕捉各类异常。

catch筛选器是可选的,默认的是catch all。在(1)中,异常举例的DivideByZeroException就是专门捕捉被零除异常的。下面代码属于一个默认的:

public static void Main()
        {
            try
            {
                int var1 = 5, var2 = 0;
                var1 /= var2;
            }
            catch
            {
                //catch remaining managed and unmanaged exceptions
            }
        }

c)finally语句:终结处理器。起到一个可以关闭文件、释放一个数据库或其他的管理资源的用途。

下面是一个典型的终结处理器:

using System;
using System.IO;
namespace ConsoleApplication1
{
    public class FileWrite
    {
        public static void Main()
        {
            StreamWriter sw = null;
            try
            {
                sw = new StreamWriter("date.txt");
                sw.Write(DateTime.Now.ToLongTimeString());
                throw new ApplicationException("exception");
                //dangling code
            }
            finally
            {
                sw.Close();
                Console.WriteLine("file closed");
            }
        }
    }
}

d)语句使用规范

使用必须与一条catch或finally语句成对使用。可以有零到多条catch语句和一条try语句结合使用,没有或只有finally一条语句。如果catch和finally语句同时出现,则catch语句应在finally语句之前。

四.总结

这一编的学习把委托又重新认识了一遍,以前总觉得很难理解,现在觉得其实从生活中去看,委托无处不在。如何去定义,如何去调用,在不同的情况下如何正确去实例化,这都是在这一次中学习中的收获。对于try-catch-finally语句,也是在很早就编写过,看似比较简单,实际上也是内容丰富,现在的我们还都是在用调试阶段就可以应付,但以后就不一样了,异常的情况可能会经常碰到,所以这也是一个很基础的学习。

在期末复习阶段,学了些C#的东西,后天就要开学了,机房重构,就要真正开始了。Come On!

				
时间: 2024-10-26 10:31:33

C#之语言详述的相关文章

HTML语言剖析(四)排版标记

HTML语言剖析 排版标记 <!--注解--> : <P> : <BR> : <HR> : <CENTER> : <PRE> : <DIV> : <NOBR> : <WBR> :  ■<!--注解-->:                                                                                             

C语言基础教程(四)指针、结构、联合和枚举(1)

本节专门对第二节曾讲述过的指针作一详述.并介绍Turbo C新的数据类型:结构.联合和枚举,其中结构和联合是以前讲过的五种基本数据类型(整型.浮点型.字符型.指针型和无值型)的组合.枚举是一个被命名为整型常数的集合.最后对类型说明(typedef)和预处理指令作一阐述. 1. 指针(point) 学习Turbo C语言,如果你不能用指针编写有效.正确和灵活的程序,可以认为你没有学好C语言.指针.地址.数组及其相互关系是C语言中最有特色的部分.规范地使用指针, 可以使程序达到简单明了,因此,我们不

C++快速学习Objective-C语言核心语法

  本文将Objective-C讨论了语言的核心语法.这部分开始详述一些具体的语法.正如你期待的一样,涉及到了定义和类. 类并不是特殊的 在Smalltalk中,类是具有一些特性的对象.在Objective-C中也一样.一个类是一个对象,对象回应消息.Objective-C和C++都分离了对象分配和初始化. 在C++中,对象分配通过新的操作.在Objective-C中,这样的操作是通过给类发送分配消息-调用malloc()或者一个等价. C++中的初始化是通过调用一个与类同名的函数.Object

C语言中的指针和内存泄漏

引言 对于任何使用C语言的人,如果问他们C语言的最大烦恼是什么,其中许多人可能会回答说是指针和内存泄漏.这些的确是消耗了开发人员大多数调试时间的事项.指针和内存泄漏对某些开发人员来说似乎令人畏惧,但是一旦您了解了指针及其关联内存操作的基础,它们就是您在 C 语言中拥有的最强大工具. 本文将与您分享开发人员在开始使用指针来编程前应该知道的秘密.本文内容包括: 导致内存破坏的指针操作类型 在使用动态内存分配时必须考虑的检查点 导致内存泄漏的场景 如果您预先知道什么地方可能出错,那么您就能够小心避免陷

《C语言接口与实现:创建可重用软件的技术》一2.4 客户程序的职责

2.4 客户程序的职责 接口是其实现和其客户程序之间的一份契约.实现必须提供接口中规定的功能,而客户程序必须根据接口中描述的隐式和显式的规则来使用这些功能.程序设计语言提供了一些隐式规则,来支配接口中声明的类型.函数和变量的使用.例如,C语言的类型检查规则可以捕获接口函数的参数的类型和数目方面的错误. C语言的用法没有规定的或编译器无法检查的规则,必须在接口中详细说明.客户程序必须遵循这些规则,实现必须执行这些规则.接口通常会规定未检查的运行时错误(unchecked runtime error

详解C语言编程中预处理器的用法_C 语言

预处理最大的标志便是大写,虽然这不是标准,但请你在使用的时候大写,为了自己,也为了后人. 预处理器在一般看来,用得最多的还是宏,这里总结一下预处理器的用法. #include <stdio.h> #define MACRO_OF_MINE #ifdef MACRO_OF_MINE #else #endif 上述五个预处理是最常看见的,第一个代表着包含一个头文件,可以理解为没有它很多功能都无法使用,例如C语言并没有把输入输入纳入标准当中,而是使用库函数来提供,所以只有包含了stdio.h这个头文

《深入理解Nginx:模块开发与架构解析》一3.9 用C++语言编写HTTP模块

3.9 用C++语言编写HTTP模块 Nginx及其官方模块都是由C语言开发的,那么能不能使用C++语言来开发Nginx模块呢?C语言是面向过程的编程语言,C++则是面向对象的编程语言,面向对象与面向过程的优劣这里暂且不论,存在即合理.当我们由于各种原因需要使用C++语言实现一个Nginx模块时(例如,某个子功能是用C++语言写成,或者开发团队对C++语言更熟练,又或者就是喜欢使用C++语言),尽管Nginx本身并没有提供相应的方法支持这样做,但由于C语言与C++语言的近亲特性,我们还是可以比较

VC中实现GB2312、BIG5、Unicode编码转换的方法_C 语言

本文主要以实例形式讨论了VC编译环境下,实现字符串和文件编码方式转换的方法,在linux下请使用Strconv来实现.具体方法如下: 一.文件编码格式转换 //GB2312 编码文件转换成 Unicode: if((file_handle = fopen(filenam,"rb")) != NULL) { //从GB2312源文件以二进制的方式读取buffer numread = fread(str_buf_pool,sizeof(char),POOL_BUFF_SIZE,file_h

C和MFC巧妙获取外网IP的两种实现方法_C 语言

本文以C与MFC的两个实例详述了取外网IP的两种实现方法,具体实现代码如下: MFC语言实现获取外网IP: # include <windows.h> # include <urlmon.h> # pragma comment(lib,"URLMON.lib") void main() { URLDownloadToFile(NULL,"http://www.ip138.com/ip2city.asp","ip.txt",