C#函数式程序设计之泛型(上)
在面向对象语言中,我们可以编写一个元素为某个专用类型(可能需要为此创建一个ListElement)的List类,或者使用一个非常通用、允许添加任何类型元素的基类(在.NET中,首先想到的是System.Object)。这两种方法都有缺点。使用通用类型的基类时,很可能会把不相容的元素添加到列表中;如果使用元素为专用类型的列表,只是推迟此问题的发生,因为实际类型是在最后封装到这个类中。泛型提供了这个问题的解决方法。
一般而言,一个泛型类型G可以应用于一个或多个其他类型——如O1、O2等——其思想是,G的实现不需要对Ox等类型知道很多。
C#函数式程序设计之泛型函数
任何方法在为方法签名添加一个或多个类型参数后,就成为泛型。如下所示:
static void OutputThing<T>(T thing)
{
Console.WriteLine("Thing: {0}",thing);
}
这里T是类型变量,它出现在方法名后面的一对尖括号中。这样声明后,这个参数就可以在参数列表中和方法体重当成类型使用。这个方法并不关心这个thing元素和它的类型,它只是把它的值传递给其他方法以进一步处理。
下面是用显示类型参数调用这个函数:
OutputThing<string>("A string");
OutputThing<int>(42);
使用显式类型参数意味着这个类型要受到Visual Studio的智能感知和C#编译器的双重检查。如下面这个调用会产生编译错误信息:
OutputThing<double>("A string");
错误如下:

尽管这个例子很简单,但它说明泛型的一个作用:不是使用类型对象的参数,而是在调用中显式说明类型,这回启动严格的类型检查。
另一方面,许多程序员认为直接使用显式类型过于草率,OutputThing方法也可以像下面这样调用:
OutputThing("A string");
OutputThing(42);
这里使用类型推断机制可以根据字面值推断它的类型。这并不是一个无类型调用,在OutputThing方法中还有一个类型变量T。上面两个调用语句实际上T分别代表了string和int,编译器会在编译时自动为它们替换为该类型。
然而,许多程序员把C#类型推断看成是一个只有在必要时才使用的功能,而不是一个任何时候都可以使用的通用工具,这是正确的,使用类型推断,会让复杂的代码的可读性变差。
下面是一个稍微复杂(同时也比较有用)的有关泛型的李子
static void Apply<T>(IEnumerable<T> sequence, Action<T> action)
{
foreach (T item in sequence)
{
action(item);
}
}
本例中也有一个类型参数T,但是它作用在这个方法的两个参数上,它们之间存在一种关联:第一个参数是事件序列,第二个参数是一个委托,此委托作用的参数就是在此序列中的事件类型。这正是泛型表现出强大功能的地方,如果不使用泛型,但仍然希望此方法可以灵活应用于不同类型,就无法表现出这种关联性。
泛型元素并不关心类型本身。下面是对Apply方法的调用:
var values = new List<int> { 10, 20, 35 };
Apply(values, v => Console.WriteLine("Value:{0}", v));
调用Apply方法,但是省略了泛型参数,编译器需要推断Apply调用语句中参数T的类型,为此需要检查参数。
C#函数式程序设计之泛型类
也可以给类添加类型信息。在这种情况下,类型参数的作用域是整个类,但其用法与前面完全一样:它代表某个类型。下面的例子是一个链表的实现(不完整):
public class ListItem<T>
{
public ListItem(T value)
{
this.value = value;
} private ListItem(T value, ListItem<T> next)
: this(value)
{
this.next = next;
} private readonly T value;
public T Value
{
get { return value; }
} private ListItem<T> next;
public ListItem<T> Prepend(T value)
{
return new ListItem<T>(value, this);
}
}
ListItem类有一个泛型参数T,这个参数封装在ListItem容器中,在类中任何需要显式类型的位置都可以使用这个类。使用泛型会使ListItem类更加灵活,因为它可以把任何其他类型的信息封装到链表列中。
同时,泛型系统会使编译器的类型检查功能更强大。上例中的Prepend方法只接受T类型的值。从ListItem类的实例角度来看,T是固定的,换言之,新的值必须与当前实例的值具有相同类型。分析下面的例子:
public static void List()
{
var intItem = new ListItem<int>(10);
var secondItem = intItem.Prepend(20);
var thirdItem = secondItem.Prepend("string");
}
这里,当ListItem类与new关键字一起使用时,要在类名中添加一个类型参数,现在保存在ListItem变量中的实例是类型化的,包含了类型为int的值。其结果是,Prepend方法可以接受一个为int的类型参数,因此,上例会报错:

