`

[转] 三层开发中的层次划分

阅读更多
    先举一个曾经在哪本书上看到的例子:现在你想在1米宽的小溪上建一座桥,你会在上面放块木板就完了。如果想在宽一点的小河上建这桥,你就需要计算木材用料,价格等,如果需要别人帮忙,你还要多一些图纸什么的让别人理解你的想法。现在你要在大江上面建桥,你需要有整体的计划,包括各个方面,比如将来可能的收费和利益分配等问题。
这里讲3层式,其实是针对“大江上面建桥”来的,对于1米宽的小溪,在实际中可能一点用都没有。不过现在我不可能去拿个长江大桥作例子来讲,所以这里还是用这条简单的小溪,讲讲怎么建桥。之所以讲这么多废话,是为了防止部分人看完此文之后“小小一个东西,搞那么麻烦干什么。。”其实这里讲的不是具体的这个例子,而是分层的思想,理解这点非常重要。
下面我就我们大家日常见最多的例子来讲,就是“用户登录”的例子。这个例子很简单,但是麻雀虽小五脏俱全。从数据访问到业务规则到界面全有了。
本文分2个部分,如果只想研究面向对象的思想,对实现已经熟悉,可以跳过第一部分。


第一部分
新建一个空白解决方案。
然后:
①“添加”-“新建项目”-“其他项目”-“企业级模版项目”-“C#生成块”-“数据访问”(数据层,下简称D层)
②“添加”-“新建项目”-“其他项目”-“企业级模版项目”-“C#生成块”-“业务规则”(业务层,下简称C层)
③“添加”-“新建项目”-“其他项目”-“企业级模版项目”-“C#生成块”-“Web用户界面”(界面层,下简称U层)
④右键点“解决方案”-“项目依赖项”,设置U依赖于D、C,C依赖于D。
⑤对U添加引用D、C,对C添加引用D。
到此为止,一个三层的架子建立起来了。
我上面说的很具体很“傻瓜”,知道的人觉得我废话,其实我这段时间很强烈的感觉到非常多的人其实对这个简单的过程完全不了解。虽然不反对建2个“空项目”和1个“Asp net Web应用程序项目”也可以作为3层的框架,而且相当多的人认为其实这些“企业级模板项目”其实就是个空项目,这是一个误区。没错,企业级模板项目你从解决方案资源管理器里看它是个什么也没有的,但是你可以用记事本打开项目文件,看见不同了吧??有些东西在背后,你是看不见的,不过系统已经做好了。也就是说,如果你在C层里的某个类里“using System Data SqlClineit”,或者使用一个SqlConnection对象,编译时候不会出错,但是会在“任务列表”里生成一些“策略警告”,警告你在C层里不要放应该放在D层的东西(虽然就程序来说没错,但是可读性可维护性就打了折扣)而这种功能,空项目是无法給你的。
我们知道建桥需要砖块,应该是先准备好砖再来建桥,不过为了讲解上的顺序性和连贯性,简单性。我们先建桥,建的过程中需要砖块再现做,这样就不会多出来“桥不需要的东西”。注意在实际中,还是应该先准备砖块。
U层其实就是桥,C层是砖块,D层是原料(石头、沙子)。这也解释前面为什么U层要引用、依赖D层(而不是U对C,C对D的层次),因为桥除了需要砖头,其实也需要石头沙子。
我们在U层建一个Login aspx(这里插入一句,我不喜欢去把系统自动生成的WebForm1 aspx拿来改成login或index或直接删除,我一般留着它当测试代码用,等到整个系统冻结再把它移除就可以了。)添加1个TextBox(id=txt),一个DropDownList(id=ddl),一个Button(id=btn)。其中DropDownList用来选择用户名,button是提交按钮, TextBox用来输入密码。
现在我们必须要添加的代码分为2部分:
1、Page_load时对ddl的初始化。
2、btn的click处理。
1: 
private void Page_Load(object sender, System.EventArgs e) 

    
if(!IsPostBack) 
    

        
this.ddl.DataSourse=DataManager.GetOneColunm(“User”,”uid”); //讲解1 
        this.ddl.DataBind(); 
    }
 
}
 

2: 
private void Btn_Click(object sender, System.EventArgs e) 

    
string uid=this.ddl.SelectedValue; 
     
string psw=this.txt.Text; 
     
    
if(psw =””) 
          MessageBox(“空密码!”); 
     
else 
     

          User theUser; 
          
try 
          

               theUser
=new User(uid); //讲解2 
          }
 
          
catch(Exception e) 
          

             MessageBox(e. Message);
//讲解2 
             return
        }
 
        
if(theUser.CheckPsw(psw)) //讲解3 
        
             theUser.SetSessions(); 
             Response.Redirect(“……………..”);  
//登录成功!  
        }
 
        
else 
        

             MessageBox(“密码错误!”);  
        }
 
     }
 
}
 

讲解1:DataManager 是D层中的一个类,提供常见的数据操作。GetOneColunm(string Table,string Colunm)方法返回一个只有1列的DataTable,值为数据库中表名为Table,的Colunm列。
public class DataManager 

    
public DataManager() 
     

     }
 
     
public static DataTable GetOneColunm(string Table,string Colunm) 
     

          
//此处省略相关代码。返回指定表指定列 
     }
 
}
 

其实这个地方演示的是在U层直接绕过C层访问D层的例子,因为该结构逻辑上很简单,而且获取用户名并不是现实社会中的业务逻辑的一部分(仅仅是界面需要,因为在这里其实用成2个TextBox的话完全不需要这一步)

讲解2:定义一个User类的实例。User类的定义可能如下:
public class User 

    
public User(string uid) 
     

          
if(DataManager.IsIn(“user”,”uid="+uid+”’”)) 
               throw "用户不存在"
          
else
               
//User()其他初始化; 
     }
 
     
public bool CheckPsw(string psw) 
     

          
if(DataManager.IsIn(“user”,”uid="+uid+”’ and psw=’”+psw+”’”)) 
               return true
          
else 
               
return false
     }
 
}
 

注意到用户类构造函数中用了个throw来抛出用户不存在的异常,在下面catch的时候用MessageBox(e. Message);来弹出“用户不存在”的错误。这里其实也是为了演示一个层间传递信息的手段,异常也是一种手段,虽然在这里其实可以有其他方式比如返回值,引用参数之类的直接用一个方法来获得用户是否存在的信息,没必要放在构造里,我这么做只是为了演示传递过程,在后面的有讨论这种用法在分层模式下某种特殊情况的应用以解决一些问题。这个类里又用了DataManager类的一个静态方法IsIn(string Table,string str)该方法其实其实是执行 “select * from Table where str”
这个Sql语句并在返回空的时候方法返回false,否则返回true。一个很简单的方法。这里演示了C层对D层的调用。
顺便说一句,因为在VS.Net中,项目的名称会默认地成为项目中的namespace,可以通过把所有自动生成的代码中的namespace改为“解决方案名称”来使3个层可以无缝地自由调用。或者在调用的地方using一下其他层的空间名,看个人喜欢了。比如上面的Login.aspx.cs里需要using2个,而User.cs里要using一个。

讲解3:这里的检查用户密码同样用到User类的一个方法CheckPsw()而这个方法 又用到了IsIn()这里就不多说了。
大家注意到我们在U层的页面里用MessageBox()方法来弹出对话框,其实这个方法写在PageBase.cs里,是U层的另外一个文件,继承Page类,Login类又继承它,这个方法其实是把Response.Write(“<script>alert(\“”+ msg+“\”)</script>”)封装起来了。
到此为止,登录结束,例子的实现也说完了。不过只讲了“然”,没有讲“所以然”。下面开始讲“所以然”。

第二部分
作为对比,我们使用一个不面向对象的,不分层的Asp式的Aspx相同登录作为对比。具体的Asp代码我就不写了,反正登录哪都有。先来看看他们2者发生的遭遇(这是不幸的,却偏偏是经常发生的):
1、  项目经理突然说“不用SqlServer了,换成Access”(正版费用问题)。看看2边分别发生什么:3层这边(A),把DataManager类里的连接改改(在实际情况下,极可能其实是改它的基类,它本身不用改),Web.config中把字符串换掉就完了。Asp式那边(B),同样要改Web.config,同样要改连接什么的,修改量在这个具体的“小溪”例子上几乎相同,在“大桥”例子上B应该会稍微多改点,不过也不会多很多。但是!请注意一点,我们在修改代码的时候,主要时间和精力不是花在“改”这个动作上,而是花在“要改什么地方”上和“寻找需要改的地方”上。在“大桥”上,B需要花费多的多的时间,对大部分文件进行查找和替换。A则仅仅在数据层里,另外2个层不需要任何修改。从这个角度出发我们想到2点原则:
a)    数据层必须要能够保证数据库的变动(任何结构变动、类型变动)对其余各层的不透明性。也就是数据库怎么变,其他层绝对不应该变哪怕1行代码!(web.config是整个应用程序的配置,虽然在物理上存在于U层的文件夹中,但个人更愿意认为它是独立的不属于任何层的,所以这里不计它)
b)    数据层越小越好(如果没有这点原则,我们把整个所有的东西都放在数据层,那当然数据库变动对外面无影响――因为外面几乎没东西――但是这显然不可行)。而且因为前面我们说了,大部分时间花在“找”上面,你小点,找起来也容易点。
2、  客户突然提出B/S版的不好,要换成C/S版的。对于(B)来说,这是晴天霹雳!!他的所有工作都要重新做,(或者几乎所有工作),虽然他有很多代码还可以用,不过他在未来一小段时间就必须不断在“复制-粘贴”中使用以前的代码。(A)发生了什么??如果你细心看会发现(A)之需要新建个项目“Windows用户界面”(和前面一样,添加引用,项目依赖),拖几个控件到上面,把控件名字起成txt,ddl,btn,然后把click代码和Pageload代码复制过去,(居然。。。)连1行代码都不需要修改!!!!当然,这是比较极端的例子(win和web都有TextBox,DropDownList,Button3种控件,而且我们在PageBase里定义的方法MessageBox()又刚好和win里面方法同名。。。)不过尽管有这么多巧合我们仍然可以也愿意相信,在“大桥”上,(A)将比(B)少做很多工作。从这个角度出发我们又想到2点类似原则:
a)    界面层应该保证界面的任何变化都不需要修改其他层的内容(不管这个具体的例子把ddl改为另外一个TextBox,或是把B/S改为C/S)
b)    界面层越小越好(理由同上。)
3、  除开了界面层和数据层,(如果你的方案中只有3个层的话)剩下的就都是逻辑层的内容了。所以和前面的相对应,我们可以得出结论:
a)    逻辑层应当不受数据库和界面变动的影响而需要修改。
b)    逻辑层越大越好(因为另外2层越小越好。。。)
有了最基本的原则,我们应该来讨论下,根据原则,要怎么分层的问题:
1、  PageBase.cs 应该放在哪个层?根据上面的原则,应该放在C层。但是实际上我习惯放在U层,或者放在另外一个(第4个层,通用底层,在比数据层还低的位置)层里。到底放在什么地方,我最开始的做法是在C层,因为按上面归纳的原则,就应该放在C,但是后来一段时间我习惯于“四层式”之后就把它放在通用底层(下简称B层,该层同时也放如本来在D层中的SqlHelper类等,包括原来3层中所有“通用”的类,这里通用的意思是说其他系统也可以用的到而不需要修改,这个层通常不用解决方案名称而用公司、小组名称等作为namespace,在有新项目的时候在建解决方案的时候就可以“添加现有项目”,简单的加进去并不断积累,实践中对提高效率和代码重用有比较大作用。)不过如果只有3层,我现在倾向于把PageBase放在U层。主要因为最近一段潜心研究面向对象的分析设计的心得。说起来又是一大匹布没完,不过我又在前面的“原则”上加1条:“如果某个类,仅为了某层的某种特殊实现而存在,那么它必须放在该层”,比如PageBase是为了U层的特殊实现(B/S实现)而存在,又比如SqlHelper是为了D层的特殊实现(SqlServer数据库)而存在。所以对应的,它们必须分别放在U层和D层(如果不加这条的话按前面他们都该放在C层,因为C层越大越好,而且数据库和界面的变动不需要改动这2个类-虽然它们可能因改动而没有用了,不过还是不需要去修改它们)
2、  Oldjacky曾经和我谈到一个问题:Datagrid中允许作删除操作,但是如果当前仅余下最后一条记录,则不允许这个删除操作!那么该删除应该放在C层还是D层还是U层?我觉得应该从另外一个角度来考虑:
a)    这种“不允许”是“业务规则的不允许”(比如表内的数据表示当前在店里的职员,删除表示职员离开店里-可能去拿货什么的,添加表示职员回来,当柜台只有一名职员时,显然他绝对不能离开去送货),这个时候,此“禁止删除”的操作应该产生在C层。
b)    这种“不允许”是“程序实现的不允许”(比如当这里为空的时候会引起其他地方比如ToString()方法产生“未将对象的引用设置到对象的实例……”的错误,或程序设计者或项目经理的主观愿望希望它“不允许”以此来减少工作量或简化程序)。这个时候,此“禁止删除”可以放在U层(比如上面说的ToString)或D层(比如违反数据库约束)
3、  细心的人可能会发现,前面的登录例子里,用户一共可以获得3种弹出错误分别是“空密码”“密码错误”“用户不存在”,而其中前2个是在U层里做的,“用户不存在”却是在C层里做的(我是指这个字符串)还是开始说的建桥,我这里是用“小溪建桥”来讲解“大江建桥”所以故意在这里转了个没用的圈,就像在计算小溪上这块木板到底够用多少年,其实对小溪没什么意义,只是为了讲解大桥需要而加上去的,毕竟大桥需要这种考虑。我这里假设“用户不存在需要弹出提示”是一种业务逻辑上的需要,而“未输入密码需要提示”则不是业务规则需要(比如实际业务中可以允许空密码,但是项目经理不同意,说一定要密码)在这个登录例子中其实根本没有什么问题,但是在大项目里,如果这个东西不是业务规则的需要,就不应该放在业务层,如果是一种业务规则,就要放在业务层。有助于业务模型与现实实体的衔接,也有益于业务逻辑更好地表现现实实体的特征。
到此为止,我再次归纳出我们的最终的原则:
1、  如果某个类,仅为了某层的某种特殊实现而存在,那么它必须放在该层。
2、  数据层应当在保证数据库变化对其他层不可见的前提下尽量小。
3、  界面层应当在保证界面变化对业务逻辑层不影响的前提下尽量小。
4、  如果某个类不是业务规则的需要,就不应该放在业务层,反之亦然。
5、  逻辑层应当在保证数据库或界面变化不会造成自身影响的前提下尽量大。
以上5点如果发生冲突,在找平衡点的时候,前面的要高于后面的。比如1和3冲突的时候更倾向于使用规则1。

第二部分结束
有一点应该是“编程代码习惯”和“面向对象”的范畴,不过因为和分层有些关系,所以也说一下。“如果你的代码,自己把它翻译成中文并加必要的标点符号后,其他不懂程序的人看了仍然觉得很乱,那么你很可能层没分好”。比如前面的btn的click:

 字符串 用户名是 下拉框 选择值;
 字符串 密码是 输入框 值;
 如果 密码是 空
   对话框(密码空!);
 否则
 {
   用户 这用户;
   尝试
  {
    这用户 是 新的 用户(用户名);
  }
  捕捉(错误)
  {
    对话框(错误 消息);
    返回;
  }
  如果 这用户检查密码(密码)
  {
    这用户 设置状态;
    响应 重定位(“。。。。。”);
  }
  否则
  {
    对话框(密码错误)
  }

代码最好能让不懂的人也能看懂到底在干什么。
最后,oldjacky的Datagrid删除的例子“删除”显然在D层,但是不允许却可能在C或U,如果在U没什么说的了,如果在C,那么这种“不允许”的一个比较合理的实现方法就是在C层里遇到这种情况throw一下。当U层里catch到该throw的时候,禁止删除操作,这样当2个层同时有原因引起禁止时,可以从代码一眼看出这种禁止的来源。类似于前面的2种弹出错误。


temptation 2007-08-06 15:35 发表评论
分享到:
评论

相关推荐

    三层开发案例

    三层架构通常意义上的三层架构就是将整个业务应用划分为:界面层、业务逻辑层、数据访问层。区分层次的目的即为了“高内聚低耦合”的思想。在软件体系架构设计中,分层式结构是最常见,也是最重要的一种结构。微软...

    .net网页开发中的三层架构

    三层架构是.NET网页开发中的一种优秀实践,它通过明确划分职责,增强了系统的灵活性、可维护性和可扩展性。每个层次的独立性意味着可以在不修改其他层次的情况下升级或替换某一层,这对于大型项目和团队合作尤为重要...

    三层开发框架

    提供的MySchool项目源码是学习三层开发的良好实例,初学者可以通过阅读和分析代码了解如何在实际项目中划分和组织这些层。源码中的每个类和方法都可能对应着特定的职责,这有助于理解各层之间的协作机制。 6. 实践...

    程序设计课程体系的维度与层次划分-程序设计-设计.pdf

    程序设计课程体系的维度与层次划分是程序设计教育中的一项重要任务。当前,程序设计课程体系存在一些问题,例如课程体系中高层次的课程比较多,低层次的、应用型的课程不足,教学目标不够明确,课程设置和教学内容...

    C# web 三层架构设计

    在实际开发中,三层架构可能会根据项目需求进一步细化,引入额外的组件和层,比如模型层(Model Layer)和通用类库(Common Library)。 ##### 模型层(Model Layer) 模型层用于封装实体类数据结构,它描述了业务...

    Java Web三层架构的配置详解

    Java Web三层架构是一种常见的软件设计模式,用于构建可扩展、可维护且易于测试的Web应用程序。这个模式将应用逻辑划分为三个主要层次:表现层(Presentation Layer)、业务逻辑层(Business Logic Layer)和数据...

    06 三层开发与数据关系映射

    其中,“三层开发”是一种常用的软件架构设计方式,它将应用程序划分为三个主要的层次:表示层(用户界面层)、业务逻辑层和数据访问层(数据处理层)。这种架构有助于清晰地划分职责,便于团队协作开发,并且能够更...

    使用三层结构开发数据库应用系统

    三层结构是一种常见的软件架构设计模式,特别是在开发数据库应用系统中被广泛应用。这种结构将应用程序划分为三个主要层次:表示层(Presentation Layer)、业务逻辑层(Business Logic Layer)和数据访问层(Data ...

    三层案例即MVC开发模式

    在软件开发领域,三层架构(Three-Tier Architecture)是一种常见的设计模式,它将应用程序分为三个独立的层次:表示层(Presentation Layer)、业务逻辑层(Business Logic Layer)和数据访问层(Data Access Layer...

    三层开发技术文档(zhangmingshao)

    三层开发技术是软件架构设计中的一种常见模式,它将应用逻辑分为三个主要层次:表示层、业务逻辑层和数据访问层。这种结构有利于提高代码的可维护性、可扩展性和可重用性,同时便于团队协作。以下是关于三层开发技术...

    三层CS信息系统开发工具OpenTools及应用

    作为国产快速开发工具的一员,OpenTools也加入了这一行列,不仅能够支持三层C/S架构的快速开发,而且在实际应用中展现出显著的优势。 #### 信息系统的三层客户/服务器结构 ##### 信息系统的逻辑分层和物理分层 从...

    NET,三层开发

    在.NET C#开发中,三层架构的实现往往涉及以下关键技术: - **ADO.NET**:用于直接操作数据库,如SQL Server,提供了DataSet、DataTable、SqlCommand等类。 - **Entity Framework**:ORM框架,通过实体类与数据库表...

    什么是三层架构_表现层/业务逻辑层/数据访问层

    三层架构是一种常见的软件开发模式,旨在提高系统的可维护性、可扩展性和可重用性。这种架构模式将应用程序划分为三个主要部分:表现层(UI)、业务逻辑层(BLL)和数据访问层(DAL)。 表现层,或用户界面层(UI)...

    三层架构入门实例(三层架构优化版)

    三层架构是一种常见的软件设计模式,尤其在Web应用开发中被广泛应用。它将应用程序分为三个主要层次:表现层(Presentation Layer)、业务逻辑层(Business Logic Layer)和数据访问层(Data Access Layer),以此来...

    动态网站三层结构.docx

    中间件在动态网站三层结构中扮演着关键角色,它提供了一种标准的通信机制,使得服务层可以与客户端进行数据交换。在本实验中,中间件帮助实现了服务端和客户端之间的数据传输,使得客户端无需直接操作数据库,而是...

    C#三层架构经典源代码 C#三成架构实例 三层架构入门实例

    三层架构是一种常见的软件设计模式,尤其在企业级应用开发中广泛应用。C#作为.NET框架的主要编程语言,非常适合构建这种架构的系统。三层架构通常包括表示层(Presentation Layer)、业务逻辑层(Business Logic ...

    深入理解三层软件架构

    三层架构作为软件工程中的一个重要组成部分,对于确保软件项目的成功至关重要。 ### 三层软件架构的重要性 #### 1. 引言 软件架构作为软件开发的核心,对整个项目的影响深远。三层架构作为一种常用的分层设计模式...

    c#三层架构项目开发的全过程,包括三层架构源码、架构文档、模块设计说明书、演示PPT等完整

    在IT行业中,C#语言是微软.NET框架下广泛使用的编程语言,尤其在企业级应用开发中,三层架构(也称为N层架构)被普遍采纳,因为它提供了良好的代码组织和职责划分,增强了系统的可维护性和可扩展性。本项目提供了一...

    web开发,三层技术指导

    在Web开发领域,MVC(Model-View-Controller)架构...总之,MVC三层架构在Web开发中扮演着至关重要的角色,通过合理划分职责,提高了开发效率和软件质量。理解和熟练掌握这一模式,对于提升Web开发技能具有深远意义。

    pb11+webservice开发分布式三层应用

    在本主题“pb11+webservice开发分布式三层应用”中,我们将探讨如何利用PowerBuilder 11结合Web服务(WebService)技术来构建分布式三层架构的应用程序。 分布式三层架构是一种常见的软件设计模式,它将应用程序...

Global site tag (gtag.js) - Google Analytics