背景

某url性能测试表明,qps单机最高只有4000多,虽然靠堆机器可以解决问题,但是显然不是什么优雅的方案。

试着把controller里的所有的逻辑都屏蔽,只是简单的返回hello world,发现并没有什么用,略略提高了一点,但还是不到5000。显然是什么地方有个坑。

问题查找过程略去不表,坑有两个:

  1. 用了log4j,1.x的版本,这货的性能。。。嗯,不说了,都是泪
  2. spring url mapping也有点坑,在开trace日志的前提下(要知道此时是log也消耗性能),大概消耗了30%的性能。

google & 代码分析

google来google去,看了看spring的源代码,4.2.x。发现这篇文章写的不错。基本说明白了问题。

spring url

盗一张图来描述一下,spring处理url的过程



解释一下什么叫 完全匹配呢?

@RequestMapping(path = "/list/cityId/123", method = RequestMethod.GET) 是完全匹配

@RequestMapping(path = "/list/cityId/{cityId}", method = RequestMethod.GET) 不是完全匹配

看出问题来了吗?

如果path不是一个完全匹配的url,那么就需要遍历所有的path,依次做正则匹配。这个。。。不慢也没有道理吧。慢也就算了,随着一个项目里的url越来越多,那么匹配速度就会越来越慢。。。。

达达的方案

服务端:

1. 在每个@RequestMapping中添加接口对应服务名的信息。

2. 实现自己定义的HandlerMethod查询逻辑,在HandlerMethod注册时记录与之对应的服务名,在查询时通过HTTP请求头中的服务名查表获得HandlerMethod。

客户端:

1. 调用服务时将服务名加入到HTTP请求头中

分析:

* 这样的查询时间复杂度是O(1)的,典型的空间换时间。理论上使用这样的查找逻辑的效率和非RESTful接口的效率是一样的。

* 由于HandlerMethod的注册是在服务启动阶段完成的,且在运行时不会发生改变,所以不用考虑注册的效率以及并发问题。

* SpringMVC提供了一系列的方法可以让我们替换它的组件,所以该方案的可行性很高。

改进

这样做的话,虽然可以解决问题,但是客户端需要配合做修改,不是一个优雅的方案。

实际上,在一个具体的项目中,一般来说path参数一般也就是两种,int和string,比如

/city/id/{id:[\d]+}

/city/shared/{sharedStr:.*}

实际上,上面的path,可以用正则表达式:

@RequestMapping(path = "/list/cityId/{cityId}", method = RequestMethod.GET)

也就是说上面的path pattern可以表示成$ /city/id/{variable} $ 的形式。

那么一个用户的url请求呢? 比如/city/id/123?那就更简单了,/\d{2, }即可。也可以表示成$ /city/id/{variable} $ 的形式。

即用户的请求这可以映射到VariableLikeUrl,通过VariableLikeUrl可以直接找到具体的处理函数。

public class OwnUrlMapping extends RequestMappingHandlerMapping {

    private static final Pattern VARIABLE_PATTERN = Pattern.compile("/\\{[^/]+:[^/]+\\}");
protected Map<String, Pair<RequestMappingInfo, HandlerMethod>> url2method; @Override
protected void handlerMethodsInitialized(Map<RequestMappingInfo, HandlerMethod> handlerMethods) {
url2method = new HashMap<>();
handlerMethods.forEach((info, method) -> { Info2key(info).forEach(s -> {
RequestMappingInfo to variable like url
s = VariableLikeUrl
if (url2method.containsKey(s)) {
throw new IllegalStateException(
"Ambiguous mapping. " + s + " --> " +
method + " alread " + url2method.get(s)); }
url2method.put(s, new Pair<>(info, method));
}
);
});
} @Override
protected HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request) throws Exception {
request to variableLikeUrl
String url = request2case(request.getRequestURI()) + ":" + request.getMethod();
Pair<RequestMappingInfo, HandlerMethod> method = url2method.get(url);
if (method != null) {
// If find it, call it directly
handleMatch(method.getFirst(), lookupPath, request);
return method.getSecond();
} else {
// Could not find, call super matching method
LOG.warn("no matching for " + url);
return super.lookupHandlerMethod(lookupPath, request);
}
}
}

总结

好处

  1. 性能可以得到有效的提升。
  2. 客户端完全没有修改
  3. 服务器controller的代码也不需要什么修改。

问题

  1. 项目by项目,如果出现不按格式写的url,就废了。。。
  2. 实际上spring考虑的东西还是比较多的,处理url,method 貌似还有很多别的东东,上述方法。
  3. 和spring的版本绑定,反正4.x版本都适用。

基本上为项目定制了一个url mapping规则,不能替代spring原有的mapping函数,但是依然有推广意义。毕竟没有几个项目把RequstMapping的所有参数都用全了不是。

