9.4.2 实现接口和类型转换
我们已经讨论过在 F# 中如何声明接口,如何使用对象表达式,创建实现接口的值。这是一个非常轻量级的解决方案,类似于 lambda 函数。正如 lambda 语法,对于创建函数来说,并不总是合适的选择一样,有时,在命名类中实现接口,是有道理的。
我们将继续使用在本章前面的示例,来看一下用 C# 和 F# 实现接口,我们回顾一下两种语言中的接口声明:
// C# interface declaration
interface IClientTest {
bool Check(Client client);
void Report(Client client);
}
// F# Interface declaration
type ClientTest =
abstract Check : Client –>bool
abstract Report : Client –> unit
接口有两个方法:一个检查客户,另一个把报告输出到屏幕。假设我们要实现接口,使用几个属性计算系数。早些时候,我们使用对象表达式创建了类似的检查,但是,当代码变得更加复杂时,把它移动到单独的类声明,或成为单独的源文件,会更好一些。
清单 9.17 显示了用 C# 实现的检查客户的收入和工作年限,使用了权重和阈值,所有这些都在构造函数中指定。类使用显式接口实现(explicit interface implementation),稍有点不寻常,但我们要等到讨论用 F#实现时,才会知道为什么。
清单 9.17 使用显式接口实现的客户检查 (C#)
class CoefficientTest : IClientTest { <-- 实现 IClientTest
readonly double incomeCoeff,yearsCoeff, minValue;
public CoefficientTest(double ic, doubleyc, double min) { | 把参数值保存在
this.incomeCoeff = ic; | 私有字段中
this.yearsCoeff = yc; |
this.minValue = min; |
}
public void PrintInfo() {
Console.WriteLine("income * {0}+ years * {1} >= {2}",
incomeCoeff, yearsCoeff,minValue);
}
bool IClientTest.Check(Client client){ |
return client.Income * incomeCoeff+ | [1]
client.YearsInJob *yearsCoeff < min; | 实现接口方法
} |
void IClientTest.Report(Client client){ |
Console.Write("Coeffficient {1}less than {0} ", minValue, |
client.Income *incomeCoeff + cl.YearsInJob * yearsCoeff); |
} |
}
要在 C# 中使用明确的语法实现接口成员,我们在写方法时[1],要包括接口的名字,并去掉访问修饰符。这只是一个微小的变化,而更重要的区别是如何使用类。接口中的方法(在这里,是Test 和 Report),在使用类时,不能直接访问;为了调用它们,首先要把类转换成接口类型。我们看一个例子:
var test = new CoefficientTest(0.001, 5.0,50.0);
test.PrintInfo();
var cltest = (IClientTest)test;
if (cltest.Check(john))cltest.Report(john);
我们不能简单地写成 test.Check(john),因为 Check 不是直接作为类的公共方法,只能作为接口的成员使用,因此,可以使用 cltest 值来访问,它的类型为 IClientTest。我们在代码中用明确的类型转换,和 var 关键字,因为这能帮助我们理解,在 F# 中接口实现是如何工作的。另一种选择是把变量类型声明成 IClientTest,然后,把值 test 赋给它,因为C# 编译器将它隐式转换为接口类型。
除了使用显式接口实现以外,类是完全不值一提;我们将使用它来对比 F# 代码。说到这......