• UID297
  • 登录2017-02-20
  • 粉丝43
  • 关注112
  • 发帖277
  • 主页http://weibo.com/2619518393/
  • 金币1516枚
喜欢达人
社区居民
最爱沙发
忠实会员
原创写手
追星一族
jkxqj 发布于2015-10-23 17:25
1/1277

SpringMVC+Mybatis实战案例

楼层直达
一、项目效果展示
1、列表展示

图片:1.jpg


2、分页功能

图片:2.jpg


3、提取IP功能

图片:3.jpg

图片:4.jpg


4、帮助页面

图片:5.jpg



二、为什么要选择SpringMVC框架?
京东、珍品网等大型网站都在使用SpringMVC框架,它优点多,主要体现在:
1、简单易用,学习成本低,开发效率高。
2、性能灵活,优于struts。
3、大众框架,遇到困难,网上解决方案众多。
三、SpringMVC框架要点
1、SpringMVC的注解类
@Contorller注解
@Contorller
public class IPContorller{
    ...
}
@RequestMapping注解,org.springframework.web.servlet.DispatcherServlet找到控制器并把请求分发到控制器:
@Contorller
@RequestMapping("/ip")
public class IPContorller{
@RequestMapping(value="/index",method=RequestMethod.GET)
public void index90{
   ...
}
}
@Service注解定义业务逻辑类
@Service
public class IPServiceImpl implements IIPService{
    ...
}
@Resource注解实现注入
@Service
public class IPServiceImpl implements IIPService{
   @Resource
   IIpDao dao;
   ...
}
四、为什么要用Mybatis框架?
1、简单易用,会SQL语句就会用Mybatis
2、灵活,完全手写SQL语句,可控性强
3、SQL和代码分离,提高了可维护性
4、大众框架,遇到问题网上有很多解决方案。
看个简单的例子:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="book.map.User">
<select id="findById" parameterType="int" resultType="book.pojo.User">
select * from User where id = #{id}
</select>
</mapper>
五、Mybatis开发流程
Mybatis开发采用DAO层映射机制,只需要简单三步:
1、定义Mybatis配置文件
2、定义DAO类
3、定义XML格式的SQL-Mapper文件
六、SpringMVC与Mybatis框架整合过程中,需要重点注意3个配置文件
1、定义web.xml
2、定义springmvc.xml
3、定义applicationContext.xml
七、初始化项目
1、配置好maven、SpringMVC
2、建立项目目录文件如下:

图片:6.jpg

图片:7.jpg


3、建立数据表如下:

图片:8.jpg


八、关键代码
package com.demo.controller;
     
import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
     
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
     
import com.demo.common.PageParam;
import com.demo.service.TableIpService;
     
@Controller
public class IndexController {
     
    @Resource
    TableIpService service;
         
    @RequestMapping(value = "index")
    public String helloWorld(HttpServletRequest request){
        String currPageStr = request.getParameter("page");
        int currPage = 1;
        try {
            currPage = Integer.parseInt(currPageStr);
        } catch (Exception e) {
        }
             
        // 获取总记录数
        int rowCount = service.getRowCount();
        PageParam pageParam = new PageParam();
        pageParam.setRowCount(rowCount);
        if (pageParam.getTotalPage() < currPage) {
            currPage = pageParam.getTotalPage();
        }
        pageParam.setCurrPage(currPage);
        pageParam = service.getIpListByPage(pageParam);
             
        request.setAttribute("pageParam", pageParam);
             
        return "index";
    }
         
}
package com.demo.dao;
     
import java.util.List;
import java.util.Map;
     
import org.springframework.stereotype.Repository;
     
import com.demo.entity.TableIp;
     
@Repository
public interface ITableIpDao {
     
    int getRowCount();
     
    List<TableIp> selectByParams(Map<String, Object> params);
     
    List<TableIp> fetchByParams(Map<String, Object> params);
     
}
package com.demo.service;
     
import java.util.HashMap;
import java.util.List;
import java.util.Map;
     
import javax.annotation.Resource;
     
import org.springframework.stereotype.Service;
     
import com.demo.common.PageParam;
import com.demo.dao.ITableIpDao;
import com.demo.entity.TableIp;
     
@Service
public class TableIpService {
     
    @Resource
    ITableIpDao dao ;
         
    public int getRowCount() {
        return dao.getRowCount();
    }
     
