Spring + SpringMVC + Druid + JPA(Hibernate impl) 给你一个稳妥的后端解决方案

532676955 3年前
   <p>最近手头的工作不太繁重,自己试着倒腾了一套用开源框架组建的 JavaWeb 后端解决方案。</p>    <p>感觉还不错的样子,但实践和项目实战还是有很大的落差,这里只做抛砖引玉之用。</p>    <p>大体采用的开源项目有:Spring + SpringMVC + Druid + JPA(Hibernate Impl)。</p>    <h3><strong>1. 采用到的开源项目漫谈</strong></h3>    <p>Spring 迷人的依赖注入特性, 使其已经稳稳的占据在 JavaEE 项目引用开源项目列表中的上层位置。</p>    <p>秉承低耦合高内聚的遵旨, Spring 提倡的对象工厂解耦类关系的思想已深入到每个攻城狮的心中。</p>    <p>SpringMVC 做为 Spring 的干儿子,最让我沉醉的是她强大的扩展能力,深邃的像大海一样。</p>    <p>前端无论是 freemarker/velocity/jsp...,后端 DAO 层无论是传统的 ORM 还是新近上位的领域模型。</p>    <p>她的态度始终如一,给你360度最贴心的呵护,有一人对你如此,此生足矣。</p>    <p>项目中关于 SpringMVC + Spring 的依赖:</p>    <pre>  <code class="language-java">       <!--spring mvc-->          <dependency>              <groupId>org.springframework</groupId>              <artifactId>spring-webmvc</artifactId>              <version>4.3.3.RELEASE</version>          </dependency>            <!-- Spring-orm -->          <dependency>              <groupId>org.springframework</groupId>              <artifactId>spring-orm</artifactId>              <version>4.3.3.RELEASE</version>          </dependency>            <!-- Spring AOP 动态代理 -->          <dependency>              <groupId>org.aspectj</groupId>              <artifactId>aspectjweaver</artifactId>              <version>1.8.9</version>          </dependency>  </code></pre>    <p>这上面我想说的是 AspectJ 这个东东, AspectJ 是最早、功能比较强大的 AOP 实现之一。</p>    <p>在 Java 领域,AspectJ 中的很多语法结构基本上已成为 AOP 领域的标准。</p>    <p>Spring 也有自己的 Spring-AOP,Spring-AOP 采用运行时生成代理类,底层可以选用 JDK 或者 CGLIB 动态代理。</p>    <p>通俗点,AspectJ 在编译时增强要切入的类,而 Spring-AOP 是在运行时通过代理类增强切入的类,效率和性能可想而知。</p>    <p>所以 Spring 在 2.0 的时候就已经开始支持 AspectJ ,现在到 4.X 的时代已经很完美的和 AspectJ  结合到一起。</p>    <p>Druid 出自阿里巴巴技术团队之手,个人认为是比较好的数据库连接池之一,尤其是监控部分是我的最爱。</p>    <p>项目中的 web.xml 配置监控配置和监控界面:</p>    <pre>  <code class="language-java">   <!--Druid 数据库连接池监控-->      <servlet>          <servlet-name>DruidStatView</servlet-name>          <servlet-class>com.alibaba.druid.support.http.StatViewServlet</servlet-class>      </servlet>      <servlet-mapping>          <servlet-name>DruidStatView</servlet-name>          <url-pattern>/druid/*</url-pattern>      </servlet-mapping>  </code></pre>    <p><img src="https://simg.open-open.com/show/5cce73681d410b4ae482ffe602e0c117.png"></p>    <p>JPA 作为 Sun 公司引入的 ORM 规范,就像是 JDBC 之于各种数据库驱动 Jar,</p>    <p>不要去在意使用了什么样的数据库,用 JDBC 提供的规范方法去撸代码即可。</p>    <p>JPA 制定持久层规范,相同与抽象接口,有 ORM 框架撸具体的实现层。</p>    <p>Sun 想实现 ORM 技术统一,可能不远的将来,你不用在纠结选择什么样子的 ORM 框架。</p>    <p>而现有热门的 ORM 框架会渐渐失去光泽,这毕竟是个漫长的过程,让我们拭目以待。</p>    <h3><strong>2. 方案整体一览</strong></h3>    <p>方案中所有的类都位于 SpringContext 中,由 Spring 统一进行管理。</p>    <p>让 Spring 统一管理的前提是你要告诉有这样一个类需要它管理,目前我接触到的告诉途径有两种。</p>    <p>传统的 xml 配置和注解方式,xml 配置和注解方式各有优劣,比如 xml 配置的优点:</p>    <p>a. 如果你公司项目在引用另外一个公司的 jar,这时候,唯一可行方式为 xml 配置。</p>    <p>b. 如果类之间的依赖关系变动频繁,xml 配置是比较优秀的,改动代码和改动配置文件,无论是技术上还是风险上,xml 都稳赢注解。</p>    <p>注解声明的方式优点:代码和声明在一起,开发的时候不用切来切去,比 xml 配置声明要简单明了的多。</p>    <p>现在很多主流的框架都引入了注解,但也无法摈弃 xml 配置声明的方式。</p>    <p> </p>    <p><img src="https://simg.open-open.com/show/f886ec6a44a8954ccb08cf1e3ae1a019.png"></p>    <p>在这个方案中我使用干净简单注解的方式,controller 包下使用注解@controller,dao-impl 包下使用@Repository,service 包下使用@service。</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/9cb31b39a6aa32a36985a718ff88be27.png"></p>    <p>控制层注入服务实例,服务层注入数据访问层对象,持久层对象由 JAP 进行注解,页面通过控制层来传输和获取数据。</p>    <p>web.xml:</p>    <pre>  <code class="language-java">  <?xml version="1.0" encoding="UTF-8"?>  <web-app xmlns="http://java.sun.com/xml/ns/javaee"           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"           xsi:schemaLocation="http://java.sun.com/xml/ns/javaee            http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"           version="3.0">      <!--Spring mvc -->      <servlet>          <servlet-name>springMVC</servlet-name>          <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>          <init-param>              <param-name>contextConfigLocation</param-name>              <param-value>                  classpath:spring-mvc-base.xml                  classpath:spring-mvc-beans.xml                  classpath:spring-hibernate-beans.xml              </param-value>          </init-param>          <load-on-startup>1</load-on-startup>          <async-supported>true</async-supported>      </servlet>        <servlet-mapping>          <servlet-name>springMVC</servlet-name>          <url-pattern>/</url-pattern>      </servlet-mapping>        <!--Druid 数据库连接池监控-->      <servlet>          <servlet-name>DruidStatView</servlet-name>          <servlet-class>com.alibaba.druid.support.http.StatViewServlet</servlet-class>      </servlet>      <servlet-mapping>          <servlet-name>DruidStatView</servlet-name>          <url-pattern>/druid/*</url-pattern>      </servlet-mapping>  </web-app>    View Code</code></pre>    <p>maven pom.xml:</p>    <pre>  <code class="language-java">  <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"           xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">      <modelVersion>4.0.0</modelVersion>      <groupId>com.rambo</groupId>      <artifactId>sdh</artifactId>      <packaging>war</packaging>      <version>1.0-SNAPSHOT</version>      <name>sdh</name>      <description>spring mvc + Druid + hibernate</description>        <dependencies>          <!--spring mvc-->          <dependency>              <groupId>org.springframework</groupId>              <artifactId>spring-webmvc</artifactId>              <version>4.3.3.RELEASE</version>          </dependency>            <!-- Spring-orm -->          <dependency>              <groupId>org.springframework</groupId>              <artifactId>spring-orm</artifactId>              <version>4.3.3.RELEASE</version>          </dependency>            <!-- Spring AOP 动态代理 -->          <dependency>              <groupId>org.aspectj</groupId>              <artifactId>aspectjweaver</artifactId>              <version>1.8.9</version>          </dependency>            <!-- Hibernate-core -->          <dependency>              <groupId>org.hibernate</groupId>              <artifactId>hibernate-core</artifactId>              <version>4.3.1.Final</version>          </dependency>            <!-- Druid -->          <dependency>              <groupId>com.alibaba</groupId>              <artifactId>druid</artifactId>              <version>1.0.26</version>          </dependency>            <!--mysql 数据驱动-->          <dependency>              <groupId>mysql</groupId>              <artifactId>mysql-connector-java</artifactId>              <version>5.1.38</version>          </dependency>            <!--日志-->          <dependency>              <groupId>log4j</groupId>              <artifactId>log4j</artifactId>              <version>1.2.17</version><!--2012.05 最后版本-->          </dependency>      </dependencies>      <build>          <finalName>sdh</finalName>          <plugins>              <plugin>                  <groupId>org.eclipse.jetty</groupId>                  <artifactId>jetty-maven-plugin</artifactId>                  <version>9.2.11.v20150529</version>                  <configuration>                      <jvmArgs>-XX:PermSize=256M -XX:MaxPermSize=1024M</jvmArgs>                      <webApp>                          <contextPath>/${project.name}</contextPath>                      </webApp>                      <httpConnector>                          <port>4040</port>                      </httpConnector>                      <stopKey>foo</stopKey>                      <stopPort>9998</stopPort>                  </configuration>              </plugin>          </plugins>      </build>  </project>    View Code</code></pre>    <h3><strong>3. DAO 层的种种设计思路</strong></h3>    <p>controller 和 service 层非常容易理解,就是不赘述了。</p>    <p>DAO 层中 BasePo 和 是希望将一些共有的属性抽象在父类当中(属性由具体项目需求决定)。</p>    <pre>  <code class="language-java">  @MappedSuperclass  public class BasePO implements Serializable {      @Id      @Column(length = 32, nullable = true)      @GeneratedValue(generator = "uuid")      @GenericGenerator(name = "uuid", strategy = "uuid")      private String uuid;        @Column(updatable = false)      private Date createDate;        private Date modifyDate;        ....getter/setter  }    BasePO</code></pre>    <p>BaseDaoImpl 希望将一些公共的数据访问方法实现在父类当中(我这里的方法可能有点少,可以由具体项目增加)。</p>    <pre>  <code class="language-java">  @Repository  public class BaseDaoImpl<T, PK extends Serializable> implements BaseDao<T, PK> {        private Class<T> entityClass;      @Resource      protected SessionFactory sessionFactory;        public BaseDaoImpl() {          this.entityClass = null;          Class<?> c = getClass();          Type type = c.getGenericSuperclass(); //反射获取超类          if (type instanceof ParameterizedType) {              Type[] parameterizedType = ((ParameterizedType) type).getActualTypeArguments();              this.entityClass = (Class<T>) parameterizedType[0];          }      }        protected Session getSession() {          return sessionFactory.getCurrentSession();      }        public T getByKey(PK id) {          Assert.notNull(id, "id is required");          return (T) getSession().get(entityClass, id);      }        public T add(T entity) {          Assert.notNull(entity, "entity is required");          getSession().save(entity);          return entity;      }        public T edit(T entity) {          Assert.notNull(entity, "entity is required");          getSession().update(entity);          return entity;      }        public T deleteByKey(PK id) {          Assert.notNull(id, "id is required");          T t = (T) getSession().load(entityClass, id);          getSession().delete(t);          return t;      }  }    BaseDaoImpl</code></pre>    <p>使用 JAP 注解编写业务使用到的持久层对象。</p>    <pre>  <code class="language-java">@Entity  @Table(name = "t_user")  public class User extends BasePO {      @Column(nullable = false)      String name;      @Column(nullable = false)      String pwd;        ....getter/setter  }  </code></pre>    <p>配置启动时扫描 POJO 的动作,至于是新建还是更新都有配置选项,可以自己查阅相关文档。</p>    <pre>  <code class="language-java">    <!-- 配置hibernate session工厂,需添加 spring-orm -->      <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">          <property name="dataSource" ref="dataSource"/>          <property name="hibernateProperties">              <props>                  <prop key="hibernate.hbm2ddl.auto">${hibernate.hbm2ddl.auto}</prop>                  <prop key="hibernate.dialect">${hibernate.dialect}</prop>                  <prop key="hibernate.show_sql">${hibernate.show_sql}</prop>                  <prop key="hibernate.format_sql">${hibernate.format_sql}</prop>              </props>          </property>            <!-- 自动扫描注解方式配置的hibernate类文件 -->          <property name="packagesToScan">              <list>                  <value>com.rambo.sdh.pojo</value>              </list>          </property>      </bean>  </code></pre>    <p>操纵数据库最主要的事务管理,采用 AOP 声明方式,在执行含有数据变动的方法前后进行拦截。</p>    <p>采用 AOP 声明方式进行拦截的好处,不用去关注数据库事务的开启和关闭,将重心放到业务逻辑上面。</p>    <pre>  <code class="language-java"><!-- 配置事务管理器 -->      <bean name="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">          <property name="sessionFactory" ref="sessionFactory"/>      </bean>        <!-- 声明式容器事务管理 ,transaction-manager指定事务管理器为transactionManager- -->      <tx:advice id="transactionAdvice" transaction-manager="transactionManager">          <tx:attributes>              <tx:method name="add*"/>              <tx:method name="save*"/>              <tx:method name="update*"/>              <tx:method name="modify*"/>              <tx:method name="edit*"/>              <tx:method name="delete*"/>              <tx:method name="remove*"/>              <tx:method name="repair"/>              <tx:method name="deleteAndRepair"/>                <tx:method name="get*" propagation="SUPPORTS"/>              <tx:method name="find*" propagation="SUPPORTS"/>              <tx:method name="load*" propagation="SUPPORTS"/>              <tx:method name="search*" propagation="SUPPORTS"/>              <tx:method name="datagrid*" propagation="SUPPORTS"/>              <tx:method name="*" propagation="SUPPORTS"/>          </tx:attributes>      </tx:advice>      <aop:config>          <aop:pointcut id="transactionPointcut" expression="execution(* com.rambo.sdh.service..*Impl.*(..))"/>          <aop:advisor pointcut-ref="transactionPointcut" advice-ref="transactionAdvice"/>      </aop:config>  </code></pre>    <p>上面配置文件的大体意思是说,在包 com.rambo.sdh.service..*Impl.* 下所执行的已 add/save/update.....开头的方法。</p>    <p>方法在执行前后都会被 HibernateTransactionManager 拦截住,进行事务的开启和关闭。</p>    <p>当然还有一些其他的事情,有兴趣可以 debug 源码去一探究竟。</p>    <p>貌似说的也差不多了,该方案为 javaweb 后端解决方案,前端用你想用的渲染技术即可。</p>    <p> </p>    <p> </p>    <p>来自:http://www.cnblogs.com/java-class/p/5951738.html</p>    <p> </p>