设计模式 —— 代理模式

一、简介

模式意图 :为其他对象提供一种代理以控制对这个对象的访问

代理模式是对象的结构模式,代理模式给某一个对象提供一个代理对象,并由此代理对象控制对原代理对象的引用。代理模式不应该让用户感觉到代理的存在,所以代理对象和原对象的对外的调用接口是一致的。

二、代理

CD 音乐商城,购买商品时链接本地mysql,由于架构调整需要链接阿里云上的数据库。可以用以下操作实现。当然现实中我们只需修改配置文件即可。

远程mysql代理

client调用

与委托模式区别

在C#语言层面上,是有将观察者模式用“事件委托”的方式来处理。
“委托”是一种应用方法的类型。
代理是一种用来解决问题的方法。
其实2者没有可比性。
从实现功能上讲,
代理是提供一种“一个类对另外一个类的控制权。”是类与类之间关系。
委托提供了“一种方法的执行会同时执行加载在上面的方法”。是方法与方法之间的关系。
委托可以代替代理,但是代理不能代替委托。
委托可以动态加载方法,代理不能实现。
委托对象所加载的方法不一定要属于同一个类。但是代理的类必须属于同一个类。

jQuery.Deferred对象

一、什么是deferred对象?

开发网站的过程中,我们经常遇到某些耗时很长的javascript操作。其中,既有异步的操作(比如ajax读取服务器数据),也有同步的操作(比如遍历一个大型数组),它们都不是立即能得到结果的。
通常的做法是,为它们指定回调函数(callback)。即事先规定,一旦它们运行结束,应该调用哪些函数。
但是,在回调函数方面,jQuery的功能非常弱。为了改变这一点,jQuery开发团队就设计了deferred对象。

1、生成Deferred对象

2、deferred对象状态 以及 改变状态

  • pending: 表示还没有完成
  • resolved: 表示操作成功
  • rejected: 表示操作失败

可以通过state方法来获取状态

3、绑定回调方法

resolve方法对应着done方法,reject方法对于着fail方法。

4、then方法

then方法的作用也是设置回调方法。他可以接受三个参数。按顺序为 done, fail, progress 这3个方法。
自jQuery1.8后,then方法有一个返回值就是promise对象。而done()返回的是原有的deferred对象。如果then()指定的回调函数有返回值,该返回值会作为参数,传入后面的回调函数。

5、pipe方法

大致的作用是将resolve、reject、notify传入的参数再做一下处理。

6、promise对象

简单说,promise对象就是不能改变状态的deferred对象,也就是deferred的只读版。或者更通俗地理解成,promise是一个对将要完成的任务的承诺,排除了其他人破坏这个承诺的可能性,只能等待承诺方给出结果。

你可以通过promise对象,为原始的deferred对象添加回调函数,查询它的状态,但是无法改变它的状态,也就是说promise对象不允许你调用resolve和reject方法。

7、$.when方法

$.when()接受多个deferred对象作为参数,当它们全部运行成功后,才调用resolved状态的回调函数,但只要其中有一个失败,就调用rejected状态的回调函数。它相当于将多个非同步操作,合并成一个。实质上,when方法为多个deferred对象,返回一个单一的promise对象。

参考链接:
http://www.ruanyifeng.com/blog/2011/08/a_detailed_explanation_of_jquery_deferred_object.html
http://www.web-tinker.com/article/20154.html
http://javascript.ruanyifeng.com/jquery/deferred.html

三级(多级)联动自动赋值

测试连接

参考:

http://api.jquery.com/jQuery.ajax/

http://api.jquery.com/jQuery.Deferred/

php-resque (三) —— 简单使用

队列有3个角色,运行php-resque只需要创建3个相应的脚本即可。

  • Job | 任务 : 一个Job就是一个需要在后台完成的任务,
  • Queue | 队列 : 也就是上文的消息队列,在Resque中,队列则是由Redis实现的。Resque还提供了一个简单的队列管理器,可以实现将Job插入/取出队列等功能。
  • Worker | 执行者 : 负责从队列中取出Job并执行,可以以守护进程的方式运行在后台。

进入php-resque目录下的demo目录下创建一下脚本

1、job

创建一个my_job.php。这个是一个任务。方法perform为必须存在,队列执行时候会自动调用这个方法。

 2、queue

创建my_queue.php。这是一个队列,主要作用是将任务插入队列(redis)。

3、work

创建my_work.php。这是个任务执行者。负责执行任务。

执行:

 

参考:

http://avnpc.com/pages/run-background-task-by-php-resque#toc9

https://github.com/chrisboulton/php-resque

php-resque (二) —— 安装

一、安装redis,php-resque是基于redis的消息队列。

参考:安装redis

二、安装Composer

 三、安装php-resque

 

 

参考链接:http://avnpc.com/pages/run-background-task-by-php-resque

php-resque (一) —— 简介

消息队列处理后台任务带来的问题

项目中经常会有后台运行任务的需求,比如发送邮件时,因为要连接邮件服务器,往往需要5-10秒甚至更长时间,如果能先给用户一个成功的提示信息,然后在后台慢慢处理发送邮件的操作,显然会有更好的用户体验。

为了实现类似的需求,Web项目中一般的实现方法是使用消息队列(Message Queue),比如MemcacheQRabbitMQ等等,都是很著名的产品。

消息队列说白了就是一个最简单的先进先出队列,队列的一个成员就是一段文本。正是因为消息队列实在太简单了,当拿着消息队列时,反而有点无从下手的感觉,因为这仅仅一个发送邮件的任务,就会引申出很多问题:

  1. 消息队列只能存储字符串类型的数据,如何将一个发送邮件这样的“任务”,转换为消息队列中的一个“消息”?
  2. 消息队列只负责数据的存放与进出,本身不能执行任何程序,那么我们要如何从消息队列中一个一个取出数据,再将这些数据转化回任务并执行。
  3. 我们无法预知消息队列何时会有数据产生,所以我们的任务执行程序还需要具备监控消息队列的能力,也就是一个常驻后台的守护进程。
  4. 一般的Web应用PHP都以cgi方式运行,无法常驻内存。我们知道php还有cli模式,那么守护进程是否能以php cli来实现,效率如何?
  5. 当守护进程运行时,Web应用能否与后台守护进程交互,实现开启/杀死进程的功能以及获得进程的运行状态?

Resque对后台任务的设计与角色划分

对以上这些问题,目前为止我能找到的最好答案,并不是来自php,而是来自Ruby的项目Resque,正是由于Resque清晰简单的解决了后台任务带来的一系列问题,Resque的设计也被Clone到Python、php、NodeJs等语言:比如Python下的pyres以及PHP下的php-resque等等,这里有各种语言版本的Resque实现,而在本篇日志里,我们当然要以PHP版本为例来说明如何用php-resque运行一个后台任务,可能一些细节方面会与Ruby版有出入,但是本文中以php版为准。

Resque是这样解决这些问题的:

后台任务的角色划分

其实从上面的问题已经可以看出,只靠一个消息队列是无法解决所有问题的,需要新的角色介入。在Resque中,一个后台任务被抽象为由三种角色共同完成:

  • Job | 任务 : 一个Job就是一个需要在后台完成的任务,比如本文举例的发送邮件,就可以抽象为一个Job。在Resque中一个Job就是一个Class。
  • Queue | 队列 : 也就是上文的消息队列,在Resque中,队列则是由Redis实现的。Resque还提供了一个简单的队列管理器,可以实现将Job插入/取出队列等功能。
  • Worker | 执行者 : 负责从队列中取出Job并执行,可以以守护进程的方式运行在后台。

那么基于这个划分,一个后台任务在Resque下的基本流程是这样的:

  1. 将一个后台任务编写为一个独立的Class,这个Class就是一个Job。
  2. 在需要使用后台程序的地方,系统将Job Class的名称以及所需参数放入队列。
  3. 以命令行方式开启一个Worker,并通过参数指定Worker所需要处理的队列。
  4. Worker作为守护进程运行,并且定时检查队列。
  5. 当队列中有Job时,Worker取出Job并运行,即实例化Job Class并执行Class中的方法。

至此就可以完整的运行完一个后台任务。

在Resque中,还有一个很重要的设计:一个Worker,可以处理一个队列,也可以处理很多个队列,并且可以通过增加Worker的进程/线程数来加快队列的执行速度。

转载:http://avnpc.com/pages/run-background-task-by-php-resque