“MDD”
Model Driven Development 模型驱动开发。取了一个搞笑的标题,其实我没有研究过MDD到底是怎么样的,只是下面的过程我感觉有那么一点这样的意思,遂写出来。
以前涉及到数据库应用的开发,离不开设计数据库表及其关系,这在复杂应用的情况下我觉得仍然是必要的,很大程度上可以起到优化的作用。但是对于一些简单的应用,我实在不愿意跟数据库表打交道,幸好hibernate提供了不错的方法和工具来简化这个过程,只要关注我们的模型和模型间的关系,就可以方便生成数据库表和必要的代码,yeah,很美。
OK,let’s DIY our own “MDD”,just for fun!
我想写一个在线的足球游戏,下面简单的模型是必要的:
球员(Player): 比如齐达内,恩,我中意他。
球队(Team): 比如曼城队
赛事(Match):定置的一项赛事,比如说英超04/05赛季
用户(User):一个玩这个游戏的人
阵容(Squad):一个用户针对某项赛事设置的一个阵容
就先这些。
它们之间有比较简单的关系:
Player n:1 Team
Team n:n Match
User 1:n Squad
Squad n:n Player
这些就是我要的模型和模型间的关系,我想得到POJO类和对应于数据库的ddl文件,那么现在我只要写一个hibernate的映射文件来描述它们就可以了。
Models.hbm.xml
<!--
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 2.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd">
<hibernate-mapping>
<class name="org.smth.j2ee.websoccer.domain.Player" table="players">
<id name="id" column="playerid" type="int">
<generator class="native"/>
</id>
<property name="name" type="string" column="playername" />
<property name="nationality" type="int" length="4" column="nationality" />
<property name="sex" type="int" length="1" column="sex" />
<property name="age" type="int" length="3" column="age" />
<property name="position" type="int" length="4" column="position" />
<property name="description" type="text" column="description" />
<many-to-one name="team" class="org.smth.j2ee.websoccer.domain.Team" column="teamid" />
</class>
<class name="org.smth.j2ee.websoccer.domain.Team" table="teams">
<id name="id" type="int" column="teamid">
<generator class="native"/>
</id>
<property name="name" type="string" column="teamname" />
<property name="nationality" type="int" length="4" column="nationality" />
<property name="level" type="int" length="4" column="level" />
<set
name="players"
inverse="true"
lazy="true"
cascade="none">
<key column="teamid" />
<one-to-many class="org.smth.j2ee.websoccer.domain.Player" />
</set>
<set name="matches" table="match_team" lazy="true">
<key>
<column name="teamid" not-null="true"/>
</key>
<many-to-many class="org.smth.j2ee.websoccer.domain.Match">
<column name="matchid" not-null="true"/>
</many-to-many>
</set>
</class>
<class name="org.smth.j2ee.websoccer.domain.Match" table="matches">
<id name="id" column="matchid" type="int">
<generator class="native" />
</id>
<property name="matchname" type="string" column="matchname" />
<property name="description" type="text" column="description" />
<set name="teams" table="match_team" lazy="true">
<key>
<column name="matchid" not-null="true"/>
</key>
<many-to-many class="org.smth.j2ee.websoccer.domain.Team">
<column name="teamid" not-null="true"/>
</many-to-many>
</set>
</class>
<class name="org.smth.j2ee.websoccer.domain.User" table="users">
<id name="id" column="userid" type="int">
<generator class="native" />
</id>
<property name="username" type="string" column="username" />
<property name="password" type="string" column="password" />
<set
name="squads"
inverse="true"
lazy="true"
cascade="delete">
<key column="userid" />
<one-to-many class="org.smth.j2ee.websoccer.domain.Squad" />
</set>
</class>
<class name="org.smth.j2ee.websoccer.domain.Squad" table="squads">
<id name="id" column="squadid" type="int">
<generator class="native" />
</id>
<property name="type" type="int" length="4" column="type" />
<property name="finance" type="int" column="finance" />
<many-to-one name="user" class="org.smth.j2ee.websoccer.domain.User" column="userid" />
<set name="players" table="squad_player" lazy="true">
<key>
<column name="squadid" not-null="true"/>
</key>
<many-to-many class="org.smth.j2ee.websoccer.domain.Player">
<column name="playerid" not-null="true"/>
</many-to-many>
</set>
</class>
</hibernate-mapping>
-->
可能稍微显的长了一点,不过其实大多是雷同的,对hibernate稍有了解就能看的明白。黄色的部分就是表示关系的,如果你希望关系是双向的,就要在两个类中分别描述它们。
现在只有描述我们的模型的一个文件,拿着这个文件加上hibernate工具我想要的东西就都有了。这些需要两个工具,hbm2java.bat来自hibernate-extensions工具包,SchemaExport.bat来自hibernate包里面的bin目录。
(1)hbm2java.bat 通过映射文件生成POJO类的.java文件
我比较懒,就将Model.hbm.xml复制到了我使用的hibernate-extensions-2.0.2\tools\bin目录下,切换到CMD,执行
>hbm2java.bat Model.hbm.xml
当前目录会生成一个generated目录,生成的类.java文件就在里面了。
(2)下面需要编译这些类,就不用ant了,写了个批处理,下面是我的,很容易改成你的:
compile.bat
@echo off
rem -------------------------------------------------------------------
rem Compile java sourse
rem -------------------------------------------------------------------
set JDBC_DRIVER=D:\j2sdk1.4.2_01\lib\mysql-connector-java-3.0.8-stable-bin.jar
set HIBERNATE_HOME=E:\Ftp\Pub\Softwares\hibernate\hibernate-2.1
set CORELIB=%HIBERNATE_HOME%\lib
set LIB=..\lib
set PROPS=%HIBERNATE_HOME%\src
set CP=%JDBC_DRIVER%;%PROPS%;%HIBERNATE_HOME%\hibernate2.jar;%CORELIB%\commons-logging-1.0.3.jar;%CORELIB%\commons-collections-2.1.jar;%CORELIB%\commons-lang-1.0.1.jar;%CORELIB%\cglib-2.0-rc2.jar;%CORELIB%\dom4j-1.4.jar;%CORELIB%\odmg-3.0.jar;%CORELIB%\xml-apis.jar;%CORELIB%\xerces-2.4.0.jar;%CORELIB%\xalan-2.4.0.jar;%LIB%\jdom.jar;%LIB%\..\hibernate-tools.jar
javac -classpath %CP% org/smth/j2ee/websoccer/domain/*.java
(3)SchemaExport.bat工具生成数据库的ddl文件
我比较懒,就将SchemaExport.bat复制到generated目录下,还要搭配一个hibernate.properties文件。SchemaExport.bat一般要改动一下,设置一下CLASSPATH,很简单,打开看看就知道要改什么了。
我的hibernate.properties
hibernate.connection.driver_class = org.gjt.mm.mysql.Driver
hibernate.connection.url = jdbc:mysql://localhost:3306/websoccer
hibernate.connection.username = root
hibernate.connection.password = ********
hibernate.dialect = net.sf.hibernate.dialect.MySQLDialect
然后generated目录下执行:
>SchemaExport.bat --text --output=models.ddl Models.hbm.xml
当前目录下就有了我想要的models.ddl了,需要什么数据库的,就设置对应的hibernate.properties文件。
感觉是有点MDD的意思。恩,就这些,要做的事情还很多……
Author:oosky@smth
Email: wzt@mails.tsinghua.edu.cn