找回密码
 立即注册
首页 业界区 业界 快速实现一个简单的bigpipe模型

快速实现一个简单的bigpipe模型

懵诬哇 2025-5-29 15:34:00
在这篇http://www.cnblogs.com/wsky/archive/2010/12/11/1902969.html文章中对facebook的bigpipe作了较长篇幅的描述,现在我们来快速实现一个简单的bigpipe模型
先描述几个概念:
Pagelet:页面功能模块化的单位
BigPipeClient:解释并呈现Pagelet的客户端,可以是javascript或服务器端语言编写(如c#)
其实重点在于模块化以及模块化之后随之带来的好处,功能开发分工,页面级别的功能隔离、功能降级等都可以以模块为单位进行,从而简化web项目的模型,在这个基础上再构建一系列的开发框架来支撑这种开发模式即可
先看一下Pagelet的定义:
  1.    1:     <%foreach (var let in ViewData["pagelets"] as IEnumerable<BigPipe.Controllers.Pagelet>) /// <summary>
复制代码
  1.    2:      /// 模块
复制代码
  1.    3:      /// </summary>
复制代码
  1.    4:      public class Pagelet
复制代码
  1.    5:      {
复制代码
  1.    6:          /// <summary>
复制代码
  1.    7:          /// 模块标识
复制代码
  1.    3:            /// </summary>
复制代码
  1.    9:          public string Name { get; set; }
复制代码
  1.   10:          /// <summary>
复制代码
  1.   11:          /// 获取数据Url
复制代码
  1.   12:          /// </summary>
复制代码
  1.   13:          public string Url { get; set; }
复制代码
  1.   14:          /// <summary>
复制代码
  1.   15:          /// 目标容器
复制代码
  1.   16:          /// </summary>
复制代码
  1.   17:          public string Target { get; set; }
复制代码
  1.   18:          /// <summary>
复制代码
  1.   19:          /// HTML内容模板
复制代码
  1.   20:          /// </summary>
复制代码
  1.   21:          public string Template { get; set; }
复制代码
  1.   22:   
复制代码
  1.   23:          //and so on
复制代码
  1.   24:      }
复制代码
它包含了模块的Html模板内容,呈现的位置,获取数据的地址等,
1.png
这是一个pagelet实例,
然后编写一个页面,把这个页面需要的模块都输出到页面中:
  1.    1:     <%foreach (var let in ViewData["pagelets"] as IEnumerable<BigPipe.Controllers.Pagelet>)
复制代码
  1.    2:        { %>
复制代码
  1.    3:        
复制代码
BigPipe.js客户端的简易实现:
  1.    8:      <%}%>
复制代码
  1.    1:  (function ($) {
复制代码
  1.    2:   
复制代码
  1.    3:      //pagelets holder
复制代码
  1.    4:      var pagelets = {};
复制代码
  1.    5:      //bipipe
复制代码
  1.    6:      var bigpipe = function () { this._init.apply(this, arguments); };
复制代码
  1.    3:            _init: function () { },
复制代码
  1.    8:          _init: function () { },
复制代码
  1.    9:          add: function (let) {
复制代码
  1.   10:              pagelets[let['Name']] = let;
复制代码
  1.   11:              $.getJSON(let['Url'], function (data) {
复制代码
  1.   12:                  $('#' + let['Target']).append(let['Template']
复制代码
  1.   13:                      .replace('${name}', data['name'])
复制代码
  1.   14:                      .replace('${description}', data['description']));
复制代码
  1.   15:              });
复制代码
  1.   16:          }
复制代码
  1.   17:      }
复制代码
  1.   18:      window.client = new bigpipe();
复制代码
调用add的时候将pagelet加入客户端缓存,同时执行ajax请求数据并和html模板装配后呈现到指定位置
 
本文实现了两种呈现方式的pagelet:服务器端并行push和客户端pull方式
下图为使用bigpipe.js客户端进行ajax pull方式拉取数据并呈现pagelet
2.png

生成的pagelet内容如下:
3.png

 
下图为使用服务器端并行push多个pagelet内容并呈现,可以看到模块并不会按顺序加载,而是根据各自加载情况决定顺序,这样可以使得更快被处理完的模块立刻被呈现给用户:
4.png

原理就是对respone并行写入内容并立刻flush到客户端:
  1.   19:  })(jQuery);
复制代码
  1.    1:  context.Response.Write(string.Format(
复制代码
  1.    2:        { %>          , new JavaScriptSerializer().Serialize(let)));
复制代码
  1.    3:                  , new JavaScriptSerializer().Serialize(let)));
复制代码
 
本文只简易实现了pagelet的处理模型,描述一下思路,相信有思路之后针对各自的应用或项目情况编写一套合适模块化框架会有些益处的
在完成bigpipe加载后,pageCache以及记录回放等在浏览器中的优化技巧也是很有意思的,之后会尝试做一些实现
 
DEMO源码可以在这里下载:https://github.com/wsky/wsky.github.com/tree/master/bigpipe
DEMO中用到简单并行实现:http://www.cnblogs.com/wsky/archive/2009/12/23/1630548.html

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
您需要登录后才可以回帖 登录 | 立即注册