写点什么

代码契约组件使用详解

2009 年 3 月 01 日

InfoQ 于不久前发表了.NET 代码契约组件提供下载的新闻。这个组件是对.NET 的重要补充,这次我们将提供更为详细的内容。

如果要在.NET 4.0 发布之前使用代码契约,我们可以在 Visual Studio 项目中引用程序集 Microsoft.Contracts.dll,该程序集安装在 %PROGRAMFILES%/Microsoft/Contracts/PublicAssemblies 目录下。.NET 4.0 会在 mscorlib.dll 中包含契约组件。我们可以指定契约验证,可在编译时(静态)或在运行时(动态)执行校验。契约包含几种类型:前置条件(Preconditions)、后置条件(Postconditions)、对象不变量(Object Invariants)、断言(Assertions)、假定(Assumptions)、量词(Quantifiers)、接口契约(Interface Contracts)和抽象方法契约(Abstract Method Contracts)。

前置条件使用 Contract.Requires() 进行定义,如果在编译时使用了符号(Symbol)CONTRACTS_FULL 或 CONTRACTS_PRECONDITIONS,那么 IL 中就会包含其编译结果。例如:

复制代码
Contract.Requires( x ! = null );

如下所示,前置条件通常作为方法体中的参数验证,如下所示:

复制代码
public Rational( int numerator, int denominator) {<br></br>  Contract.Requires( denominator ! = 0 );<br></br>  this .numerator = numerator;<br></br>  this .denominator = denominator;<br></br>}

如果不符合 Contract.Requires() 指定的条件,就会调用 Debug.Assert(false),然后调用 Environment.FailFast()。如果不管在编译时使用哪个符号,您都希望程序集中包含前置条件,那么可以使用 Contract.RequiresAlways()。

当方法结束时,后置条件表示其结果需要满足的契约。它通过 Contract.Ensures() 方法指定,如下例所示:

复制代码
public int Denominator {<br></br> get {<br></br> Contract.Ensures( Contract.Result<int></int>() != 0 );<br></br> return this .denominator;<br></br> }<br></br>}

虽然似乎在返回结果之前就指定了条件,实际它还是会在返回结果之后,调用者得到结果之前进行验证。

对象不变量则为每个实例指定条件。

复制代码
[ContractInvariantMethod]<br></br>protected void ObjectInvariant () {<br></brContract. Invariant ( this .denominator ! = 0 );<br></br>}<br></br>

至于其他类型的契约,断言表示为 Contract.Assert(),假定则表示为 Contract.Assume()。一个失败的 Assert() 会调用 Debug.Assert(false)。假定与运行时断言相似,不同之处在于静态检验的方式。假定用于指定“期望”应该符合的条件,而由于某些限制,该条件无法得到编译器的验证。

接口契约为接口指定条件。它们使用在关联于接口的独立类上,因为接口方法只能声明,而不能拥有方法体。对于抽象方法契约同样如此。

以下为一个使用契约的类:

contracts

可用链接: InfoQ 关于代码契约的新闻代码契约下载(MSI),代码契约的在线文档(PDF),微软研究院的代码契约网站

查看英文原文: Details on Using Code Contracts

2009 年 3 月 01 日 03:491411
用户头像

发布了 109 篇内容, 共 35.8 次阅读, 收获喜欢 9 次。

关注

评论

发布
暂无评论
发现更多内容

【操作系统】存储器管理

学Java的猪猪侠

API 工具链研发的理论基础 - 导读

李宇飞

工具链 API sdk

结合 Flink 学习装饰者模式

shengjk1

flink源码 flink源码分析

Vue中如何在线预览pdf文件

y

2021金三银四总结面试必备清单:字节/蚂蚁金服/腾讯/百度

比伯

Java 程序员 架构 互联网 技术宅

K8s 原生 Serverless 实践:ASK 与 Knative

阿里巴巴云原生

Serverless 容器 云原生 k8s 存储

一年增加 1.2w 星,Dapr 能否引领云原生中间件的未来?

阿里巴巴云原生

容器 微服务 云原生 k8s 中间件

基于docker部署jenkins(一)

李日盛

docker jenkins

Android 系统开发做什么?

吴小龙同学

产品经理第 0 期训练营第九周作业提交

Krystal

一文搞懂 Flink Stream Join原理

shengjk1

flink源码 flink join

继承

y

继承

作业 - 第八章 数据分析

hao hao

Flink SQL 自定义 Source format

shengjk1

flink sql

直播预告|4月1日与你相约腾讯云共探TcaplusDB

TcaplusDB

数据库 nosql 后端 TcaplusDB

携手百度智能云推动实现工业AR空间智能化

百度大脑

AR 百度智能云

户口?大厂?高薪?生活?聊聊应届程序员的职业选择

流沙

职业发展

行业首创,百度自主研发下一代区块链操作系统

CECBC区块链专委会

原子操作

this指向

y

this指针

systemedctl使用指南

happlyfox

3月日更

技术中台之DevOps动态表单体系构建

EAWorld

通俗地理解面向服务的架构(SOA)以及微服务之间的关系

读字节

微服务 软件架构 ESB SOA BPM

产品 0 期 - 第八周作业

曾烧麦

产品训练营

一文搞懂 Flink 处理水印全过程

shengjk1

阿里混沌工程平台实践

心远

阿里巴巴 混沌工程

设计模式--享元模式

学Java的猪猪侠

浅论指针(二)

Integer

c 指针

关于全球央行数字货币实验的若干认识与思考

CECBC区块链专委会

银行

你有没有领导力?

石云升

领导力 28天写作 职场经验 管理经验 3月日更

「产品经理训练营」第八章作业

Sòrγy_じò ぴé

操作系统--虚拟存储器概述

学Java的猪猪侠

演讲经验交流会|ArchSummit 上海站

演讲经验交流会|ArchSummit 上海站

代码契约组件使用详解-InfoQ