问答文章1 问答文章501 问答文章1001 问答文章1501 问答文章2001 问答文章2501 问答文章3001 问答文章3501 问答文章4001 问答文章4501 问答文章5001 问答文章5501 问答文章6001 问答文章6501 问答文章7001 问答文章7501 问答文章8001 问答文章8501 问答文章9001 问答文章9501

javaweb如何解决高并发

发布网友 发布时间:2022-04-26 09:13

我来回答

2个回答

懂视网 时间:2022-04-07 10:15

Java高并发秒杀系统API之Web层开发

既然是Web层的会肯定要先引入SpringMvc了

  • 修改web.xml,引入SpringMvcDispatcherServlet

  • <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"
             metadata-complete="true">    <!--用maven创建的web-app需要修改servlet的版本为3.0-->
    
        <servlet>
            <servlet-name>seckill-dispatchServlet</servlet-name>
            <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>        <!--配置springmvc的配置文件-->
            <init-param>
                <param-name>contextConfigLocation</param-name>
                <param-value>classpath:spring/applicationContext-*.xml</param-value>
            </init-param>
            <load-on-startup>
                1
            </load-on-startup>
        </servlet>
        <servlet-mapping>
            <servlet-name>seckill-dispatchServlet</servlet-name>        <!--直接拦截所有请求,不再采用spring2.0的/*或者*.do方式-->
            <url-pattern>/</url-pattern>
        </servlet-mapping>
    </web-app>

    在这里的话如果你不配置这一段代码的:

    <!--配置springmvc的配置文件-->
            <init-param>
                <param-name>contextConfigLocation</param-name>
                <param-value>classpath:spring/applicationContext-*.xml</param-value>
            </init-param>

    SpringMvc默认就会默认去WEB-INF下查找默认规范的配置文件,像我这里配置的servlet-nameseckill-dispatchServlet的话,则默认会寻找WEB-INF一个名为seckill-dispatchServlet-Servlet.xml的配置文件

    接下来编写Controller SeckillController

    首先在com.suny下建立包为Controller的包,然后在里面新建一个类SeckillController

    package com.suny.controller;/** * Created by 孙建荣 on 17-5-24.下午10:11 */@Controller@RequestMapping("/seckill")public class SeckillController {    private final SeckillService seckillService;    @Autowired
        public SeckillController(SeckillService seckillService) {        this.seckillService = seckillService;
        }    /**     * 进入秒杀列表.     *     * @param model 模型数据,里面放置有秒杀商品的信息     * @return 秒杀列表详情页面     */
        @RequestMapping(value = "/list", method = RequestMethod.GET)    public String list(Model model) {        List<Seckill> seckillList = seckillService.getSeckillList();
            model.addAttribute("list", seckillList);        return "list";
        }    @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";
        }    /**     * 暴露秒杀接口的方法.     *     * @param seckillId 秒杀商品的id     * @return 根据用户秒杀的商品id进行业务逻辑判断,返回不同的json实体结果     */
        @RequestMapping(value = "/{seckillId}/exposer", method = RequestMethod.GET)    @ResponseBody
        public SeckillResult<Exposer> exposer(@PathVariable("seckillId") Long seckillId) {        // 查询秒杀商品的结果
            SeckillResult<Exposer> result;        try {            Exposer exposer = seckillService.exportSeckillUrl(seckillId);
                result = new SeckillResult<>(true, exposer);
            } catch (Exception e) {
                e.printStackTrace();
                result = new SeckillResult<>(false, e.getMessage());
            }        return result;
        }    /**     * 用户执行秒杀,在页面点击相应的秒杀连接,进入后获取对应的参数进行判断,返回相对应的json实体结果,前端再进行处理.     *     * @param seckillId 秒杀的商品,对应的时秒杀的id     * @param md5       一个被混淆的md5加密值     * @param userPhone 参与秒杀用户的额手机号码,当做账号密码使用     * @return 参与秒杀的结果,为json数据     */
        @RequestMapping(value = "/{seckillId}/{md5}/execution", method = RequestMethod.POST)    @ResponseBody
        public SeckillResult<SeckillExecution> execute(@PathVariable("seckillId") long seckillId,                                                   @PathVariable("md5") String md5,                                                   @CookieValue(value = "userPhone", required = false) Long userPhone) {        // 如果用户的手机号码为空的说明没有填写手机号码进行秒杀
            if (userPhone == null) {            return new SeckillResult<>(false, "没有注册");
            }        // 根据用户的手机号码,秒杀商品的id跟md5进行秒杀商品,没异常就是秒杀成功
            try {            // 这里换成储存过程
     SeckillExecution execution = seckillService.executeSeckill(seckillId, userPhone, md5);            return new SeckillResult<>(true, execution);
            } catch (RepeatKillException e1) {            // 重复秒杀
                SeckillExecution execution = new SeckillExecution(seckillId, SeckillStatEnum.REPEAT_KILL);            return new SeckillResult<>(false, execution);
            } catch (SeckillCloseException e2) {            // 秒杀关闭
                SeckillExecution execution = new SeckillExecution(seckillId, SeckillStatEnum.END);            return new SeckillResult<>(false, execution);
            } catch (SeckillException e) {            // 不能判断的异常
                SeckillExecution execution = new SeckillExecution(seckillId, SeckillStatEnum.INNER_ERROR);            return new SeckillResult<>(false, execution);
            }        // 如果有异常就是秒杀失败
        }    /**     * 获取服务器端时间,防止用户篡改客户端时间提前参与秒杀     *     * @return 时间的json数据     */
        @RequestMapping(value = "/time/now", method = RequestMethod.GET)    @ResponseBody
        public SeckillResult<LocalDateTime> time() {        LocalDateTime localDateTime = LocalDateTime.now();        return new SeckillResult<>(true, localDateTime);
        }
    
    
    }

    建立一个全局ajax请求返回类,返回json类型

    SeckillResult:

    package com.suny.dto;/** * 封装所有的ajax请求返回类型,方便返回json * Created by 孙建荣 on 17-5-24.下午10:18 */public class SeckillResult<T> {    private boolean success;    private T data;    private String error;    public SeckillResult() {
        }    public SeckillResult(boolean success, T data) {        this.success = success;        this.data = data;
        }    public SeckillResult(boolean success, String error) {        this.success = success;        this.error = error;
        }    public boolean isSuccess() {        return success;
        }    public void setSuccess(boolean success) {        this.success = success;
        }    public T getData() {        return data;
        }    public void setData(T data) {        this.data = data;
        }    public String getError() {        return error;
        }    public void setError(String error) {        this.error = error;
        }    @Override
        public String toString() {        return "SeckillResult{" +
                    "状态=" + success +
                    ", 数据=" + data +
                    ", 错误消息='" + error + ''' +
                    '}';
        }
    }

    页面的编写

    因为项目的前端页面都是由Bootstrap开发的,所以我们要先去下载Bootstrap或者是使用在线的CDN.
    -Bootstrap中文官网
    -Bootstrap中文文档 使用在线CDN引入的方法:

    <!-- 最新版本的 Bootstrap 核心 CSS 文件 --><link rel="stylesheet" href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"><!-- 可选的 Bootstrap 主题文件(一般不用引入) --><link rel="stylesheet" href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap-theme.min.css" integrity="sha384-rHyoN1iRsVXV4nD0JutlnGaslCJuC7uwjduW9SVrLvRYooPp2bWYgmgJQIXwl/Sp" crossorigin="anonymous"><!-- 最新的 Bootstrap 核心 JavaScript 文件 --><script src="https://cdn.bootcss.com/bootstrap/3.3.7/js/bootstrap.min.js" integrity="sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa" crossorigin="anonymous"></script>

    文档里面写的很详细,然后我这里是使用离线版本的,方便我们本地调试,避免出现什么别的因素干扰我们:

  • 首先下载JQuery,因为Bootstrap就是依赖JQuery

  • 然后下载Bootstrap

  • 然后下载一个倒计时插件jquery.countdown.min.js -再下载一个操作Cookie插件jquery.cookie.min.js 如图放置:
    资源图

  • 首先编写一个公共的头部jsp文件,位于WEB-INFcommon中的head.jsp

  • <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta charset="utf-8">
    <link rel="stylesheet" href="${pageContext.request.contextPath}/resources/plugins/bootstrap-3.3.0/css/bootstrap.min.css" type="text/css">
    <link rel="stylesheet" href="${pageContext.request.contextPath}/resources/plugins/bootstrap-3.3.0/css/bootstrap-theme.min.css" type="text/css">
  • 然后编写一个公共的jstl标签库文件,位于WEB-INFcommon中的tag.jsp

  • <%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
    <%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %>
    <%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %>
  • 编写列表页面,位于WEB-INFcommon中的list.jsp

  • <%@page contentType="text/html; charset=UTF-8" language="java" %>
    <%@include file="common/tag.jsp" %>
    <!DOCTYPE html>
    <html lang="zh-CN">
    <head>
        <title>秒杀列表</title>
        <%@include file="common/head.jsp" %>
    </head>
    <body>
    
    <div class="container">
        <div class="panel panel-default">
            <div class="panel-heading text-center">
                <h2>秒杀列表</h2>
            </div>
    
            <div class="panel-body">
                <table class="table table-hover">
                    <thead>
                    <tr>
                        <td>名称</td>
                        <td>库存</td>
                        <td>开始时间</td>
                        <td>结束时间</td>
                        <td>创建时间</td>
                        <td>详情页</td>
                    </tr>
                    </thead>
                    <tbody>
                    <c:forEach items="${list}" var="sk">
                        <tr>
                            <td>${sk.name}</td>
                            <td>${sk.number}</td>
                            <td><fmt:formatDate value="${sk.startTime}" pattern="yyyy-MM-dd HH:mm:ss"/></td>
                                                     <td><fmt:formatDate value="${sk.endTime}" pattern="yyyy-MM-dd HH:mm:ss"/></td>
                                                     <td><fmt:formatDate value="${sk.createTIme}" pattern="yyyy-MM-dd HH:mm:ss"/></td>
                            <td><a class="btn btn-info" href="/seckill/${sk.seckillId}/detail" target="_blank">详情</a></td>
                        </tr>
                    </c:forEach>
                    </tbody>
                </table>
            </div>
        </div>
    </div>
    </body>
    <script src="${pageContext.request.contextPath}/resources/plugins/jquery.js"></script>
    <script src="${pageContext.request.contextPath}/resources/plugins/bootstrap-3.3.0/js/bootstrap.min.js"></script>
    </html>
  • 编写列表页面,位于WEB-INFcommon中的detail.jsp,秒杀详情页面

  • <%-- Created by IntelliJ IDEA. User: jianrongsun Date: 17-5-25 Time: 下午5:03 To change this template use File | Settings | File Templates.--%><%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <%@include file="common/tag.jsp" %>
    <html>
    <head>
       <title>秒杀商品详情页面</title>
       <%@include file="common/head.jsp" %>
    </head>
    <body>
    <div class="container">
       <div class="panel panel-default">
           <div class="panel-heading">
               <h1>${seckill.name}</h1>
           </div>
           <div class="panel-body">
               <h2 class="text-danger">
                   <span class="glyphicon glyphicon-time"></span>
                   <span class="glyphicon" id="seckill-box"></span>
               </h2>
           </div>
       </div>
    </div>
    
    <div id="killPhoneModal" class="modal fade">
       <div class="modal-dialog">
           <div class="modal-content">
               <div class="modal-header">
                   <h3 class="modal-title text-center">
                       <span class="glyphicon glyphicon-phone"></span>秒杀电话:
                   </h3>
               </div>
           </div>
    
           <div class="modal-body">
               <div class="row">
                   <div class="col-xs-8 col-xs-offset-2">
                       <input type="text" name="killPhone" id="killPhoneKey" placeholder="填写手机号码" class="form-control">
                   </div>
               </div>
           </div>
    
           <div class="modal-footer">
               <span id="killPhoneMessage" class="glyphicon"></span>
               <button type="button" id="killPhoneBtn" class="btn btn-success">
                   <span class="glyphicon glyphicon-phone"></span>
                   提交
               </button>
           </div>
       </div>
    </div>
    </body>
    <script src="${pageContext.request.contextPath}/resources/plugins/jquery.js"></script>
    <script src="${pageContext.request.contextPath}/resources/plugins/bootstrap-3.3.0/js/bootstrap.min.js"></script>
    <script src="${pageContext.request.contextPath}/resources/plugins/jquery.cookie.min.js"></script>
    <script src="${pageContext.request.contextPath}/resources/plugins/jquery.countdown.min.js"></script>
    <script src="${pageContext.request.contextPath}/resources/script/seckill.js"></script>
    <script type="text/javascript">   $(function () {       var startTimeVal = "${seckill.startTime.toLocalDate()} " + seckill.cloneZero("${seckill.startTime.toLocalTime()}");       var endTimeVal = "${seckill.endTime.toLocalDate()} " + seckill.cloneZero("${seckill.endTime.toLocalTime()}");       console.log("startTimeVal========" + startTimeVal);       console.log("endTimeVal========" + endTimeVal);       // 传入参数       seckill.detail.init({           seckillId:${seckill.seckillId},           startTime: startTimeVal,           endTime: endTimeVal       })   })</script>
    
    </html>

    然后把项目运行一下我们又会碰到一个错误就是jstl中的fmt标签格式化时间只能格式化java.Util.Date类型的日期跟时间,而在我们这里我么使用了java8LocalDateTIme,所以解析时间会出异常,这时我们应该想到自己去实现jstl标签来自定义解析这个时间日期 自定义标签步骤如下:

  • /WEB-INF创建目录 tags

  • 然后创建一个文件 localDateTime.tag 在tags目录下

  • localData.tag用来格式化日期

  • localDataTime.tag用来格式化日期跟时间的组合,也就是数据库中的Timestamp类型 -然后在localDataTime.tag中写自己自定义的格式化流程

  • <%--格式化java8的LocalDatime,解决jstl不支持java8时间的问题--%><%@ tag body-content="empty" pageEncoding="UTF-8" trimDirectiveWhitespaces="true" %><%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %><%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %><%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %><%--        这里是定义页面使用标签中的属性设置,<tags:localDataTime dateTime="${sk.createTIme}"/>     --%><%@ attribute name="dateTime" required="true" type="java.time.LocalDateTime" %><%@ attribute name="pattern" required="false" type="java.lang.String" %><%--首选判断日期时间转换规则是否存在,不存在给出默认的规则--%><c:if test="${empty pattern}">
      <c:set var="pattern" value="yyyy-MM-dd HH:mm:ss"/>
    </c:if>
    <c:set var="datetime" value="${dateTime}"/>                        <%-- 获取jsp页面传入的【 日期时间 】,格式为【 2017-5-26T13:59:12  】 --%><c:set var="time" value="${fn:substringAfter(datetime, 'T')}"/>     <%--   获取页面传过来的【时间T】后面的 【  时:分:秒 】的值  --%><c:set var="timeLength" value="${fn:length(time)}"/>                <%--  获取页面传来的 【 时:分:秒 的长度  】 --%><c:set var="generalLength" value="${fn:length('123456')}"/>         <%--  这里定义了一个【Integer】类型的值,值为字符串 【123456 】的长度   --%><c:set var="cloneZero" value=":00"/>                                 <%--   这里设置一个值为【String】的字符串,     --%><%-- 当  时:分:秒 不足6位的时候就说明缺少秒,我们给它自动补充 :00    --%><c:if test="${timeLength lt generalLength}">
      <c:set var="datetimeCloneZero"
             value="${datetime}${cloneZero}"/>          <%--  拼接页面传过来的  【 时:分 】  ,补充一个【秒数】,EL中 + 为相加,非拼接字符串   --%>
      <c:set var="cleandDateTime"
             value="${fn:rep 
    
    
    
    
                                            

    热心网友 时间:2022-04-07 07:23

    你指的高并发量大概有多少?
    几点需要注意:
    尽量使用缓存,包括用户缓存,信息缓存等,多花点内存来做缓存,可以大量减少与数据库的交互,提高性能。
    用jprofiler等工具找出性能瓶颈,减少额外的开销。
    优化数据库查询语句,减少直接使用hibernate等工具的直接生成语句(仅耗时较长的查询做优化)。
    优化数据库结构,多做索引,提高查询效率。
    统计的功能尽量做缓存,或按每天一统计或定时统计相关报表,避免需要时进行统计的功能。

    能使用静态页面的地方尽量使用,减少容器的解析(尽量将动态内容生成静态html来显示)。
    解决以上问题后,使用服务器集群来解决单台的瓶颈问题。
    基本上以上述问题解决后,达到系统最优。

    至于楼上有人提到别用JAVA来做,除非是低层的连接数过大(如大量的端口占用需求),这种情况下考虑直接C来写,其他的可以用JAVA来做。
    声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com
    现在厨师一个月工资是多少,炒菜和配菜,在什 成都厨房工资多少钱一个月 现在厨师一个月工资是多少,炒菜和配菜,在什么地方的价格,请详细点,谢 ... 华硕电脑上没有触摸板设置华硕笔记本电脑触摸板怎么开启 新买的光碟DVD-R是空白的,我想把他制作成一个系统光碟,谁知道怎么制 ... 请问空白光盘刻录车载DVD后,播放时只有目录放不出来是怎么回事啊?_百 ... 请问将一张DVD-R空白光盘作为数据盘使用能够重复写入或者续写数据吗?谢 ... 我现在有DVD-R盘,怎样刻录?一定需要刻录机吗? 做辣椒酱为什么要放酒 做辣椒酱为啥要放酒 2024年区块链的趋势和方向 手机看铜铁价涨跌有什么软件 C#web开发中出现高并发具体处理方法有哪些 如何搭建一个高并发的web框架 调侃狗的句子搞笑有哪些? 聊聊语音聊天里的视频怎么看到啊 web开发怎么解决高并发的问题 怎样能看出血常规化验单铁缺不缺 芝麻信用是重塑阿里金融的那根肋骨? 聊聊语音聊天视频问题 建筑业到底是实体经济还是虚拟经济? 在网上怎么查询铁的当日价格 如何区别铁和钢 为什么说金融机构在金融市场上的地位非常特殊? 区块链项目注册马耳他基金会还可以申请金融数字牌照吗? 如何辨别铁和铝 虚拟经济是什么?什么人可以做这个行业 什么叫虚拟经济??怎样区分其与实体经济? 如何分辨铁质物品? 为什么我不能点亮视频用户图标呢 从投资的角度看,比特币到底是金融工具,还是单纯的资本骗局? 高炉炼铁如何通过看渣、看铁块样品分析出铁水的成份? java web如何解决瞬间高并发 谁知道猪肉哨子怎么弄啊? 狗年了,关于狗的搞笑段子有哪些? 铜和铁怎么看呀?铜什么样子的,铁什么样的 描写小狗很搞笑的段落 猪肉哨子怎么做 java web高并发是什么意思??高并发要涉及到什么技术? 有没有好看的搞笑的可爱小狗电影? 1 java web项目你是如何处理高并发的?2 在高访问期间项目出现了一个bug... 1.javaweb中高并发和多线程之间的有必然联系吗?2.tomcat的多线程和你... 唐朝民间有什么好玩的东西? 四年级搞笑:小动物的作文20小狗 请问java高手,web的高并发请求如何处理啊? 民间游戏有哪些还有怎么玩的 如何利用Java开发高性能,高并发Web应用 web应用 高并发编程和普通编程一样么 怎样做出很香的米线猪肉哨子 盘点一个小狗搞笑的动画片,【免费高清】在线观看百度网盘资源 猪肉哨子