hibernate的一些缺陷(转)

例如用户在系统中,保存的信息包括简要信息(用户名、联系电话、Email、性别)和一些图像信息(照片)。

但是在系统设计时,我的设计方式都是遵循业务的需要,设计一个“用户”类,包含用户名、联系电话、Email、性别和照片信息。这个时候我是不会考虑数据库设计的,这是一个设计原则:“ 不因为实现妨碍设计 ”。

在后面的数据库设计中因为照片比较大,所以保存的时候,会设计成两个表:用户简要信息表和用户照片表,两个表通过“用户ID”字段关联。

如果使用JDBC实现比较简单,直接使用SQL,使用联合查询查到用户所有信息,返回成为对象,比较简单。但是因为项目早期的设计开发人员选择了Hibernate,只有沿着这个方向做了,我发现比较困难,因为Hibernate的机制就是一个表对应一个“实体对象”。

没有办法,我只能建立两个实体对象,业务对象User中,改成这样:

class User{
          UserSimpleInfo sinfo;
          UserPhotoIfo   pinfo;
     }

感觉真够滑稽的。也许还有好的解决方案,只是我没有找到而已,我也会继续去寻找。就现在看来,没有找到这方面好的资料。

我现在觉得这是不可思议的一个做法。对象的设计,应该是从业务的分析和业务逻辑中来,和设计没有那么紧密的关系。应该是由设计觉得实现,不应该由实现决定设计。一种实现或是一种框架,不管功能是否强大,都不能“绑架”设计,要设计“屈从”,这是框架设计的一个基本原则。

实际上,一个表对应一个实体对象是很多情况下一个很自然的选择,但并不是必须的做法,如果限制了一定要这么设计实现,那就是非常荒谬的。我经常面向的领域和客户,经常面对的场景里,一个对象拆成几个表是非常有可能的事情。所以,ORM应该支持这种场景。

从另外一个方面看,一个表对应一个实体类是一个有欠考虑的理念。当然,ORM如果支持一个类映射多个表,复杂度增大了很多,用起来也更困难了。没办法,想做通用的东西,肯定比转做一个领域和业务要困难啊。

我本来就不喜欢ORM这东西,这让我更加讨厌Hibernate这类ORM框架,为此特意找了一些网上的资料,觉得这东西还是很不错的,尤其是一些需求简单的场景里,但是支持它的一些说法和理由却站不住脚:

有人说: Hibernate出现的目的,是为了可以让我们这些写代码的,可以更集中精力处理业务代码,而不是把心思放在怎么构建SQL语句。

  我的意见: SQL本身也就是体现业务逻辑的。一个产品,本身包含的不仅有功能需求,还有效率、存储方面等等需求。

有人说: 在还没有O/R MAPPING之前,我们在团队开发的时候,实现一个业务逻辑前的事情,就是跑去问DBA或者找系统的数据库 字典,要先把这个逻辑所用到的字段类型、大小、约束都搞清楚,才能开始做编码工作,因为我们需要构建特定的SQL语句、在代码放入各种的逻辑判断…… 
     在有了O/R MAPPING之后,这种现象才得到基本解脱,因为我们要操作表里的数据,只需要直接对映射类操作即可,O/R MAPPING会自动生成所需要的SQL语句……

我的意见: 类型、大小、约束和需要在代码里放入的逻辑判断本身也是业务逻辑的一个部分。不管你用何种理念和工具,都要考虑和面对的。

有人说: 不使用ORM,那么多的get/set多烦人啊,写那些SQL多烦人啊。而且不容易维护。

我的意见: 即便不使用现有的ORM,也肯定会采用面向对象的方式,把数据库的对象和业务对象封装起来给上层使用。业务层面看到的,也是非常清晰的存、取功能调用而已。不会到处都是JDBC的调用和ResultSet的set和get,更不会SQL满天飞。这在一定程度上其实和Hibernate、MyBatis差不多,但是因为不用考虑通用性,所以设计实现起来比较容易,且都是根据自己产品需要来设计,量体裁衣,最最适合自己的产品使用。

另外说到写SQL,我认为,写好SQL是设计和实现人员的基本功,不能在这里偷懒。我觉得有个人说的(忘了是谁了):“SQL是丑陋的,难以理解的。”这句话是站不住脚的。如果SQL是邪恶的,那么JNI是什么样的呢?SQL是一门非常精炼和优雅的语言,更是一门艺术。SQL中蕴藏了非常简练干净、清晰的概念,非常稳定的架构。

我不会只是从效率方面考虑问题,我更多的考虑是:

对象应该从业务的分析中来,对象和数据库表毫无关系,对象应该能够从数据存储层取出来,应该能通过存储层保存起来------至于什么表格式等等,都不影响这些上层设计。从数据库表产生实体对象很糟糕;从实体对象产生表结构也一样不好。如上一条所说,对象从业务中来;表的设计,是要根据对象的设计和数据库自身的特点、产品或项目的需求(例如效率需求和不同数据库自身的不同特性)而来的。

一个项目和产品,很多时候就是应该设计师、实现者和DBA共同努力来做好的,或者另外一种情况,一个团队里,本身就应该有一个或几个精通数据库或SQL的人,来一起做,不能因为说“SQL我不熟悉”或者“写好SQL太困难”为理由,对效率和存储方面的需求视而不见。或者简单的说“OO和关系数据库直接由天然阻抗”。做好数据持久层,一定是需要熟悉或精通数据库和SQL的人。

或者,你做一个对效率、存储、持续性都没有什么要求的项目或产品,那另当别论。又或者经过一段时间的学习和体验,我会改变自己现在的看法。

时间: 2024-10-13 06:17:30

hibernate的一些缺陷(转)的相关文章

SSH深度历险(一)深入浅出Hibernate架构(一)-------映射解析——七种映射关系

        ORM.全称是(Object Relational Mapping),即对象关系映射.ORM的实现思想就是将关系数据库中表的数据映射成对象.以对象的形式展现.这样开发者就能够把对数据库的操作转化为对这些对象的操作.Hibernate正是实现了这样的思想,达到了方便开发者以面向对象的思想来实现对数据库的操作.    Hibernate在实现ORM功能的时候主要用到的文件有:映射类(*.java).映射文件(*.hbm.xml)和数据库配置文件(*.properties/*.cfg.

从JDBC到hibernate再到mybatis之路

一.传统的JDBC编程 在java开发中,以前都是通过JDBC(Java Data Base Connectivity)与数据库打交道的,至少在ORM(Object Relational Mapping)框架没出现之前是这样,目前常用的ORM框架有JPA.hibernate.mybatis.spring jdbc等,我一开始也是使用JDBC编程,后面开始使用hibernate,有一次开发一个CRM管理系统使用的是Spring JDBC操作数据库,但个人还是不太喜欢这个框架,本人目前使用的最多还是

Hibernate映射解析——七种映射关系

        ORM,全称是(Object Relational Mapping),即对象关系映射.ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现,这样开发人员就可以把对数据库的操作转化为对这些对象的操作.Hibernate正是实现了这种思想,达到了方便开发人员以面向对象的思想来实现对数据库的操作.    Hibernate在实现ORM功能的时候主要用到的文件有:映射类(*.java).映射文件(*.hbm.xml)和数据库配置文件(*.properties/*.cfg

crm项目总结

1.struts2上传文件 上传文件需要几个条件: 1.必须是post   还需改变enctype属性 enctype="multipart/form-data" 2.input  file 3.在action中配置用于接收的文件,文件名属性及其get  set方法 4.调用FileUtils.copefile(file1,file2)方法   file1本地文件  file2为存储在服务中的文件 5.struts2上传文件的大小的限制  2M    通过常量的配置,但是不能根本的解决

Hibernate的七种映射关系之七种关联映射(二)

继续上篇博客 七.Hibernate双向一对多关联映射:让多的一端来维护关系. 主要是解决一对多单向关联的缺陷,而不是需求驱动的. 1.在Student.java实体类里添加Classes引用.private Classes classes; 2.Student.hbm.xml里添加many-to-one标签:<many-to-one name="classes" column="classesid"/>.Classes.hbm.xml在例子(六)里的那

一口一口吃掉Hibernate(五)——一对多单向关联映射

版权声明:本文为博主原创文章,未经博主允许不得转载.如需转载请声明:[转自 http://blog.csdn.net/xiaoxian8023 ] 在上一篇博客<一口一口吃掉Hibernate(四)--多对一单向关联映射>中,介绍了多对一的关联映射,今天就反过来说一下一对多的单向关联映射. 可能有人会对这2篇博客的题目有点混淆不清,跟日常说的关系有点不同.我们日常说的比如父子关系,夫妻关系都是说的双向关系,而现在讨论的则是单向关系,所以也就有了多对一和一对多的说法. 二者的关系其实很简单,只是

Hibernate (开放源代码的对象关系映射框架)

Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它将POJO与数据库表建立映射关系,是一个全自动的orm框架,hibernate可以自动生成SQL语句,自动执行,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库. Hibernate可以应用在任何使用JDBC的场合,既可以在Java的客户端程序使用,也可以在Servlet/JSP的Web应用中使用,最具革命意义的是,Hibernate可以在应用EJB的J2EE架构中取代CMP,完成数据持久

hibernate基本映射

随着对hibernate一步步实践,感觉做一做总结是很有必要的,这样可以让自己的思路更为清晰! 一.知识概括 通过思维导图整体了解下: 二.对应结构图和实现 (一)单向一对一关联映射(one-to-one): 两个对象之间一对的关系,例如:Person(人)-IdCard(身份证) 有两种策略可以实现一对一的关联映射: *主键关联:即让两个对象具有相同的主键值,以表明它们之间的一一对应的关系:数据库表不会有额外的字段来维护它们之间的关系,仅通过表的主键来关联.如下图: 例子:单向一对一主键关联例

【Hibernate步步为营】--(一对多映射)之双向关联

上篇文章讨论了单向关联的一对多映射,在一的一端维护双向的关系这样的做法尽管能实现可是存在非常多缺陷,首先生成非常多多余的SQL语句,由于多的一端不维护关系,仅仅有一的一端维护,在进行操作时一的一端会发出多余的update语句:其次,由于多的一端不知道一的一端存在,所以在保存多的一端时假设外键为null值,而且在设计数据库时关系字段设为非空,则将无法保存数据.由于单向关联一对多存在非常多缺点那就没有其他的办法了吗,能够採用双向关联来优化. 一.一对多双向关联 这里继续採用上篇文章的学生和班级作为演