结构化编程的三重境界:见山是山:朴素的直观逻辑编程

本文通过一个酒店退房业务案例,介绍了结构化编程的基本思想和常见问题。通过对业务需求的分析,逐步展示了如何实现一个合理的退房计费程序,并讨论了其中可能遇到的逻辑陷阱。

《五灯会元》卷十七中,有一则青原惟信禅师的语录:“老僧三十年前未参禅时,见山是山,见水是水。及至后来亲见知识,有个入处,见山不是山,见水不是水。而今得个休歇处,依前见山只是山,见水只是水。”这描述了我们对世界的一个观察和思考的过程,只要我们对事物进行执着的观察和思考,那在认识事物的过程中总是会经过以上的三个不同的过程和境界。

我们在本节,将通过一个案例描述作为程序员应如何对待结构化编程的思考。首先描述业务:在酒店退房业务中有如下业务规则:按酒店的住房规则客户过夜算一天,次日12时以后、18时以前办理退房手续者,酒店加收半天房费;次日18时以后办理退房手续者,加收一天房费。现在我们将按这个业务规则编写一段程序来计算用户住了几天。

在编写具体的代码前,一般笔者的建议是先想一想你要做什么?不要想都不想就开始编写代码,如果是这样的话,你所有的代码都将是无效,或者很难管理。

从需求分析,在酒店退房的业务中至少需要以下数据:入住日期,退房日期,住了多少天。然后计算从入住日期到退房日期过了几个晚上,再以及退房日期的具体时间,计算是否要多算天数。我们按最朴实的逻辑出发,编写第一次的代码。

ContractedBlock.gif ExpandedBlockStart.gif Code
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->staticvoidMain(string[]args)
{
DateTimeindate
=newDateTime(2008,9,30);//入住时间
DateTimeoutdate=newDateTime(2008,10,5);//退房时间
doubledays=(outdate-indate).Days;//计算入住了几天

if(outdate.Hour<=12)//在12点之前
{
days
+=0;
}

if(outdate.Hour>12&&outdate.Hour<=18)//在12点和18点之间
{
days
+=0.5;
}

if(outdate.Hour>18)//在18点之后
{
days
++;
}
System.Console.WriteLine(
"你的入住结算信息/n入住时间{0}/n退房时间{1}/n一共入住了{2}天",indate,outdate,days);

}

从结果的输出可以了解到,客户是在2008年9月30日午夜入住,在2008年10月5日午夜退房(这个入住退房时间实在诡异啊),我们的程序计算得到客户住了5天,这个数据和我们预期的数据是一样的。

但如果我们将退房时间改成以下数据

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> DateTimeindate = new DateTime( 2008 , 9 , 30 ); // 入住时间
DateTimeoutdate = new DateTime( 2008 , 10 , 5 , 12 , 15 , 12 ); // 退房时间

你会惊讶的发现不是我们预期的5.5天,而依然是5天。换句话说,当退房时间是121512秒的时候,程序的逻辑没有进入到以下的判断

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> if (outdate.Hour > 12 && outdate.Hour <= 18 ) // 在12点和18点之间
{
days
+= 0.5 ;
}

冷静的想想,其实计算机并没有做错,的确是我们的逻辑有漏洞。虽然现在退房时间超过了12点,但该日期时间的小时部分的值还是12,因此出现了我们不乐意见到的逻辑错误。所以我们得到一个概念:DataTime结构的Hour成员的值不是我们通常意义上的几点钟,而是时间的小时部分的值。具体是否一个整时,需要我们自己进行判断和处理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值