Структура разработки проекта — SSM

Java база данных Spring MyBatis

1.Spring

Само собой разумеется, как один из лучших примеров открытого исходного кода, разработка проектов повсюду;
Основной IOC-контейнер используется для загрузки бинов (классов в java) — IOC-контейнер Spring используется для управления жизненным циклом бинов, С таким механизмом нам не нужно повторять новую операцию в коде.
aop, аспектно-ориентированное программирование, самое главное в spring — использование транзакций.

2.Spring MVC

Воздействуя на веб-уровень, он эквивалентен контроллеру и, как и действие в struts, используется для обработки запросов пользователей. При этом, по сравнению со struts2, он более мелкозернистый, основан на уровне метода, а struts — на уровне класса.

3.MyBatis

MyBatis — это превосходная структура уровня сохраняемости, которая поддерживает пользовательский SQL, хранимые процедуры и расширенное сопоставление. MyBatis избегает почти всего кода JDBC и ручной настройки параметров и выборки наборов результатов. MyBatis может использовать простой XML или аннотации для настройки и сопоставления исходной информации, а также сопоставления интерфейсов и Java POJO (обычные старые объекты Java, обычные объекты Java) с записями в базе данных. [От:woohoo. не добавил elevate.org/ не добавил elevate-3/ в…]

