本文为原创,如需转载,请注明作者和出处,谢谢!
一、用var定义变量
在C#3.0中提供了一种新的声明变量的方式,这就是var。通过这个关键字,在声明变量时就无需指定类型了,变量类型是在初始化时由编译器确定的。代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->varss="abcd";
MessageBox.Show(ss.GetType().ToString());
上面的代码将显示System.String,从而证明C#编译器已经将ss编译成了String变量了。而在输出ss后,再输入“.”后,会看到将String类型变量的相应方法和属性也列出来了,因此可以断定,C#将ss看成了String类型,而不是Object。所以使用var定义变量同时可以拥有Object和强类型的优点。
不过大家不要将var看成是javascript的var,它们的区别是,javascript是弱类型的语言,而且javascript中的变量(也包括用var声明的变量)可以变换类型,如下面的javascript所示:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->vars="abcd";
s=3;
alert(s);
上面的代码第一次给s赋了一个字符串,而第二行代码又给赋了一个整数。这样的代码在javascript中没有任何问题。但在C#3.0中,var变量一但被初始化,确定类型后,就无法改变类型了。如下面的代码是无法编译通过的:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->varss="abcd";
ss=44;
综上所述,在使用var定义变量时有以下四个特点:
1.
必须在定义时初始化。也就是必须是var s = “abcd”形式,而不能是如下形式:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->vars;
s=“abcd”;
2.
一但初始化完成,就不能再给变量赋与初始化值类型不同的值了。
3. var要求是局部变量。
4.
使用var定义变量和object不同,它在效率上和使用强类型方式定义变量完全一样。但笔者建议如果事先知道变量的类型,尽量使用强类型方式来声明变量。否则,就会造成由于大量使用var,而使得开发人员很难断定某个变量是什么类型。这样不利于程序的维护和升级。
虽然var有利有弊,但笔者个人认为,如果将动态语言转换成C#语言,可以考虑使用var来定义变量。这是因为动态语言没有类型,而要将其转换成强类型的C#语言,就必须给变量指定个类型,但事先确定类型是很费劲的,不如将其指定成var,再由C#编译器去确定变量的具体类型。那么如果在转换的过程中,发现动态语言的变量改变了类型,该怎么办呢?这个可以使用第三部分要讲的“匿名类”来解决这个问题。
二、初始化
如果一个类有public字段,在建立类的对象实例时可以使用下面的代码来初始化这些字段;
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->publicclassMyClass
{
publicStringfield1;
publicintfield2;
publicboolfield3;
}
MyClassmy=newMyClass();
my.field1=“abcd”;
my.field2=44;
my.field3=true;
在C#3.0中提供了一种更简便的方法来初始化这些public变量,代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->MyClassmy=newMyClass
{
field1=“abcd”,
field2=44;
field3=true;
};
上面的代码的写法有些象带参数的构造方法,但这将不是调用了MyClass的构造方法(因为MyClass并没有带三个参数的构造方法),而只是C#编译器玩的一个魔术。实际上,上面的代码在编译后,仍然和使用传统的初始化字段的方法一样。只是在语法上看起来更简单(至少不用写那么多个my)。要注意的的,使用这种方法初始化,必须是public的字段(不能是protected、private或默认修饰符的字段)。
在C#3.0中还改进了对集合类的初始化方式(使其初始化的方式类似于数组)。但遗憾的是,这种初始化方式只支持用泛型的集合类,也就是说,只有实现了System.Collections.Generic.ICollection<T>的集合类才可以使用这种初始化方法。代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->List<string>myList=newList<string>{"data1","data2","data3"};
foreach(stringdatainmyList)
{
textBox1.AppendText(data);
}
三、匿名类
在C#3.0中提供了一种新的建立类的方法,代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->varmy=new
{
field1="abcd",
field2=12
};
MessageBox.Show(my.field1);
C#编译器会自动推断my是一个有两个public字段的类的对象实例。也就是说相当于下面的代码:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->publicclassMyClass
{
publicStringfield1;
publicintfield2;
}
varmy=newMyClass();
my.field1="abcd";
my.field2=25;
MessageBox.Show(my.field1);
在第一部分讲到如果动态语言在给变量赋值的过程中改变了变量类型,如果将其转换为强类型语言。当然,一种方法是将变量声明成object类型,或是使用匿名类来解决这个问题。代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->varmyVar=new
{
field_string=“abcd”
field_int=12;
};
然后根据当前这个变量所使用的类型来决定该使用哪个类字段。
四、扩展方法
这个世界上总是存在着很多奇妙的东西。然而,在这部分所介绍的扩展方法就是其中之一。从字面上看可能读者很难猜透“扩展方法”是什么意思。然而,看了下面的例子,就会感觉到非常的奇妙。
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->
namespaceExtMethod
{
publicclassClass1
{
publicStrings="bill";
}
publicclassClass2:Class1
{
}
publicstaticclassAnyClassName
{
publicstaticStringgetName(thisClass1class1)
{
returnclass1.s+class1.s;
}
}
publicpartialclassForm1:Form
{
privatevoidbutton1_Click(objectsender,EventArgse)
{
Class1c=newClass1();
MessageBox.Show(c.getName());
Class2c=newClass2();
MessageBox.Show(c.getName());
}
}
}
看到上面的代码,也许很多人会感到奇怪,在Class1和Class2中并没有getName方法,怎么在调用时出来个getName方法呢?实际上,这就是扩展方法的用法,从本质上说,扩展方法就是将静态方法(必须声明成static)插入到某个类和其子类中(也就是说,在这些类中可以使用在外部定义的静态方法)。那么要往哪个类中插入呢?这就要在定义静态方法时指定了。大家可以看看getName方法的第一个参数,使用了this关键字,这就表明这个方法是一个扩展方法,后面的类型就是要插入该方法的类,在本例中是Class1,也就是说在Class1及其子类中都可以使用getName方法。上面的调用代码也相当于下面的代码:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->Class2c=newClass2();
MessageBox.Show(AnyClassName.getName(c));
但使用c.getName可能会更好一些,而且也降低了对静态方法所在的类(AnyClassName)的依赖性。
在使用扩展方法时应注意以下几点:
1.
扩展方法所在的类名可以是任意合法的类名。
2.
扩展方法所在的类必须和使用扩展方法的代码在同一个命名空间里,否则无法编译通过。
3.
在本例中,Class1和Class2只能声明成public,因为AnyClassName被声明为public。如果AnyClassName不加修饰符,Class1和Class2也可以不加修饰符,当然,也可以被声明为public。也就是说,Class1和Class2必须有比AnyClassName具有更强的访问性。如下面代码所示:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->classClass1
{
publicStrings="bill";
}
classClass2:Class1
{
}
staticclassAnyClassName//这时如果前面加public是无法编译通过的。
{
publicstaticStringgetName(thisClass1class1)
{
returnclass1.s+class1.s;
}
}
4.
如果在Class1或Class2中已经有getName方法了,那么Class1或Class2中的getName优先级更高。也就是说,扩展方法是无法覆盖原类中的同名(参数名和类型也相同)的方法的。
扩展方法尤其在很多类需要同样的方法,而这些类又无法继承其它类时特别有用。当然,在要对某个类进行扩展,但我们并没有源代码时,扩展方法也可以派上用场。
分享到:
相关推荐
.NET 3.5相对于.NET 3.0的主要增强在于C# 3.0的新特性,以及对.NET 3.0的进一步扩展和完善。下面将详细介绍.NET 3.5中的一些关键新特性: #### C# 3.0新特性 - **自动属性 (Auto-Implemented Properties)**:允许...
.NET Framework 3.5是.NET Framework的一个重要版本,它是在.NET Framework 2.0和.NET Framework 3.0的基础上进行的升级,包含了许多新的功能和技术。该版本主要增加了对LINQ(Language Integrated Query)的支持,...
4. **隐式类型局部变量(Implicitly Typed Local Variables)**:使用`var`关键字声明变量,编译器会根据初始化表达式推断变量类型。 5. **匿名类型(Anonymous Types)**:创建临时对象而不需定义类。 6. **扩展...
可以更直观地创建和初始化对象和集合,如`var person1 = new Person { Name = "tang", Age = 21 };` 和 `var persons = new List<Person> { new Person { Name = "TEW", Age = 21 }, new Person { Name = "RSA", Age...
扩展方法是一种在不修改原有类型的情况下为其添加新方法的技术。这些方法必须定义在一个静态类中。例如: ```csharp public static class StringExtensions { public static bool IsShort(this string s) { ...
- **多.NET Framework版本支持**:Visual Studio 2008支持.NET Framework 2.0/3.0 和 3.5,能够自动适应不同框架版本下的特性与工具箱。 - **ASP.NET AJAX**:ASP.NET AJAX 成为了.NET Framework 3.5的一部分,增强...
- C#是ASP.NET的主要编程语言,3.0版本引入了多项新特性,如匿名方法、lambda表达式、对象初始化器、类型推断(var关键字)和泛型委托等。理解这些概念对于进行ASP.NET开发至关重要。 2. **第4章:ASP.NET的网页...
- **扩展方法**:允许向现有类添加新方法,而无需修改该类的源代码。 - **匿名类型**:创建只在当前作用域内使用的临时类型,用于返回包含多个字段的数据结构。 - **查询表达式**:通过类似SQL的语法编写LINQ...
- **ASP.NET 3.5技术蓝图总览**:概述了ASP.NET 3.5的关键技术和新增特性,为读者提供了整体的技术背景。 - **ASP.NET 3.5技术脉络**:详细阐述了ASP.NET 3.5中的各项技术如何相互关联,以及它们是如何构建在一个...
4. **ASP.NET 3.5**:ASP.NET是微软的Web应用程序开发框架,3.5版本在2.0基础上进行了扩展,新增了ASP.NET AJAX、Linq to SQL、Entity Framework等功能,使Web开发更加高效和动态。 **C# 3.0的关键知识点:** 1. **...
.NET Framework 3.5之后引入了一个新特性,即使用`var`关键字进行隐式类型化,例如: ```csharp var user = new User(); ``` 这种语法类似于JavaScript中的变量声明方式,它简化了局部变量的实例化过程。需要注意的...
VS2008集成了.NET Framework 3.5,这是.NET Framework的一个重要版本,包含了大量新特性和改进,如Language Integrated Query (LINQ)、扩展方法、匿名类型等。 ### 2. LINQ (Language Integrated Query) LINQ是VS...
6. **扩展方法**:为现有类添加新的方法,无需修改原始类的定义,这为代码重用提供了新的途径。 7. **表达式树**:允许Lambda表达式以数据的形式存在,而非简单的执行代码,这对于元编程等高级特性尤为重要。 ####...
.NET Framework 3.5 是 .NET Framework 的一个重要版本,它包含了以前所有版本的功能,并引入了 LINQ、WF(Windows Workflow Foundation)、WCF(Windows Communication Foundation)等新技术。C# 3.0 作为 .NET ...
2. 扩展方法(Extension Methods):这是一种可以在不修改原有类的情况下向其添加新方法的机制,增强了代码的可读性和灵活性。 3. 集合初始化器(Collection Initializers):允许在创建集合对象时一次性初始化其...
- .NET Framework 3.5包含了对WCF(Windows Communication Foundation)、WF、WPF和ASP.NET AJAX的支持,以及大量新类库和API。 8. **多语言支持** - 改进了对多种语言的IDE支持,包括C#、VB.NET、C++、F#等,...
3. **对象初始化器与集合初始化器**:简化了对象实例化和集合填充的过程。 4. **匿名类**:在不需要定义具体类的情况下创建临时对象。 5. **扩展方法**:允许在不修改已有类的情况下为其添加方法。 6. **Lambda...