    public PageParam getIpListByPage(PageParam pageParam) {
        int currPage = pageParam.getCurrPage();
        // limit offset, size
        int offset = (currPage - 1) * PageParam.pageSize ;
        int size = PageParam.pageSize;
        Map<String, Object> params = new HashMap<String, Object>();
        params.put("offset", offset);
        params.put("size", size);
             
        List<TableIp> ipList = dao.selectByParams(params);
        pageParam.setData(ipList);
             
        return pageParam;
    }
     
    public String printIp(String number, String country, String isp) {
        int size = 100;
        try {
            size = Integer.parseInt(number);
        } catch (Exception e) {
        }
        Map<String, Object> params = new HashMap<String, Object>();
        params.put("size", size);
        params.put("country", country);
        params.put("isp", isp);
             
        List<TableIp> ipList = dao.fetchByParams(params);
        StringBuilder sBuilder = new StringBuilder();
        for (TableIp tableIp : ipList) {
            sBuilder.append(tableIp.getIp()).append(":").append(tableIp.getPort()).append(",").append(tableIp.getCountry()).append(",").append(tableIp.getIsp());
            sBuilder.append("\r\n");
        }
             
        return sBuilder.toString();
    }
     
}
package com.demo.controller;
     
import java.io.IOException;
import java.io.PrintWriter;
     
import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
     
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
     
import com.demo.service.TableIpService;
     
@Controller
public class FetchController {
     
    @Resource
    TableIpService service;
         
    @RequestMapping(value = "fetch")
    public String helloWorld(){
        return "fetch";
    }
     
    @RequestMapping(value = "get", method = RequestMethod.POST)
    public void get(HttpServletRequest request, HttpServletResponse response){
        String number = request.getParameter("num");
        String country = request.getParameter("country");
        String isp = request.getParameter("isp");
             
        String txt = service.printIp(number, country, isp);
             
        PrintWriter writer;
        try {
            response.setCharacterEncoding("GBK");
            writer = response.getWriter();
            writer.write(txt);
            writer.flush();
            writer.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
             
    }
}
九、关键配置文件
1、tableIpMapper.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!-- namespace必须指向Dao接口 -->
<mapper namespace="com.demo.dao.ITableIpDao">
     
    <select id="getRowCount" resultType="int">
        select count(*) from table_ip
    </select>
         
    <select id="selectByParams" resultType="com.demo.entity.TableIp" parameterType="map">
        select * from table_ip
        order by findTime desc
        limit ${offset}, ${size}
    </select>
         
    <select id="fetchByParams" resultType="com.demo.entity.TableIp" parameterType="map">
        select * from table_ip
        where 1=1
        <if test="country != null and country != ''">
            and country = '${country}'
        </if>
        <if test="isp != null and isp != ''">
            and isp = '${isp}'
        </if>
        limit ${size}
    </select>
         
</mapper>
2、applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:util="http://www.springframework.org/schema/util" xmlns:context="http://www.springframework.org/schema/context"
    xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:jee="http://www.springframework.org/schema/jee"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xmlns:task="http://www.springframework.org/schema/task"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
        http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-4.0.xsd
        http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-4.0.xsd
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd 
        http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-4.0.xsd 
        http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-4.0.xsd">
     
    <description>Spring公共配置</description>
         
    <!-- 开启定时任务 -->
    <task:annotation-driven/>
     
    <!-- MyBatis配置 -->
    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <!-- 显式指定Mapper文件位置 -->
        <property name="mapperLocations" value="classpath*:/mybatis/*Mapper.xml" />
        <!-- mybatis配置文件路径 -->
        <property name="configLocation" value="classpath:/mybatis/config.xml"/>    
    </bean>
         
    <bean id="sqlSessionTemplate" class="org.mybatis.spring.SqlSessionTemplate">
       <constructor-arg index="0" ref="sqlSessionFactory" />
       <!-- 这个执行器会批量执行更新语句, 还有SIMPLE 和 REUSE -->
       <constructor-arg index="1" value="BATCH" />
    </bean>
     
    <!-- 扫描basePackage接口 -->
    <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
        <!-- 映射器接口文件的包路径, -->
        <property name="basePackage" value="com.demo.dao" />
    </bean>
     
    <!-- 使用annotation定义事务 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true" />
     
    <!-- 数据源配置, 使用Tomcat JDBC连接池 -->
    <bean id="dataSource" class="org.apache.tomcat.jdbc.pool.DataSource" destroy-method="close">
        <!-- Connection Info -->
        <property name="driverClassName" value="${jdbc.driver}" />
        <property name="url" value="${jdbc.url}" />
        <property name="username" value="${jdbc.username}" />
        <property name="password" value="${jdbc.password}" />
     