другие резюмируют

  • Hibernate является мощным, независимым от базы данных и имеет сильные возможности сопоставления O / R. Если вы хорошо владеете Hibernate и правильно инкапсулировали Hibernate, весь код уровня сохраняемости вашего проекта будет довольно простым, и вам нужно будет написать очень мало кода. , скорость разработки очень быстро, очень здорово.
  • Недостатком Hibernate является то, что порог обучения не низкий, порог освоения выше, а то, как спроектировать O/R-отображение, как сбалансировать производительность и объектную модель, и как правильно использовать Hibernate, требует вашего опыта и способностей. сильный Просто сделай это.
  • С MYBATIS легко начать работу, и его можно использовать сразу после изучения.Он обеспечивает функцию автоматической привязки объектов для запросов к базе данных и поддерживает хороший опыт работы с SQL.Он идеально подходит для проектов, которые не предъявляют таких высоких требований к объектной модели.
  • Недостатком MYBATIS является то, что структура по-прежнему относительно проста, а функции по-прежнему отсутствуют.Хотя код привязки данных упрощен, весь базовый запрос к базе данных фактически должен быть написан сам по себе, рабочая нагрузка относительно велика, и она не легко адаптироваться к быстрой модификации базы данных.

    4. Интеграция SSM-фреймворка

    Этот проект будет основан на покупках, в основном включая информацию о продукте и инвентаре [потому что, кстати, я хочу узнать об обработке транзакций], информацию о заказе. Далее будет дано пошаговое описание создания базы данных, описания структуры проекта, конфигурационного файла, бизнес-кода и т.д.

    4.1 Создание базы данных

    1. Товарная таблица
    CREATE TABLE `goods` (
    `goods_id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '商品ID',
    `goodsname` varchar(100) NOT NULL COMMENT '商品名称',
    `number` int(11) NOT NULL COMMENT '商品库存',
     PRIMARY KEY (`goods_id`)
    ) ENGINE=InnoDB AUTO_INCREMENT=1000 DEFAULT CHARSET=utf8 COMMENT='商品表'
    Инициализировать данные таблицы
    INSERT INTO `goods` (`goods_id`, `goodsname`, `number`)
    VALUES (1001, 'SN卫衣', 15)
    2. Форма заказа
    CREATE TABLE `orderinfo` (
    `order_id` varchar(20) NOT NULL COMMENT '订单编号',
    `goods_id` bigint(18) NOT NULL COMMENT '商品ID',
    `user_id` bigint(10) NOT NULL COMMENT '用户ID',
    `order_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '下单时间' ,
     PRIMARY KEY (`order_id`),
     INDEX `idx_order_id` (`order_id`)
    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='订单表'
    Хорошо, пока структура таблицы и данные инициализации завершены, давайте поговорим о структуре проекта на основе Mavan.

    Описание структуры проекта

    Поскольку проект использует maven для управления пакетом jar, давайте сначала вставим его, конфигурацию pom.xml
  • pom.xml
    Чтобы не учиться у друзей, которые ратуют за доктрину заимствования [то есть удаление таких вещей, как xmlns], здесь перечислены только зависимости пакетов jar, от которых зависит проект; этот случай будет основан на принципе «использовать по мере необходимости», чтобы не видеть его в Интернете. В ситуацию вбрасываются всевозможные беспорядочные зависимости, вызывающие трату ресурсов и мешающие чтению.

    <dependencies>
          <!-- 单元测试 -->
          <dependency>
              <groupId>junit</groupId>
              <artifactId>junit</artifactId>
              <version>4.11</version>
          </dependency>
    
          <!-- 1.日志 slf4j-->
          <dependency>
              <groupId>ch.qos.logback</groupId>
              <artifactId>logback-classic</artifactId>
              <version>1.1.1</version>
          </dependency>
    
          <!-- 2.数据库连接驱动 -->
          <dependency>
              <groupId>mysql</groupId>
              <artifactId>mysql-connector-java</artifactId>
              <version>5.1.37</version>
              <scope>runtime</scope>
          </dependency>
          <!-- 2.数据库连接池 -->
          <dependency>
              <groupId>c3p0</groupId>
              <artifactId>c3p0</artifactId>
              <version>0.9.1.2</version>
          </dependency>
    
          <!-- 3.MyBatis 以及 spring-mybatis -->
          <dependency>
              <groupId>org.mybatis</groupId>
              <artifactId>mybatis</artifactId>
              <version>3.3.0</version>
          </dependency>
          <dependency>
              <groupId>org.mybatis</groupId>
              <artifactId>mybatis-spring</artifactId>
              <version>1.2.3</version>
          </dependency>
    
          <!-- 4.Servlet 相关依赖 -->
          <dependency>
              <groupId>taglibs</groupId>
              <artifactId>standard</artifactId>
              <version>1.1.2</version>
          </dependency>
          <dependency>
              <groupId>jstl</groupId>
              <artifactId>jstl</artifactId>
              <version>1.2</version>
          </dependency>
          <dependency>
              <groupId>com.fasterxml.jackson.core</groupId>
              <artifactId>jackson-databind</artifactId>
              <version>2.5.4</version>
          </dependency>
          <dependency>
              <groupId>javax.servlet</groupId>
              <artifactId>javax.servlet-api</artifactId>
              <version>3.1.0</version>
          </dependency>
    
          <!-- 5.Spring -->
    
          <!-- 5.1 Spring核心 :core bean context -->
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-core</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-beans</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-context</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
          <!-- 5.2 Spring jdbc依赖,事务依赖 -->
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-jdbc</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-tx</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
          <!-- 5.3 Spring web依赖>
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-web</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-webmvc</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
          <!-- 5.4 Spring test -->
          <dependency>
              <groupId>org.springframework</groupId>
              <artifactId>spring-test</artifactId>
              <version>4.1.7.RELEASE</version>
          </dependency>
    
          <!-- 6.redis客户端:Jedis【不使用的话可以直接去除】 -->
          <dependency>
              <groupId>redis.clients</groupId>
              <artifactId>jedis</artifactId>
              <version>2.7.3</version>
          </dependency>
          <dependency>
              <groupId>com.dyuproject.protostuff</groupId>
              <artifactId>protostuff-core</artifactId>
              <version>1.0.8</version>
          </dependency>
          <dependency>
              <groupId>com.dyuproject.protostuff</groupId>
              <artifactId>protostuff-runtime</artifactId>
              <version>1.0.8</version>
          </dependency>
    
          <!-- 7.工具类 -->
          <dependency>
              <groupId>commons-collections</groupId>
              <artifactId>commons-collections</artifactId>
              <version>3.2</version>
          </dependency>
      </dependencies>

    *Структурная схема проекта


src/test/java: тестовые классы для junitsrc/main/java:
дао: обработка базы данных
услуга: бизнес-процессинг
enums: перечисление элементов
Метод в mapper:dao соответствует файлу сопоставления mybatis, и Sql находится в нем.
сеть: контроллер, контроллер
сущность: класс сущности в проекте, например: класс товара и класс заказа

конфигурационный файл

  • jdbc.properties

    jdbc.driver=com.mysql.jdbc.Driver
    jdbc.url=jdbc:mysql://serverName:port/dbname?useUnicode=true&characterEncoding=utf8
    jdbc.username=[填写自己的数据库用户名]
    jdbc.password=[填写自己的数据库登录密码]
    • logback.xml
      Здесь непосредственно используется консольный вывод.Если это производственная среда, ее можно настроить в соответствии с конкретными потребностями.
      <?xml version="1.0" encoding="UTF-8"?>
      <configuration debug="true">
      <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
      ch.qos.logback.classic.encoder.PatternLayoutEncoder -->
         <encoder>
             <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
         </encoder>
      </appender>
      <root level="debug">
         <appender-ref ref="STDOUT" />
      </root>
      </configuration>
    • mybatis-config
      В основном это конфигурация файла глобальной конфигурации MyBaties, вы можете настроить некоторые псевдонимы классов, конфигурацию автоинкремента первичного ключа, конфигурацию правил именования верблюжьих регистров и т. д.

      <configuration>
      <!-- 配置全局属性 -->
      <settings>
         <!-- 使用jdbc的getGeneratedKeys获取数据库自增主键值 -->
         <setting name="useGeneratedKeys" value="true" />
      
         <!-- 使用列别名替换列名 默认:true -->
         <setting name="useColumnLabel" value="true" />
      
         <!-- 开启驼峰命名转换:Table{create_time} -> Entity{createTime} -->
         <setting name="mapUnderscoreToCamelCase" value="true" />
      </settings>
      </configuration>
    • Файлы конфигурации, связанные с Spring
      Чтобы более четко понять роль каждого компонента Spring, здесь разделены конфигурация источника данных, конфигурация транзакции и конфигурация преобразователя представления.
      spring-dao.xml
      В основном это специфический процесс конфигурации пружины и интеграции mybatis, в том числе:
      1. Введите файл конфигурации базы данных
      2. Настройте источник данных [пул соединений с базой данных]
      3. Настройте объект SqlSessionFactory.
      4. Настройте и отсканируйте пакет интерфейса Dao, динамически реализуйте интерфейс Dao и внедрите его в контейнер Spring.

      <!-- 1.配置数据库相关参数properties的属性:${url} -->
      <context:property-placeholder location="classpath:jdbc.properties" />
      
      <!-- 2.数据库连接池 -->
      <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
         <!-- 配置连接池属性 -->
         <property name="driverClass" value="${jdbc.driver}" />
         <property name="jdbcUrl" value="${jdbc.url}" />
         <property name="user" value="${jdbc.username}" />
         <property name="password" value="${jdbc.password}" />
      
         <!-- c3p0连接池的私有属性 -->
         <property name="maxPoolSize" value="30" />
         <property name="minPoolSize" value="10" />
         <!-- 关闭连接后不自动commit -->
         <property name="autoCommitOnClose" value="false" />
         <!-- 获取连接超时时间 -->
         <property name="checkoutTimeout" value="10000" />
         <!-- 当获取连接失败重试次数 -->
         <property name="acquireRetryAttempts" value="2" />
      </bean>
      
      <!-- 3.配置SqlSessionFactory对象 -->
      <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
         <!-- 注入数据库连接池 -->
         <property name="dataSource" ref="dataSource" />
         <!-- 配置MyBaties全局配置文件:mybatis-config.xml -->
         <property name="configLocation" value="classpath:mybatis-config.xml" />
         <!-- 扫描entity包 使用别名 -->
         <property name="typeAliasesPackage" value="com.glmapper.framerwork.entity" />
         <!-- 扫描sql配置文件:mapper需要的xml文件 -->
         <property name="mapperLocations" value="com.glmapper.framerwork.mapper/*.xml" />
      </bean>
      
      <!-- 4.配置扫描Dao接口包,动态实现Dao接口,注入到spring容器中 -->
      <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
         <!-- 注入sqlSessionFactory -->
         <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory" />
         <!-- 给出需要扫描Dao接口包 -->
         <property name="basePackage" value="com.glmapper.framerwork.dao" />
      </bean>
    • spring-service
      В реальном процессе разработки транзакции обычно выполняются на сервисном уровне. Поэтому для конфигурации, связанной с транзакциями, используется отдельный файл spring-service.xml.
      <!-- 扫描service包下所有使用注解的类型 -->
      <context:component-scan base-package="com.glmapper.framerwork.service" />
      <!-- 配置事务管理器 -->
      <bean id="transactionManager"
         class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
         <!-- 注入数据库连接池 -->
         <property name="dataSource" ref="dataSource" />
      </bean>
      <!-- 配置基于注解的声明式事务 -->
      <tx:annotation-driven transaction-manager="transactionManager" />
    • spring-web.xml
      Настройте SpringMVC; нужно пояснить, что обычно мы будем настраивать парсинг json2map в реальном процессе разработки. Если он не используется здесь, он не будет опубликован, читатели могут найти его в Интернете.

      <!-- 1.开启SpringMVC注解模式 -->
      <mvc:annotation-driven />
      <!-- 2.静态资源默认servlet配置
         (1)加入对静态资源的处理:js,css,图片等
         (2)允许使用"/"做整体映射
      -->
      <mvc:default-servlet-handler/>
      
      <!-- 3.配置视图解析器ViewResolver -->
      <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
          <property name="viewClass" value="org.springframework.web.servlet.view.JstlView" />
          <property name="prefix" value="/WEB-INF/jsp/" />
          <property name="suffix" value=".jsp" />
      </bean>
      
      <!-- 4.扫描web相关的bean -->
      <context:component-scan base-package="com.glmapper.framerwork.web" />
    • web.xml
      <!-- 编码过滤器 -->  
      <filter>  
         <filter-name>encodingFilter</filter-name>  
         <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>  
         <async-supported>true</async-supported>  
         <init-param>  
             <param-name>encoding</param-name>  
             <param-value>UTF-8</param-value>  
         </init-param>  
      </filter>  
      <filter-mapping>  
         <filter-name>encodingFilter</filter-name>  
         <url-pattern>/*</url-pattern>  
      </filter-mapping>  
      <!-- Spring监听器 -->  
      <listener>  
         <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>  
      </listener>  
      <!-- 防止Spring内存溢出监听器 -->  
      <listener>  
         <listener-class>org.springframework.web.util.IntrospectorCleanupListener</listener-class>  
      </listener> 
      <!-- 配置DispatcherServlet -->
      <servlet>
         <servlet-name>mvc-dispatcher</servlet-name>
         <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
         <!-- 配置springMVC需要加载的配置文件
             spring-dao.xml,spring-service.xml,spring-web.xml
             Mybatis - > spring -> springmvc
          -->
         <init-param>
             <param-name>contextConfigLocation</param-name>
             <param-value>classpath:spring/spring-*.xml</param-value>
         </init-param>
      </servlet>
      <servlet-mapping>
         <servlet-name>mvc-dispatcher</servlet-name>
         <!-- 默认匹配所有的请求 -->
         <url-pattern>/</url-pattern>
      </servlet-mapping>
      На этом все файлы конфигурации закончились, и конкретные ссылки на код будут выполняться ниже.

      Бизнес-код

      Файл xml в маппере здесь выкладывать не буду, а решать вам. . . .
    • Класс сущности: включает товары и заказы

товар

/**
 * 商品信息类
 * @author glmapper
 *
 */
public class Goods {
    private long goodsId;// 商品ID
    private String goodsName;// 商品名称
    private int number;// 商品库存

    public long getGoodsId() {
        return goodsId;
    }
    public void setGoodsId(long goodsId) {
        this.goodsId = goodsId;
    }
    public String getGoodsName() {
        return goodsName;
    }
    public void setGoodsName(String goodsName) {
        this.goodsName = goodsName;
    }
    public int getNumber() {
        return number;
    }
    public void setNumber(int number) {
        this.number = number;
    }
}

Класс заказа

/**
 * 订单信息类
 * @author glmapper
 *
 */
public class OrderInfo {
    private String orderId;//订单ID
    private long goodsId;//商品ID
    private long userId;//用户ID
    private Date orderTime;//下单时间
    public String getOrderId() {
        return orderId;
    }
    public void setOrderId(String orderId) {
        this.orderId = orderId;
    }
    public long getGoodsId() {
        return goodsId;
    }
    public void setGoodsId(long goodsId) {
        this.goodsId = goodsId;
    }
    public long getUserId() {
        return userId;
    }
    public void setUserId(long userId) {
        this.userId = userId;
    }
    public Date getOrderTime() {
        return orderTime;
    }
    public void setOrderTime(Date orderTime) {
        this.orderTime = orderTime;
    }
}
  • продукт
public interface GoodsDao {

    /**
     * 通过ID查询单件商品信息
     * 
     * @param id
     * @return
     */
    Goods queryById(long id);

    /**
     * 查询所有商品信息
     * 
     * @param offset 查询起始位置
     * @param limit 查询条数
     * @return
     */
    List<Goods> queryAll(@Param("offset") int offset, @Param("limit") int limit);

    /**
     * 减少商品库存
     * 
     * @param bookId
     * @return 如果影响行数等于>1,表示更新的记录行数
     */
    int reduceNumber(long goodsId);

}
  • приказ

    public interface OrderInfoDao {
    
      /**
       * 插入订单记录
       * 
       * @param OrderInfo orderInfo
       * @return 插入的行数
       */
      int insertOrderInfo(OrderInfo orderInfo);
    
      /**
       * 通过主键查询订单记录,返回订单实体 
       * @param orderId
       * @return
       */
      OrderInfo queryByOrderId(String orderId);
    }
  • Интерфейс сервиса заказа

    @Service("orderService")
    public class OrderServiceImpl implements OrderService {
      //log生成器
      private Logger logger = LoggerFactory.getLogger(OrderServiceImpl.class);
    
      // 注入dao依赖【商品dao,订单dao】
      @Autowired
      private GoodsDao goodsDao;
      @Autowired
      private OrderInfoDao orderInfoDao;
    
      @Override
      public Goods getById(long goodsId) {
          // TODO Auto-generated method stub
          return goodsDao.queryById(goodsId);
      }
    
      @Override
      public List<Goods> getList(int offset,int limit) {
          // TODO Auto-generated method stub
          return goodsDao.queryAll(offset, limit);
      }
    
      @Override
      @Transactional
      public OrderInfo buyGoods(long goodsId, long userId) {
          //扣减库存,插入订单 =一个事务  如果失败则执行回滚
          try {
              // 减库存
              int update = goodsDao.reduceNumber(goodsId);
              if (update <= 0) {// 库存不足
                  throw new NoNumberException("no number");
              } else {
                  // 执行预约操作
                  OrderInfo orderInfo=new OrderInfo();
                  orderInfo.setGoodsId(goodsId);
                  orderInfo.setUserId(userId);
                  orderInfo.setOrderTime(new Date());
                  String orderId=getRandomOrderId(goodsId);
                  orderInfo.setOrderId(orderId);
                  int insert = orderInfoDao.insertOrderInfo(orderInfo);
                  if (insert <= 0) {// 重复预约
                      throw new RepeatAppointException("repeat appoint");
                  } else {// 预约成功
                      return orderInfo;
                  }
              }
          } catch (Exception e) {
              //这里可以丰富下具体的返回信息
              logger.error("下单失败");
          } 
          return null;
      }
    
      private String getRandomOrderId(long goodsId) {
          SimpleDateFormat dateFormater = new SimpleDateFormat("yyyyMMddhhmmss");
          String prefix=dateFormater.format(new Date());
          String goodsIdStr=goodsId+"";
          String temp="";
          for (int i = 0; i < 6; i++) {
              Random random=new Random(goodsIdStr.length()-1);
              temp+=goodsIdStr.charAt(random.nextInt());
          }
          return prefix+temp;
      }
    }

    Хорошо, пока все основные коды и конфигурационные файлы перечислены; [xml и конкретные контроллеры в маппере выкладывать не буду, думаю, с этим все знакомы. Основная цель этой статьи - разобраться в некоторых моментах моего собственного обучения.Фреймворк SSM будет сочетаться со многими другими технологиями с открытым исходным кодом в реальной разработке приложений, такими как кварц, Redis и т. д. Нынешняя Лиези в этой статье — просто пустая оболочка, для справки.]