泛型语法的最后一个部分是多个类型参数。在任何情况下,只要使用类型参数,就不会只有一个。看下面的代码:
public class Test<T1, T2>
{
private Test(T1 val1, T2 val2)
{
this.val1 = val1;
this.val2 = val2;
}
private readonly T1 val1;
public T1 Val1
{
get
{
return val1;
}
} private readonly T2 val2;
public T2 Val2
{
get
{
return val2;
}
}
}
使用多个泛型参数实际上并没有什么特别之处。重要的是必须认识到这是完全可行的,最后一点是:类中的类型参数与方法中的类型参数可以同时使用,但是必须保证它们不会发生冲突。
C#函数式程序设计之泛型(上)的更多相关文章
- C#函数式程序设计之泛型
Intellij修改archetype Plugin配置 2014-03-16 09:26 by 破狼, 204 阅读, 0 评论,收藏, 编辑 Maven archetype plugin为我们提供 ...
- C#函数式程序设计之泛型(下)
C#函数式程序设计之泛型(下) 每当使用泛型类型时,可以通过where字句对泛型添加约束: + 这个例子直观地声明了一个约束:类型T必须与ListItem<string>相匹配.泛型类 ...
- C#函数式程序设计之局部套用与部分应用
函数式设计的核心与函数的应用以及函数如何作为算法的基本模块有关.利用局部套用技术可以把所有函数看成是函数类的成员,这些函数只有一个形参,有了局部套用,才有部分应用.部分应用是使函数模块化成为可能的两个 ...
- C#函数式程序设计之函数、委托和Lambda表达式
C#函数式程序设计之函数.委托和Lambda表达式 C#函数式程序设计之函数.委托和Lambda表达式 相信很多人都听说过函数式编程,提到函数式程序设计,脑海里涌现出来更多的是Lisp.Haske ...
- C#函数式程序设计之代码即数据
自3.5版本以来,.NET以及微软的.NET语言开始支持表达式树.它们为这些语言的某个特定子集提供了eval形式的求值功能.考虑下面这个简单的Lambda表达式: Func<int, int, ...
- C#函数式程序设计之用闭包封装数据
如果一个程序设计语言能够用高阶函数解决问题,则意味着数据作用域问题已十分突出.当函数可以当成参数和返回值在函数之间进行传递时,编译器利用闭包扩展变量的作用域,以保证随时能得到所需要的数据. C#函数式 ...
- 2018 ACM-ICPC 中国大学生程序设计竞赛线上赛 H题 Rock Paper Scissors Lizard Spock.(FFT字符串匹配)
2018 ACM-ICPC 中国大学生程序设计竞赛线上赛:https://www.jisuanke.com/contest/1227 题目链接:https://nanti.jisuanke.com/t ...
- 2018 ACM-ICPC 中国大学生程序设计竞赛线上赛 F题 Clever King(最小割)
2018 ACM-ICPC 中国大学生程序设计竞赛线上赛:https://www.jisuanke.com/contest/1227 题目链接:https://nanti.jisuanke.com/t ...
- C#函数式程序设计之惰性列表工具——迭代器
有效地处理数据时当今程序设计语言和框架的一个任务..NET拥有一个精心构建的集合类系统,它利用迭代器的功能实现对数据的顺序访问. 惰性枚举是一个迭代方法,其核心思想是只在需要的时候才去读取数据.这个思 ...
随机推荐
- OD 实验(一) - 修改程序标题
需要修改的程序 把 I love fishc.com 修改为 hello world sch01ar 用 OD 打开程序 在程序入口处开始一直按 F8 运行程序,看看在哪里弹出对话框 运行到该地址的时 ...
- Word域介绍文章
https://www.cnblogs.com/ahuo/archive/2007/05/04/735520.html pageref 书签名 :返回书签所在页码 styleref 标题 1 在当前位 ...
- Python if判断语句
a=input('输入你的用户名:') if a == "lilei": print('李磊,等你好久了') elif a == "wanghui": prin ...
- linux 进程通信 :流套接字
消息队列是可以实现没有共同关系的进程之间的通信.Socket则可以实现不同计算机的不同进程之间的通信. //地址的结构体 struct sockaddr_in{ short int sin_famil ...
- Swift 延迟运行代码
// // DelayRun.swift // // Created by XuQing on 16/7/1. // Copyright © 2016年 xuqing. All rights rese ...
- 「小程序JAVA实战」小程序注册与后端联调(35)
转自:https://idig8.com/2018/09/01/xiaochengxujavashizhanxiaochengxuzhuceyuhouduanliandiao35/ 小程序的后端spr ...
- centos7 时间修改
转子 http://blog.csdn.net/kuluzs/article/details/52825331 在CentOS 6版本,时间设置有date.hwclock命令,从CentOS 7开始, ...
- new 约束
[new 约束] new 约束指定泛型类声明中的任何类型参数都必须有公共的无参数构造函数.如果要使用 new 约束,则该类型不能为抽象类型. 当泛型类创建类型的新实例,请将 new 约束应用于类型参数 ...
- 高性能Web服务器Nginx的配置与部署研究(14)平滑升级你的Nginx
1.概述(可以直接跳过看第2部分) Nginx方便地帮助我们实现了平滑升级.其原理简单概括,就是: (1)在不停掉老进程的情况下,启动新进程. (2)老进程负责处理仍然没有处理完的请求,但不再接受处理 ...
- 平衡二叉树之AVL树
AVL树(命名来源于作者姓名,Adelson-Velskii和Landis),即平衡二叉树,满足以下的条件: 1)它的左子树和右子树都是AVL树 2)左子树和右子树的高度差不能超过1 从条件1可能看出 ...