Spring urlMapping的更多相关文章

  1. spring mvc 快速入门

    ---------- 转自尚学堂 高淇 --------- Spring  MVC 背景介绍 Spring 框架提供了构建 Web 应用程序的全功能 MVC 模块.使用 Spring 可插入的 MVC ...

  2. 【转载】Spring MVC 整合 Freemarker

    前言 1.为什么要使用Spring MVC呢? 2.为什么要使用Freemarker呢? 3.为什么不使用Struts2呢? 此示例出现的原因就是发现了struts2的性能太差,所以学习Spring ...

  3. dwr和spring的整合

    1.dwr在spring配置文件的配置: <!-- 注意这里新增加的dwr tag, 为使其生效,文件头中要声明namespace --> <dwr:configuration /& ...

  4. spring filter拦截器

    实现的功能:判断用户是否已登录,未登录用户禁止访问任何页面或action,自动跳转到登录页面.比较好的做法是不管什么人都不能直接访问jsp页面,要访问就通过action,这样就变成了一个实实在在的权限 ...

  5. spring MVC学习笔记

    为开发团队选择一款优秀的MVC框架是件难事儿,在众多可行的方案中决择需要很高的经验和水平.你的一个决定会影响团队未来的几年.要考虑方面太多: 1.简单易用,以提高开发效率.使小部分的精力在框架上,大部 ...

  6. spring+hibernate常见异常集合

    spring+hibernate出错小结: (1)java.lang.NoClassDefFoundError: org/hibernate/context/CurrentSessionContext ...

  7. 搭建spring+mybatis+struts2环境的配置文件

    1.web.xml配置 <?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi=& ...

  8. Spring mvc源码url路由-我们到底能走多远系列(38)

    我们到底能走多远系列38 扯淡: 马航的事,挺震惊的.还是多多珍惜身边的人吧. 主题: Spring mvc 作为表现层的框架,整个流程是比较好理解的,毕竟我们做web开发的,最早也经常接触的就是一个 ...

  9. Spring使用拦截器支持国际化(转)

    Spring使用拦截器支持国际化很方便,使用时只需要两个步骤: 一.spring配置 具体配置方式如下: <!-- 资源文件绑定器,文件名称:messages.properties(没有找到时的 ...

随机推荐

  1. linux开机启动mongodb

    方式一(不推荐) ubuntu编辑/etc/rc.local /home/wyt/bin/mongodb-linux-x86_64-ubuntu1404-3.2.8/bin/mongod --dbpa ...

  2. Centos 6.5 rsync+inotify 两台服务器文件实时同步

    rsync和inotify是什么我这里就不在介绍了,有专门的文章介绍这两个工具. 1.两台服务器IP地址分别为: 源服务器:192.168.1.2 目标服务器:192.168.1.3 @todo:从源 ...

  3. Ubuntu14.04+Beanstalkd1.9最佳实践

    目录 [TOC] 1.基本概念 1.1.什么是Beanstalkd?   Beanstalkd 是一个轻量级消息中间件,它最大特点是将自己定位为基于管道 (tube) 和任务 (job) 的工作队列. ...

  4. Mac10.9用brew搭建Eclipse4.4+Maven3.2.3+JDK1.8运行环境

    --------------------------------------- 博文作者:迦壹 博客标题:Mac10.9用brew搭建Eclipse4.4+Maven3.2.3+JDK1.8运行环境 ...

  5. Eclipse for j2ee 无法调式问题

    首先要配置tomcat,window--preference--server--Installed Runtimes, add--选择tomcat--next--选择tomcat的安装目录(根目录)- ...

  6. Effective Java学习笔记

    创建和销毁对象 第一条:考虑用静态工厂方法替代构造器 For example: public static Boolean valueOf(boolean b){ return b ? Boolean ...

  7. 使用curl命令获取文件下载速度

    使用curl可以下载网络内容,那如何获取curl下载时的下载速度呢,使用下面的命令即可: # curl -Lo /dev/null -skw "%{speed_download}\n&quo ...

  8. Files 的值“ < < < < < < < .mine”无效。路径中具有非法字符。

    出现这个问题的原因是,用svn时,发生了冲突.解决方法:先解决代码冲突,然后在你的工程OBJ/DEBUG目录下,找到 工程名.csproj.FileListAbsolute.txt的文件打开并删除含有 ...

  9. 关于Winform 2.0以后多线程不能更新UI的办法

    DotNet 2.0以后Winform在多线程Debug模式下更新UI会报这个错: 线程间操作无效: 从不是创建控件"XXX"的线程访问它. 解决办法如下: 1.在Winform的 ...

  10. SparkConf加载与SparkContext创建(源码阅读四)

    sparkContext创建还没完呢,紧接着前两天,我们继续探索..作死... 紧接着前几天我们继续SparkContext的创建: 接下来从这里我们可以看到,spark开始加载hadoop的配置信息 ...