`
yiminghe
  • 浏览: 1431560 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

url 映射问题

 
阅读更多

背景

 

url mapping 我最早知道是作为 java web 容器的一个功能点,遵从 servlet 规范 ,大致的形式是通过在配置文件中配置 url 路径和指定 servlet 的对应关系,当请求过来时根据 url 路径来调用相应的服务器端逻辑。

 

例如:

 

<servlet-mapping>
<servlet-name>milk</servlet-name>
<url-pattern>/drink/*</url-pattern>
</servlet-mapping>

 

那么凡是以 /drink/ 开头的 url 路径代表的请求都会被名为 milk 的servlet 处理,可以说 url 路径就是 web 上的方法名称。

 

后来出现了 jsp ,asp 简化了用户配置,基本上是约定优先配置,将 url 路径对应到文件系统的层次关系,url 请求转交给对应的 jsp、asp 代码文件执行。

 

如请求

 

 

/x/z/y.jsp

 

 

则该请求封装后会转交给 x 应用所在目录下 z 目录的 y.jsp 程序代码执行。

 

再往后出现了 struts ,spring 等 web 框架,可以更灵活的做路径分发(支持路径参数),更方便得支持 restful (这里根据 《restful 实战》特指超媒体服务成熟程度模型的第一层: restful url ).,例如 spring3 mvc 可以用注解实现请求路径和函数的对应:

 

 

@RequestMapping("/x/y/{id}") 
public  String  z(@PathParam String id,
@RequestParam String q
) {
}
 

那么当请求

 

/x/y/1?q=2

 

路径以及查询部分的参数会注入到对应的函数中,相对于早期的 servlet 规范开发效率以及可维护性/可读性都有很大的提升。

 

 

前端 mvc

 

随着单页面应用的日益普遍,mvc 分层在前端也越来越多得出现,而其中一点就是:在单页面应用中如何 bookmark,可以像多页面时一样随时记录当前的状态,早期一般用 url hash 来记录,而 html5 history api 则提供了更优雅的解决方案,而这也就不可避免得要将传统由服务器端实现的 router 也要在客户端实现一边,甚至服务器端完全不需要 router,只提供统一数据访问入口(但不符合 http 协议的初衷)。

 

功能需求

 

router 一般要能达到:

 

1. 路由功能

 

可以根据指定的规则,将对应的客户端 hash/url 映射到指定的函数

 

例如片段匹配:

 

/x/y/:id : fn

 

可以匹配 /x/y/z 以及 /x/y/a 到 fn 函数上,但不能匹配 /x/y/z/a

 

再如整段路径匹配

 

/x/y/*path : fn
 

可以匹配 /x/y/z 以及 /x/y/z/a 到 fn 函数,即凡是以 /x/y/ 开头的都可匹配。

 

更进一步,规则可以达到最优匹配,譬如请求:

 

/x/y

 

既可以匹配规则1:

 

/x/:id

 

又可以匹配规则2:

 

/*path

 

那么最优匹配的结果应该是模糊信息最少的匹配(规则1),而后一种通常作为一种异常处理机制(404)。

 

2.参数抽取功能

 

将路径匹配到对应的路由规则是第一步,第二步则是根据规则抽取路径上附带的用户信息,交给对应处理逻辑,用户信息可以由两种方式传递给用户

 

2.1 路径参数(path param)

 

指包含在查询标记(?)之前的参数,如 /x/:id 可以匹配到 /x/y?q=1 那么 id 即为 y

 

2.2 请求/查询参数(request param)

 

即通常服务端通过 request.getParameter 以及客户端通过 location.query 获取到的查询信息.

 

然后同样可以通过参数注入到指定的处理函数,但由于客户端 js 代码会经过压缩,变量名和规则配置名部署阶段会有不一致情况,虽然也可以根据参数位置进行匹配:

 

/x/:id - > fn(id)
 

 

但这种情况下,规则参数配置的变化会连带需要处理函数参数位置的调整。

 

更好的做法则是将参数与参数值作为键值对传递给对应的处理函数

 

 

/x/:id -> fn(pathParam,queryPatam) // pathParam => { id: xx }
 

 

最终在客户端实现router功能后,用户可以随时将单页面应用当前的状态作为地址保存下来,随后再输入地址经过客户端router 后还原对应的应用状态,达到和多页面时类似的效果,只不过这时 router 的功能从后端转移到了前端。

 

 

 

Refer :

 

KISSY MVC

 

spring3 mvc

 

 

 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics