新浪xweibo代码架构分析与二次开发
框架概要与特色
xweib o使用MVC结构但VC层任务分离不明显 V层也完成部分C层的任务调用model层获取数据。
没有使用smarty模板类视图采用组件的形式可以任意拼装可以方便模板模块的复用可以通过后台修改显示细节。
core文件作为所有调用的入口使用的所有类函数适配器都是用core提供的函数和类进行调用便于集中控制。
对于sina open API调用有统一错误处理通过core层提供的数据交互组件集中对数据进行缓存过滤格式化等操作。
框架整体流程图
如图所示
1请求到达后首先进入预处理模块。
2->3->4预处理调用控制器层的类中的函数对xweibo全局数据进行设置(如session) 。
5->6->7应用初始化如模板皮肤应用安装位置 log位置格式 ip过滤等操作并调用model层设置用户的相关信息。
8->9路由层将用户请求交给相应的控制器(controller)进行处理。
10->11->12->13控制器调用open api获取相关数据。
14->15->16->17控制器缓存相关数据或者获取缓存的相关信息,或者站点数据库内定制的信息(结果显示数量等) 。
18控制器包含相关模板(view)进行显示。
19->20->21->22每个模板(view)包含自己需要显示的组件(子view) 每个组件调用model层获取自己所需的数据进行显示。
xweibo介绍各个子功能和模块
预处理模块
用户请求进入后可以指定加载预处理模块这些预处理模块就是执行一些controller中的成员函数 比如初始化站点信息,比如检验用户是否登录。加载预处理模块时可以指定哪些请求跳过预处理模块比如登录请求就不需要加载登录校验模块。
初始化操作
初始化操作初始化工作包括如下
1.应用配置 log日志格式获取程序的安装位置是否来自ajax请求等等。
2.访问控制检测过滤预定义的ip或请求路由(不允许直接请求的controller)
3.初始化模板皮肤
4.执行用户定义的预处理模块
路由层
路由层根据路由配置获取请求参数中相应的变量加载相应的php文件实例化相应的类并将请求其他参数传入如果访问地址无效执行function中的error404函数。
控制器
在项目controllers目录下的所有文件为xweibo的控制器代码。
XWEIBO中的控制器按功能分为两种一种是作为预加载模块在初始化过程中调用如account.mod.php.另一种是请求通过路由层进入控制器调用控制器的相关方法获取数据显示相关的模板VIEW 。
以微博广场为例http://apptest/xweibo_upload/index.php?m=pub
则会进入pub.mod.php 当没有指定action时候默认调用控制器的default_action() 获取数据并包含相应模板页。
微博广场上的随便看看标签urlhttp://apptest/xweibo_upload/index.php?m=pub. look
该url指定了pub控制器的其action为look.调用pub.mod.php的look方法获取数据包含相应模板。
控制器|钩子
对于控制器的每一个方法在执行前和执行后会执行钩子函数。即调用方法前执行"_before_方法名 "在调用方法后执行“_after_方法名( )” 函数。例如在pub.mod.php的look方法前会执行_before_look()方法执行完look方法后会再执行一个_after_look() 。提供钩子方法的好处是当需要时候可以对数据做预处理或者当函数执行完后根据已经获取的数据选择性的执行相应操作。
TIPS
可能存在疑问是那无论预处理或者调用后处理都可以在函数里执行或者加判断为什么用钩子
我的理解是有些方法会被其他方法或者其他类显式的调用如果在方法中增加额外代码会影响使用该方法的其他类使用钩子方法可以只在用户请求控制层的时候执行相应的钩子操作钩子调用可以在cfg.php中关闭。
模板(view)
在templates/default目录下的所有*. tpl.php文件都是模板文件。xweibo controller层与VIEW层在任务分配上界限不明显。这体现在VIEW层也可以调用model层的操作获取、组织、过滤相应数据,当然获取完数据后需要对获取的数据进行显示使用html+php混合的写法。
TIPS
XWEIBO没有使用模板引擎很大原因是因为它想让数据组织和显示更自由化从哪获取显示什么样子在一个模板文件里可以更改一个模板文件可以被其他文件包含实现相应模块。如果使用模板引擎在调用核心文件(core)提供的方法上不自由另外所有的模块在使用模板引擎情况下按照使用习惯应该再拆成两部分一部分组织数据(php文件) 一部分显示数据(tpl模板文件) 。并将所有组织数据部分放到controller部分。
MODEL层
XWEIBO的model层是modules下的所有文件提供数据库、缓存、 open api等数据获取或改写操作。
例如需要从open api获取数据需使用modules目录下xwb.com.php类中的getPublicTimeline方法。
$list = DR('xweibo/xwb.getPublicTimeline' , ' ' , null, false) ;
即调用modeles目录下xweibo目录下的xwb.com.php文件中的getPublicTimeline方法。 model层可以调用class或者function或者adapter目录下的相应函数或类来实现数据处理。
核心文件(core)
核心文件其实提供调用其它公用类(class文件夹下) 、公用函数(funciton文件夹下) 、 model层函数module文件夹下的统一入口。提供所有实例的管理单实例缓存操作适配器选择公用数据类(USER)的维护.
常用方法如下
DS函数
调用形式 DS('xweibo/xwb.getUnread' ) ;
调用module下xwb类的getUnread方法 出错后直接处理跳转到相应错误处理页面。
DR函数
调用形式 $res = DR('xweibo/xwb.getUnread' ) ;
调用后如果出错后不直接处理而是将错误返回 由用户做相应处理。DR与DS其实就是相当于提供了调用model层的统一入口。 出错或者不出错的判定由xweibo做了统一判定处理。实际上这两个函数是两个宏为了使用方便的缩写实际上是调用了CORE文件中的数据交互组件管理类dsMgr来进行处理。TIPS:使用数据交互组件dsMgr好处
正常使用情况下我直接建立类的实例进行相应的数据请求就可以了如果只是出错处理是否返回问题无法再传一个参数罢了。数据交互组件好处除了错误统一处理外还封装了对返回数据处理。因为model层的请求有的是数据库有些是API所以对于返回数据可能需要做额外的处理 比如说缓存 比如说数据过滤数据格式化。如果没有这个组件那么API的相关操作数据库相关操作每一个都要对数据单独做缓存、过滤、格式化等处理。
C O RE文件|适配器
使用方法def ine('HTTP_ADAPTER' , 'curl' ) ;
$this->http = APP: :ADP('http' ) ;
ADP适配器实际将一些可选的操作类或者平台相关操作类封装了。
以http适配器为例 adapter/http/目录下的所有类都提供了相同的方法无论适配器选择了哪个类调用时候都返回相同的结果只不过curl_http库使用了curl库 fsockopen_http使用的是自己拼装的http数据。 sae_http应该是在sae平台上使用的http库。
TIPS
这里xweibo有一个我感觉可以调整地方这里适配器都是使用常量定义的如果我在项目中需要根据某些判定条件动态改变适配器只能在宏定义那做判定不方便或者跳过ADP函数自己建一个操作类实例。感觉xweibo开发人员可以调整下。
C O RE文件|用户类
XWEIBO为用户的数据和操作单独抽象了一层提供了公用的方法具体的操作代理给具体的类进行操作。
举例说明
需要获取用户UID的信息只需要调用 USER:uid() 具体如何获取用户uid是从session还是cookie还是数据库memcache这些可以代理类实现。xweibo提供了一个用session保存用户信息的clientUser类在class目录下。在我们项目中我们将代理类改成使用cookie加密保存用户信息只需要重新建立一个cookieClientUser类提供相同的方法在USER类初始化时候换成实例化cookieClientUser类即可。
Tips
其实在功能上来看这个用户类实际上应该算是model提供和封装了针对用户的数据交互不清楚xweibo设计人员为什么将它提到core这层来。 core的文件已经很大(2200行) 如果修改出现问题会对所有访问照成影响。另外其实建议对与完成实际用户操作的类也提供一层适配器操作这样开发用户可以定制自己的用户操作类无论是session或者cookie或者session服务器等等。
C O RE文件|AP P类
是xweibo的核心类提供了建立和调用其他类入口function N($oRoute) //根据路由获取一个类实例function M($mRoute) //执行一个模块(包括对应钩子函数)function &O($oRoute) //获取一个单例调用后缓存对象function F($fRoute) //根据路由调用funciton目录下的函数
/*获取还原后的 $_GET $_POST , $_FILES $_COOKIE $_REQUEST $_SERVER$_ENV中的变量例如V('G:TEST/BB' ) ;表示获取 $_GET['TEST' ] ['BB' ] 该
函数内使用静态变量会缓存已经获取的变量*/function V($vRoute, $def_v=NULL, $setVar=false)
CORE|TPL类/IO类/cache类
提供了模板类但模板只是将assign的变量转化为global然后在display前将全局的变量转化成局部变量在模板所对应的文件直接使用。extract($GLOBALS[V_GLOBAL_NAME] ['TPL' ]) ;
提供了统一的IO操作使用IO类来执行文件读写增删等操作。
提供了统一缓存操作通过适配器可以选择文件缓存 memcache缓存 目前xweibo主要缓存了用户一些配置信息比如皮肤一类的信息调用API显示表示需要缓存数据可以设置缓存时间。
小结
感觉xweibo提供了一个完整的微博内容展示框架 内容以组件的形式表现可以层层拼装包含模板在可复用性极强。提供了简单的管理后台允许用户对显示规则进行定制也方便二次开发。提供了完整的缓存策略提供了对API封装和访问出错处理控制。提供了微博相关的各种实用函数和类 比如内容的format(文本中出现@等替换成链接) 加解密处理(des) http的curl封装等等。但就我个人使用感觉上还有细节上需要改进的地方比如说现在的类和函数这块应该也提供适配器拿openapi举例现在的第三方应用的api调用形式根据需要各有不同(如认证) 用户对于各种类的使用也各有不同比如说format。虽然说放弃了smarty可以减少文件的层次(数据组织和显示) 但前端开发人员可能更习惯于使用模板类建议将templates下的所有模板再拆分两层一层放到controller一层完全是smarty模板(想办法也能继续使用plugin函数) 。细节上把一些能公用的小功能再抽出来成一个util类 比如oauth文件里的对http_query头的封装就可以单拿出一个静态函数。另外感觉xweibo为了兼容sae也加入不少代码虽然通过配置一点不影响使用但查看代码时候会遇到很多平台相关判断开发者体验上怪点。感觉上如果人手充足针对sae平台单独设计优化对外提供一个纯净版本的呵呵 以上都是个人意见对于xweibo自己也有很多地方要学习很好很强大第三方开发的好框架。
我的新浪微博 http://weibo.com/319799989
DiyVM是一家比较低调的国人主机商,成立于2009年,提供VPS主机和独立服务器租用等产品,其中VPS基于XEN(HVM)架构,数据中心包括香港沙田、美国洛杉矶和日本大阪等,CN2或者直连线路,支持异地备份与自定义镜像,可提供内网IP。本月商家最高提供5折优惠码,优惠后香港沙田CN2线路VPS最低2GB内存套餐每月仅50元起。香港(CN2)VPSCPU:2cores内存:2GB硬盘:50GB/R...
ReliableSite怎么样?ReliableSite好不好。ReliableSite是一家成立于2006年的老牌美国商家,主要经营美国独立服务器租赁,数据中心位于:洛杉矶、迈阿密、纽约,带宽1Gbps起步,花19美元/月即可升级到10Gbps带宽,月流量150T足够各种业务场景使用,且免费提供20Gbps DDoS防护。当前商家有几款大硬盘美国独服,地点位于美国洛杉矶或纽约机房,机器配置很具有...
春节期间我们很多朋友都在忙着吃好喝好,当然有时候也会偶然的上网看看。对于我们站长用户来说,基本上需要等到初八之后才会开工,现在有空就看看是否有商家的促销。这里看到来自HMBcloud半月湾服务商有提供两款春节机房方案的VPS主机88折促销活动,分别是来自洛杉矶CN2 GIA和日本CN2的方案。八八折优惠码:CNY-GIA第一、洛杉矶CN2 GIA美国原生IP地址、72小时退款保障、三网回程CN2 ...