        <!-- Connection Pooling Info -->
        <property name="maxActive" value="${jdbc.pool.maxActive}" />
        <property name="maxIdle" value="${jdbc.pool.maxIdle}" />
        <property name="minIdle" value="0" />
        <property name="defaultAutoCommit" value="false" />
    </bean>
     
    <!-- production环境 -->
    <beans profile="production">
        <context:property-placeholder ignore-unresolvable="true" file-encoding="utf-8"
        location="classpath:config.properties,classpath:jdbc.properties" />
    </beans>
     
</beans>
jdbc.properties
jdbc.driver=com.mysql.jdbc.Driver
     
jdbc.url=jdbc:mysql://127.0.0.1:3306/ip?useUnicode=true&characterEncoding=utf-8&autoReconnect=true&allowMultiQueries=true
jdbc.username=root
jdbc.password=你的密码
     
#connection pool settings
# maxIdle是最大的空闲连接数,这里取值为20,表示即使没有数据库连接时依然可以保持20个空闲的连接,它们不会被清除,随时处于待命状态
jdbc.pool.maxIdle=20
# maxActive是最大激活连接数,这里取值为190,表示同时最多有190个数据库连接
jdbc.pool.maxActive=190
logback.xml
<?xml version="1.0" encoding="UTF-8" ?>
<configuration>
    <appender name="console" class="ch.qos.logback.core.ConsoleAppender">
        <encoder>
            <pattern>%date{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
        </encoder>
    </appender>
     
   <!-- 时间滚动输出 level为 DEBUG 日志 -->
    <appender name="file-debug" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>DEBUG</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY </onMismatch>
        </filter>
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <FileNamePattern>/export/data/logs/debug.%d{yyyy-MM-dd}.log</FileNamePattern>
            <MaxHistory>30</MaxHistory>
        </rollingPolicy>
        <encoder>
            <pattern>%date{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
        </encoder>
    </appender>
      
    <!-- 时间滚动输出 level为 INFO 日志 -->
    <appender name="file-info" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>INFO</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY </onMismatch>
        </filter>
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <FileNamePattern>/export/data/logs/info.%d{yyyy-MM-dd}.log</FileNamePattern>
            <MaxHistory>30</MaxHistory>
        </rollingPolicy>
        <encoder>
            <pattern>%date{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
        </encoder>
    </appender>
        
    <!-- 时间滚动输出 level为 INFO 日志 -->
    <appender name="file-error" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>ERROR</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY </onMismatch>
        </filter>
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <FileNamePattern>/export/data/logs/error.%d{yyyy-MM-dd}.log</FileNamePattern>
            <MaxHistory>30</MaxHistory>
        </rollingPolicy>
        <encoder>
            <pattern>%date{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
        </encoder>
    </appender>
        
    <!-- 很多人使用Mybatis的时候,控制台不能输出SQL语句,造成调试困难。只需要让DAO层的日志级别调整为DEBUG就可以了 -->
    <logger name="com.dao" level="DEBUG" />
     
    <!-- 指定项目可输出的最低级别日志 -->
    <root level="INFO">
        <appender-ref ref="console" />
        <appender-ref ref="file-debug" />
        <appender-ref ref="file-info" />
        <appender-ref ref="file-error" />
    </root>
</configuration>
十、项目工程源码
http://share.weiyun.com/39856d14f5580b2c6bba8371571a4a8c
欢迎关注我的微信公众号:oldriverNote

0人打赏
  • UID297
  • 登录2017-02-20
  • 粉丝43
  • 关注112
  • 发帖277
  • 主页http://weibo.com/2619518393/
  • 金币1516枚
喜欢达人
社区居民
最爱沙发
忠实会员
原创写手
追星一族
jkxqj 发布于2015-11-30 23:13
沙发F
沙发~!
欢迎关注我的微信公众号:oldriverNote
您需要登录后才可以回帖
发表回复
极贡献
技术问答
专题荟萃
程序人生
视觉设计
Android开发
iOS开发
编程语言
前端开发
后端开发
服务器架构
软件测试
运维方案
创业路上



最热文章墙

  • 60444/343   【精品推荐】200多种Android动画效果的强悍框架,太全了,不看这个,再有动画的问题,不理你了^@^

  • 33651/188   情人节福利,程序员表白的正确姿势:改几行代码就变成自己的表白了

  • 32068/141   省时省力的Android组件群来了,非常棒的原型参考

  • 29333/226   【精品推荐】Android版产品级的音乐播放器源码,功能太强大了,最好的产品原型有木有?

  • 26288/2   超全!整理常用的iOS第三方资源

  • 26209/0   Python爬虫:常用浏览器的useragent

  • 24834/70   原创表白APP,以程序员的姿势备战新年后的7夕,持续完善中!

  • 23141/138   2016抢红包软件及源码

  • 20039/29   麻省理工的一帮疯子,真的实现了随意操控万物!(绝对黑科技)

  • 19878/25   Android工程师面试题大全

  • 19300/27   2016程序员跳槽全攻略

  • 19080/9   GitHub上排名前50的iOS项目:总有一款你用得着

  • 18571/20   码魂:程序员的牛B漫画

  • 17016/149   Android版类似UC浏览器:非常赞,产品级的源码

  • 16922/3   吐槽那些程序员的搞笑牛逼注释

  • 15749/1   iOS 动画总结

  • 15489/10   2016年最全的Android面试考题+答案 精编版

  • 15324/41   一个绚丽的loading动效分析与实现!

  • 14902/73   【持续更新中】Android福利贴(二):资料源码大放送

  • 14862/83   Android小而全的博客源码:非常适合全面掌握开发技巧

  • 14726/10   女程序员的梦,众网友的神回复

  • 14612/44   惊艳的App引导页:背景图片切换加各个页面动画效果

  • 14515/11   年会上现场review代码是怎么样的体验!

  • 14293/23   个人收集的Android 各类功能源代码

  • 14170/5   新一代Android渠道打包工具:1000个渠道包只需要5秒

  • 13352/19   珍藏多年的素材,灵感搜寻网站

  • 13322/53   基于瀑布流的美女图片浏览App,有注释的源代码

  • 12964/17   用JavaScript 来开发iOS和Android 原生应用:React Native开源框架中文版来啦

  • 12861/74   仿京东商城客户端Android最新版,不错的原型和学习资料

  • 12811/15   基于Android支付宝支付设计和开发方案

  • 12597/20   Android福利第三波【Android电子书】

  • 12391/17   什么是真正的黑客:收获12200+Stars,人气远超微软开源VS

  • 12314/94   Android带弹幕的视频播放器源码,来自大名鼎鼎的Bilibili弹幕网站

  • 12258/18   65条最常用正则表达式,你要的都在这里了

  • 12134/70   【精品推荐】类似360安全卫士安Android源码:非常赞的产品原型

  • 11787/7   用程序员的姿势抢过年的火车票

  • 11772/0   iOS中文版资源库,非常全

  • 11723/7   一张图搞定iOS学习路线,非常全面

  • 11700/11   有木有这样一张酷图帮你集齐所有git命令超实用

  • 11220/10   成为Java顶尖程序员 ,看这11本书就够了

  • 10976/18   一张图搞定Android学习路线,非常全面

  • 10892/10   微信支付终于成功了(安卓,iOS),在此分享

  • 10590/44   在线音乐播放器完整版(商用级的源码):非常赞,可听免费高品质专辑

  • 10531/3   基于Node.js的强大爬虫,能直接发布抓取的文章哦

  • 10442/29   【持续更新中】Android福利贴(一):资料源码

  • 10415/0   GitHub iOS 库和框架Top100 

  • 9991/4   46 个非常有用的 PHP 代码片段

  • 9746/3   即时通信第三方库

  • 9722/61   【技巧一】搭配Android Studio,如何实现App远程真机debug?

  • 9550/9   烧了5亿美金,这家神秘的公司即将颠覆人类未来!

  • 9513/8   流媒体视频直播方案

  • 9392/9   B站建开源工作组:APP想支持炫酷弹幕的看过来

  • 9253/18   八个最优秀的Android Studio插件

  • 9202/2   【精品推荐】高质量PHP代码的50个实用技巧:非常值得收藏

  • 9118/9   中国黑客的隐秘江湖:攻守对立,顶尖高手月入千万美元

  • 8687/6   开箱即用!Android四款系统架构工具

  • 8571/3   一张图看清Linux 内核运行原理

  • 8473/10   十大技巧快速提升Android应用开发性能

  • 8450/3   10款GitHub上最火爆的国产开源项目——可以媲美西半球

  • 8139/1   Android性能优化视频,文档以及工具

  • 返回顶部