发明内容
本说明书实施例提供一种消息处理方法、装置和设备,用于解决如下问题:以提供一种更便利的消息处理方案。
基于此,本说明书实施例提供一种消息处理方法,包括:
消息代理节点接收消息生产方所发送的消息;
将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量;
生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号;
将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
同时,本说明书的实施例还提供一种消息处理装置,包括:
接收模块,消息代理节点接收消息生产方所发送的消息;
第一写入模块,将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量;
生成模块,生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号;
第二写入模块,将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
对应的,本说明书实施例还提供一种消息处理设备,包括:
存储器,存储有消息处理程序;
处理器,调用存储器中的消息处理程序,并执行:
消息代理节点接收消息生产方所发送的消息;
将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量;
生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号;
将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
对应的,本说明书的实施例还提供一种非易失性计算机存储介质,存储有计算机可执行指令,所述计算机可执行指令设置为:
消息代理节点接收消息生产方所发送的消息;
将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量;
生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号;
将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
本说明书实施例采用的上述至少一个技术方案能够达到以下有益效果:
通过将消息代理节点和存储节点进行分离,对消息进行分布式的写入以及集中式的存储,对根据数据类型对各计算节点给予分配相应的权限。实现了对于所有消息代理节点可见的消息存储。在这种方式下,消息传输过程中所产生的数据可以在各消息代理节点之间自动分配,做自动迁移。消息代理节做到完全无状态,减任何一台节点不影响正常服务,能自动恢复;而增加一台消息代理节点则可以横向扩展能力,并自动分配业务流量,可以实现动态扩容,适应性更广。
具体实施方式
为使本申请的目的、技术方案和优点更加清楚,下面将结合本申请具体实施例及相应的附图对本申请技术方案进行清楚、完整地描述。显然,所描述的实施例仅是本申请一部分实施例,而不是全部的实施例。基于本说明书中的实施例,本领域普通技术人员在没有做出创造性劳动前提下所获得的所有其他实施例,都属于本申请保护的范围。
随着基于本地文件系统的业务的发展,本地存储模式其所需要的机器规模以及存储成本,成倍上涨。如果还是采取传统的分散式存储管理方式,不但带来高昂的管理分散式运维的成本,而且还会增加存储成本。
基于此,本说明书实施例提供一种消息代理节点和存储节点分离的分布式消息处理方案,在这种方案下,对消息进行分布式的写入以及集中式的存储,对根据数据类型对各计算节点给予分配相应的权限。实现了对于所有消息代理节点可见的消息存储。图1为本说明书实施例所涉及的系统架构示意图,在该架构中,所述的消费端可以是各种业务系统、服务端、用户终端等等,其中的消息代理节点可以是集群式或者分布式的多个节点,其执行的功能基本相同。在该架构中,消息代理节点所代理的消息不再存储于本地,均发送至存储节点进行存储。
下面将基于如图1所示的架构,详细说明本说明书的实施例提供的消息处理过程,如图2所示,图2为本说明书实施例所提供的消息处理过程的流程示意图,包括:
S201,消息代理节点接收消息生产方所发送的消息。
S203,将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量。
如上所述,消息存储文件是位于存储节点中,存储节点对于消息代理节点开放读取、写入、删除等等权限。对于存储节点而言,其中可以具有多个消息存储文件,每个消息存储文件具有一定的容量,可以容纳多个消息。一旦消息存储文件到达容纳上限,即可以调用下一个消息存储文件进行写入。
对于存储节点而言,写入的方式可以是并发式的。即,可以同时接收多个消息代理节点对于多个消息存储文件的写入。在写入的过程中,一般而言,按照接收到的消息的顺序进行写入。从而,对于一个消息存储文件而言,每个消息在其中的顺序是确定的,即,逻辑偏移量logicOffset为消息在该消息存储文件中的写入顺序(例如,1、2、3、4等等)。同时,每个消息在该消息存储文件中的写入点(或者也可以称为存储点)也不同,即需要记录消息在消息存储文件的物理偏移量phyOffset(相对于该消息存储文件的起始位置的偏移量,例如,497、1101、2605、3220等等)。
S205,生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号。
所述的索引index文件实际上就是消息代理节点所维护的一个自增的序列文件(即,每个索引文件的名称是一个序列号,随着消息的写入,该序列号不断增加),每往消息存储文件中写入一个消息,就会在该文件中自动生成一个索引数据块,索引数据块包含了当前消息的逻辑偏移量logicOffset和指向log文件的地址,在拉取消息时,可以根据消费者提供的序列号,定位索引文件,进而定位消息存储文件,再根据逻辑偏移量和物理偏移量访问消息存储文件中的消息内容。
S207,将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
在消费队列中,为便于查找,消息存储文件的序列号和所述索引文件将具有相同的序列号,一般而言,即将该序列号作为消息存储文件再消费队列中的名称,例如,名称为0000345678.log消息存储文件,对应的,在消费队列中即存在0000345678.index的索引文件。通过如上方式,对于其他任意的消息代理节点,即可以根据索引文件方便的查询得到存储节点中消费队列里的消息存储文件,进而得到其中的消息。
通过将消息代理节点和存储节点进行分离,对消息进行分布式的写入以及集中式的存储,对根据数据类型对各计算节点给予分配相应的权限。实现了对于所有消息代理节点可见的消息存储,避免了消息代理节点对消息进行本地存储时所带来的序列失效问题。
作为一种具体的实施方式,在对存储节点进行写入操作时,对于S203中的将所述消息写入存储节点中的消息存储文件,可以执行同步写入操作,将所述消息同步写入存储节点中的消息存储文件,换言之,在接收到客户端消息的同时,将消息写入写入存储节点中的消息存储文件,在写入完毕之后,再发送消息接收成功的信息至客户端。而对于将所述索引文件和消息存储文件写入消费队列时,可以采用异步写入的方式,换言之,在生成了索引文件和将消息写入消息存储文件之时,不必同时就写入消费队列ConsumeQueue,而是再次创建其他进程,例如在内存中创建DispatchRequest进程,调用相关服务将数据写入消费队列,将所述索引文件和消息存储文件异步写入消费队列。如图3所示,图3为本说明书实施例所提供的消息代理节点对存储节点进行写入操作的示意图。
在实际应用中,对于步骤S207中,将所述索引文件和消息存储文件写入消费队列,包括:确定所述消息存储文件所写入的进度和时间戳;根据所述进度和时间戳生成检测文件checkpoint;将所述索引文件、消息存储文件和检测文件写入消费队列。具体而言,这里检测文件checkpoint的作用是确定在消费队列consumeQueue中写入的消息存储文件的offset进度(即,写到了第几个消息以及什么位置等等)和时间戳,用来当消息代理节点宕机并重启后进行进度恢复和检查。
对于已经写入的消费队列的消息存储文件和索引文件而言,消息代理节点还可以对其执行如下操作:删除所述存储节点已写入消费队列的消息存储文件。这里删除的文件并不包括消费队列中的消息存储文件,而是指最先用于写入消息的消息存储文件,通过删除重复的文件,可以尽可能的节省存储节点的空间,避免浪费。
在对于消费队列执行了写入或者读取的操作之后,此时,消息代理节点还可以执行如下操作:针对任一消费队列,确定对其最近一次写入或者读取的时间与当前时间的时间间隔;若所述时间间隔符合预设条件,保存所述消费队列至本地缓存。
此处的预设条件一般即为时间间隔小于一定值(例如,30分钟),换言之,对于近期写入或读取过的数据,会缓存在系统中,下次再次读取时,直接从内存读取,节约网络开销。
此外,对于客户端所发送而来的消息,一般而言根据业务内容而言,是具有不同的主题的,这里的主题可以是业务名称、应用名称、业务类型等等。例如,其可以是诸如“支付”、“保险”、“信贷”等等主题。因此,还可以采取如下方式生成消费队列:将所述消费队列根据主题进行分区,生成具有不同主题的多个消费队列分区;从而将所述索引文件和消息存储文件写入消费队列,包括:根据客户端所发送的消息中所包含的主题,将所述将所述索引文件和消息存储文件写入具有相同主题的消费队列分区。通过将不同的消息根据主题进行分区,从而具有相同主题的业务消息将会被划分至同一消费队列分区,有利于后续的消息拉取处理。
在将消息存储至消费队列之后,在读取时可以根据如下方式进行:接收消费端所发送的的消费消息请求,其中,所述消费消息请求携带序列号;根据所述消费请求所携带的序列号和索引文件的序列号,确定所对应的索引文件;根据所述索引文件的序列号确定其对应的消息存储文件;根据所述索引文件中所包含的逻辑偏移量和物理偏移量,确定消息在消息存储文件中的地址;根据所确定的地址拉取消息,并发送至消费端。
具体而言,消费端需要消费消息时,消费者会带上一定的序列号(此处的序列号不一定和索引文件的序列号相同),然后根据序列号定位到index文件的索引块,然后再读出指向消息存储文件的地址,进而再访问消息内容。如图4所示,图4为本说明书实施例所提供的消息代理节点拉取消息的示意图。需要说明的是,在拉取消息的时候,可以不是之前对该消息进行写入操作的消息代理节点,但是基于各节点的功能基本相同,在执行读取的过程中其可以视为同一执行主体。
为使读取消息的方式更为浅显明白,下边给出一个具体的例子,如图5所示,图5为本说明书实施例所提供的具体的读取消息的示意图。在该示例中,index本身具有一定的序号“00123456”。文件中存储了大量的消息辑偏移量和物理偏移量;消息存储文件具有和其相同的序号,其中存储了大量的消息。现在消费端表示需要拉取序号=123458的消息。此时,即可以根据用户所发送的序号“123458”执行二分查找,得到对应的序号为“123456”的index文件,并找到对应的消息存储文件“123456.log”文件,最后,根据index文件中的逻辑偏移量和物理偏移量查询得到123458号消息,从消息存储文件起始位置偏移量500的位置开始读取消息
此外,在消息代理节点对消息进行处理之前,还可以预先确定消息代理节点的相应权限,这里的权限可以本地确定,也可以有专门的服务端进行分配。棘突而言,包括如下方式:确定本地消息代理节点对存储节点中数据的操作权限,其中,所述数据包括消费队列或消息存储文件中的至少一种;根据所述分配得到的操作权限对所述数据执行对应操作。操作权限可以包括是对读取权限、写入权限、或者删除权限的限制,例如,对于消费队列,定义为共享数据,确定为全部的消息代理节点均有读写权限;而对于还未写入消费队列的消息存储文件或者其他一些在本地处理时所产生的相关数据或者本地配置信息、操作记录等等,则定义为私有数据,只限于本地可以写入数据,其他节点则只有只读权限。
换言之,对于确定本地消息代理节点对存储节点中数据的操作权限,包括:针对任一消费队列,分配至本地处理时,确定对其具有读取权限和写入权限;针对任一消息存储文件,分配至本地处理时,判断本地消息代理节点是否指定的消息代理节点,若是,具有读取权限和写入权限,若否,具有读取权限,没有写入权限。
通过上述方式,对于共享数据(消费队列),以Queue的粒度在broker之间分配,同一时刻一个Queue只有一个broker可以对其进行处理,提供Queue的读写服务。而对于私有数据(例如还未写入消费队列的消息存储文件),只被当前固定的消息代理节点有写权限,当broker故障时,可由其他broker托管提供只读服务,当broker恢复时,可以重新获取owner权限,提供写服务。从而实现数据可以在计算节点之间自动分配,自动迁移。减任何一台节点不影响正常服务,能自动恢复,而加一台节点可以横向扩展能力,并自动分配业务流量。
本说明书实施例所提供的方案中,对于各消息代理节点和消费队列的数据逻辑结构而言,如6所示,图6为本说明书实施例所提供的关于消息代理节点和消费队列的逻辑结构示意图。图中,在消费代理节点一端的commite.log中的数据一般即可认为是前文所提及的私有数据,其中的timer与时限相关,其中的config与本地节点的配置相关等等。通过将消息代理节点和存储节点进行分离,对消息进行分布式的写入以及集中式的存储,对根据数据类型对各计算节点给予分配相应的权限。实现了对于所有消息代理节点可见的消息存储。在这种方式下,消息传输过程中所产生的数据可以在各消息代理节点之间自动分配,做自动迁移。消息代理节做到完全无状态,减任何一台节点不影响正常服务,能自动恢复;而增加一台消息代理节点则可以横向扩展能力,并自动分配业务流量,可以实现动态扩容,适应性更广。
基于同样的思路,本发明还提供一种消息处理装置,如图7所示,图7为本说明书实施例所提供的消息处理装置的结构示意图,包括:
接收模块701,消息代理节点接收消息生产方所发送的消息;
第一写入模块703,将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量;
生成模块705,生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号;
第二写入模块707,将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
进一步地,所述第一写入模块703,将所述消息同步写入存储节点中的消息存储文件;所述第二写入模块707,将所述索引文件和消息存储文件写入消费队列,包括:将所述索引文件和消息存储文件异步写入消费队列。
进一步地,所述第二写入模块707,确定所述消息存储文件所写入的进度和时间戳;根据所述进度和时间戳生成检测文件;将所述索引文件、消息存储文件和检测文件写入消费队列。
进一步地,所述装置还包括删除模块709:删除所述存储节点已写入消费队列的消息存储文件。
进一步地,所述装置还包括存储模块711,针对任一消费队列,确定对其最近一次写入或者读取的时间与当前时间的时间间隔;若所述时间间隔符合预设条件,保存所述消费队列至本地缓存。
进一步地,所述装置还包括分区模块713,将所述消费队列根据主题进行分区,生成具有不同主题的多个消费队列分区;所述第二写入模块707,根据客户端所发送的消息中所包含的主题,将所述将所述索引文件和消息存储文件写入具有相同主题的消费队列分区。
进一步地,所述接收模块701还用于,接收消费端所发送的的消费消息请求,其中,所述消费消息请求携带序列号;还包括确定模块715,根据所述消费请求所携带的序列号和索引文件的序列号,确定所对应的索引文件;根据所述索引文件的序列号确定其对应的消息存储文件;根据所述索引文件中所包含的逻辑偏移量和物理偏移量,确定消息在消息存储文件中的地址;还包括,消息拉取和发送模块717,根据所确定的地址拉取消息,并发送至消费端。
进一步地,所述装置还包括确权模块719,确定本地消息代理节点对存储节点中数据的操作权限,其中,所述数据包括消费队列或消息存储文件中的至少一种;根据所述分配得到的操作权限对所述数据执行对应操作。
进一步地,所述确权模块719,针对任一消费队列,分配至本地处理时,确定对其具有读取权限和写入权限;针对任一消息存储文件,分配至本地处理时,判断本地消息代理节点是否指定的消息代理节点,若是,具有读取权限和写入权限,若否,具有读取权限,没有写入权限。
对应的,本申请实施例还提供一种消息处理设备,包括:
存储器,存储有消息处理程序;
处理器,调用存储器中的消息处理程序,并执行:
消息代理节点接收消息生产方所发送的消息;
将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量;
生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号;
将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
基于同样的发明思路,本申请实施例还提供了对应的一种非易失性计算机存储介质,存储有计算机可执行指令,所述计算机可执行指令设置为:
消息代理节点接收消息生产方所发送的消息;
将所述消息写入存储节点中的消息存储文件,确定消息在所述消息存储文件中的逻辑偏移量和物理偏移量;
生成包含所述逻辑偏移量和物理偏移量的索引文件,其中,所述索引文件具有序列号;
将所述索引文件和消息存储文件写入消费队列,以便根据索引文件从所述消费队列中拉取消息至消费端,其中,所述消息存储文件在消费队列中与所述索引文件具有相同序列号。
本说明书中的各个实施例均采用递进的方式描述,各个实施例之间相同相似的部分互相参见即可,每个实施例重点说明的都是与其他实施例的不同之处。尤其,对于装置、设备和介质类实施例而言,由于其基本相似于方法实施例,所以描述的比较简单,相关之处参见方法实施例的部分说明即可,这里就不再一一赘述。
上述对本说明书特定实施例进行了描述。其它实施例在所附权利要求书的范围内。在一些情况下,在权利要求书中记载的动作或步骤或模块可以按照不同于实施例中的顺序来执行并且仍然可以实现期望的结果。另外,在附图中描绘的过程不一定要求示出的特定顺序或者连续顺序才能实现期望的结果。在某些实施方式中,多任务处理和并行处理也是可以的或者可能是有利的。
在20世纪90年代,对于一个技术的改进可以很明显地区分是硬件上的改进(例如,对二极管、晶体管、开关等电路结构的改进)还是软件上的改进(对于方法流程的改进)。然而,随着技术的发展,当今的很多方法流程的改进已经可以视为硬件电路结构的直接改进。设计人员几乎都通过将改进的方法流程编程到硬件电路中来得到相应的硬件电路结构。因此,不能说一个方法流程的改进就不能用硬件实体模块来实现。例如,可编程逻辑器件(Programmable Logic Device,PLD)(例如现场可编程门阵列(Field Programmable GateArray,FPGA))就是这样一种集成电路,其逻辑功能由用户对器件编程来确定。由设计人员自行编程来把一个数字系统“集成”在一片PLD上,而不需要请芯片制造厂商来设计和制作专用的集成电路芯片。而且,如今,取代手工地制作集成电路芯片,这种编程也多半改用“逻辑编译器(logic compiler)”软件来实现,它与程序开发撰写时所用的软件编译器相类似,而要编译之前的原始代码也得用特定的编程语言来撰写,此称之为硬件描述语言(Hardware Description Language,HDL),而HDL也并非仅有一种,而是有许多种,如ABEL(Advanced Boolean Expression Language)、AHDL(Altera Hardware DescriptionLanguage)、Confluence、CUPL(Cornell University Programming Language)、HDCal、JHDL(Java Hardware Description Language)、Lava、Lola、MyHDL、PALASM、RHDL(RubyHardware Description Language)等,目前最普遍使用的是VHDL(Very-High-SpeedIntegrated Circuit Hardware Description Language)与Verilog。本领域技术人员也应该清楚,只需要将方法流程用上述几种硬件描述语言稍作逻辑编程并编程到集成电路中,就可以很容易得到实现该逻辑方法流程的硬件电路。
控制器可以按任何适当的方式实现,例如,控制器可以采取例如微处理器或处理器以及存储可由该(微)处理器执行的计算机可读程序代码(例如软件或固件)的计算机可读介质、逻辑门、开关、专用集成电路(Application Specific Integrated Circuit,ASIC)、可编程逻辑控制器和嵌入微控制器的形式,控制器的例子包括但不限于以下微控制器:ARC 625D、Atmel AT91SAM、Microchip PIC18F26K20以及Silicone Labs C8051F320,存储器控制器还可以被实现为存储器的控制逻辑的一部分。本领域技术人员也知道,除了以纯计算机可读程序代码方式实现控制器以外,完全可以通过将方法步骤进行逻辑编程来使得控制器以逻辑门、开关、专用集成电路、可编程逻辑控制器和嵌入微控制器等的形式来实现相同功能。因此这种控制器可以被认为是一种硬件部件,而对其内包括的用于实现各种功能的装置也可以视为硬件部件内的结构。或者甚至,可以将用于实现各种功能的装置视为既可以是实现方法的软件模块又可以是硬件部件内的结构。
上述实施例阐明的系统、装置、模块或单元,具体可以由计算机芯片或实体实现,或者由具有某种功能的产品来实现。一种典型的实现设备为计算机。具体的,计算机例如可以为个人计算机、膝上型计算机、蜂窝电话、相机电话、智能电话、个人数字助理、媒体播放器、导航设备、电子邮件设备、游戏控制台、平板计算机、可穿戴设备或者这些设备中的任何设备的组合。
为了描述的方便,描述以上装置时以功能分为各种单元分别描述。当然,在实施本说明书的实施例时可以把各单元的功能在同一个或多个软件和/或硬件中实现。
本领域内的技术人员应明白,本发明的实施例可提供为方法、系统、或计算机程序产品。因此,本发明可采用完全硬件实施例、完全软件实施例、或结合软件和硬件方面的实施例的形式。而且,本发明可采用在一个或多个其中包含有计算机可用程序代码的计算机可用存储介质(包括但不限于磁盘存储器、CD-ROM、光学存储器等)上实施的计算机程序产品的形式。
本发明是参照根据本发明实施例的方法、设备(系统)、和计算机程序产品的流程图和/或方框图来描述的。应理解可由计算机程序指令实现流程图和/或方框图中的每一流程和/或方框、以及流程图和/或方框图中的流程和/或方框的结合。可提供这些计算机程序指令到通用计算机、专用计算机、嵌入式处理机或其他可编程数据处理设备的处理器以产生一个机器,使得通过计算机或其他可编程数据处理设备的处理器执行的指令产生用于实现在流程图一个流程或多个流程和/或方框图一个方框或多个方框中指定的功能的装置。
这些计算机程序指令也可存储在能引导计算机或其他可编程数据处理设备以特定方式工作的计算机可读存储器中,使得存储在该计算机可读存储器中的指令产生包括指令装置的制造品,该指令装置实现在流程图一个流程或多个流程和/或方框图一个方框或多个方框中指定的功能。
这些计算机程序指令也可装载到计算机或其他可编程数据处理设备上,使得在计算机或其他可编程设备上执行一系列操作步骤以产生计算机实现的处理,从而在计算机或其他可编程设备上执行的指令提供用于实现在流程图一个流程或多个流程和/或方框图一个方框或多个方框中指定的功能的步骤。
在一个典型的配置中,计算设备包括一个或多个处理器(CPU)、输入/输出接口、网络接口和内存。
内存可能包括计算机可读介质中的非永久性存储器,随机存取存储器(RAM)和/或非易失性内存等形式,如只读存储器(ROM)或闪存(flash RAM)。内存是计算机可读介质的示例。
计算机可读介质包括永久性和非永久性、可移动和非可移动媒体可以由任何方法或技术来实现信息存储。信息可以是计算机可读指令、数据结构、程序的模块或其他数据。计算机的存储介质的例子包括,但不限于相变内存(PRAM)、静态随机存取存储器(SRAM)、动态随机存取存储器(DRAM)、其他类型的随机存取存储器(RAM)、只读存储器(ROM)、电可擦除可编程只读存储器(EEPROM)、快闪记忆体或其他内存技术、只读光盘只读存储器(CD-ROM)、数字多功能光盘(DVD)或其他光学存储、磁盒式磁带,磁带磁磁盘存储或其他磁性存储设备或任何其他非传输介质,可用于存储可以被计算设备访问的信息。按照本文中的界定,计算机可读介质不包括暂存电脑可读媒体(transitory media),如调制的数据信编号和载波。
还需要说明的是,术语“包括”、“包含”或者其任何其他变体意在涵盖非排他性的包含,从而使得包括一系列要素的过程、方法、商品或者设备不仅包括那些要素,而且还包括没有明确列出的其他要素,或者是还包括为这种过程、方法、商品或者设备所固有的要素。在没有更多限制的情况下,由语句“包括一个……”限定的要素,并不排除在包括所述要素的过程、方法、商品或者设备中还存在另外的相同要素。
本领域技术人员应明白,本说明书中一个或多个的实施例可提供为方法、系统或计算机程序产品。因此,本说明书的实施例可采用完全硬件实施例、完全软件实施例或结合软件和硬件方面的形式。而且,本说明书的实施例可采用在一个或多个其中包含有计算机可用程序代码的计算机可用存储介质(包括但不限于磁盘存储器、CD-ROM、光学存储器等)上实施的计算机程序产品的形式。
本说明书的实施例可以在由计算机执行的计算机可执行指令的一般上下文中描述,例如程序模块。一般地,程序模块包括执行特定事务或实现特定抽象数据类型的例程、程序、对象、组件、数据结构等等。也可以在分布式计算环境中实践本说明书的实施例,在这些分布式计算环境中,由通过通信网络而被连接的远程处理设备来执行事务。在分布式计算环境中,程序模块可以位于包括存储设备在内的本地和远程计算机存储介质中。