秒杀系统Web层设计的实现方法

秒杀系统web层设计的实现方法

一、restful接口设计

使用资源+名词的方式来为url链接命名。例如:

访问详情页的链接可以是: seckill/{seckillid}/detail

二、springmvc配置

1、首先要在web.xml中配置中央控制器。

<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" xsi:schemalocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" version="3.1" metadata-complete="true">   <!-- 修改servlet版本为3.1 -->
  <!-- 配置中央控制器dispatcherservlet -->
  <servlet>     <servlet-name>seckill-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>seckill-dispatcher</servlet-name>     <!-- 默认匹配所有的请求 -->
    <url-pattern>/</url-pattern>   </servlet-mapping> </web-app> 

2、为了让spring管理controller层的bean,需要新建一个spring-web.xml配置文件,

<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:conext="http://www.springframework.org/schema/context" xsi:schemalocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd">   

 <!--配置spring mvc-->

   <!--开启springmvc注解模式-->

   <!--简化配置

   1、自动注册defaultannotationhandlermapping,annotationmethodhandleradapter

   2、提供一系列功能:数据绑定,数字和日期的转化@numberformat,@datatimeformat

     xml,json默认读写支持

   -->

   <mvc:annotation-driven>    <!--servlet-mapping映射路径-->

   <!--静态资源默认servlet配置

     1、加入对静态资源的处理:js,css,img

     2、允许使用/做整体映射

   -->

   <mvc:default-servlet-handler>    <!--配置jsp显示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">    </property></property></property></bean>    <!--扫描web相关的bean-->

   <conext:component-scan base-package="org.seckill.web"> </conext:component-scan></mvc:default-servlet-handler></mvc:annotation-driven></beans> 

三、controller层开发 

项目中的每一个url都刚好对应着controller层的一个方法。我们有两种返回值类型。一种是让页面跳转到某个网页,在model中带上从service层中获得的数据。在下例中,前端的detail.jsp就能够以${seckill.name}取得放在model中的sekill实体的名字。

  /**
   * 秒杀详情页
   * 
   * @param seckillid
   * @param model
   * @return
   */
  @requestmapping(value = "/{seckillid}/detail", method = requestmethod.get)
  public string detail(@pathvariable("seckillid") long seckillid, model model) {
    if (seckillid == null) {
      return "redirect:/seckill/list";
    }
    seckill seckill = seckillservice.getbyid(seckillid);
    if (seckill == null) {
      return "forward:/seckill/list";
    }
    model.addattribute("seckill", seckill);
    return "detail";
  }

另外一种是jsp页面中点击某个按钮,通过ajax来刷新页面的某部分,需要后端给前端一个json格式的数据。使用@responsebody告诉springmvc返回一个json类型的数据seckillresult。由jsp页面在jqeury的回调函数内拿到该json数据,并进行对应的操作。

@requestmapping(value = "/{seckillid}/exposer", 
      method = requestmethod.post, 
      produces = {"application/json;charset=utf-8" })
  @responsebody
  public seckillresult exposer(@pathvariable long seckillid) {
    seckillresult result;
    try {
      exposer exposer = seckillservice.exportseckillurl(seckillid);
      result = new seckillresult(true, exposer);
    } catch (exception e) {
      logger.error(e.getmessage(), e);
      result = new seckillresult(false, e.getmessage());
    }

    return result;
  }

js代码中回调函数的处理方式:

$.post(seckill.url.exposer(seckillid),{},function(result){
      //在回调函数中,执行交互流程
      if(result && result['success']){
        var exposer = result['data'];
        if(exposer['exposed']){
          //开启秒杀
          //获取秒杀地址
          var md5 = exposer['md5'];          
          //绑定一次点击事件,防止连续点击
          var killurl = seckill.url.execution(seckillid,md5);
          console.log("秒杀地址:"+killurl);
});         

四、请求方法的细节处理

1、请求参数的绑定

@requestmapping(value = “/{seckillid}/exposer” 
public seckillresult exposer(@pathvariable long seckillid) 

2、请求方式的限制

@requestmapping(method = requestmethod.post, 

3、请求转发、请求重定向

return “redirect:/seckill/list”;(发送两次请求,浏览器地址改变) 
return “forward:/seckill/list”;(发送一次请求,浏览器地址不变) 

4、数据模型赋值

model.addattribute(“seckill”, seckill); 

5、返回json数据

@requestmapping(value = “/{seckillid}/exposer”, 
method = requestmethod.post, 
produces = {“application/json;charset=utf-8” }) 
@responsebody 

6、cookies访问

@requestmapping(value = "/{seckillid}/{md5}/execution",
      method = requestmethod.post,
      produces = {"application/json;charset=utf-8"})
  @responsebody
  public seckillresult execute(@pathvariable("seckillid") long seckillid,
                          @pathvariable("md5") string md5,
                          @cookievalue(value = "killphone", required = false) long phone) {...}

@cookievalue(value = “killphone”, required = false) long phone)

(1)value(default “”):参数名例如: jsessionid

(2)required(default true):是否请求路头中必须带value指定的参数。如果没有设置cookies我们这个业务也要能够访问并让用户填写相应信息,所以设为false即可。

五、其他

其实课程的这一部分在前端js交互中有很多值得学习的地方,比如jquery的使用,js模块化开发,js交互设计等内容。因为时间关系以及复习侧重点不在js部分的原因,我就暂时不去做总结。

相关文章