这两天没事,看了一下Memcached和libevent的源码,做个小总结。

1、入门

1.1、概述
Libevent是一个用于开发可扩展性网络服务器的基于事件驱动(event-driven)模型的网络库。Libevent有几个显著的亮点:
(1)事件驱动(event-driven),高性能;
(2)轻量级,专注于网络,不如 ACE 那么臃肿庞大;
(3)源代码相当精炼、易读;
(4)跨平台,支持 Windows、Linux、*BSD和 Mac Os;
(5)支持多种 I/O多路复用技术, epoll、poll、dev/poll、select 和kqueue 等;
(6)支持 I/O,定时器和信号等事件;
(7)注册事件优先级;
 Libevent 已经被广泛的应用,作为底层的网络库;比如 memcached、 Vomi t、 Nylon、 Netchat等等。

1.2、一个简单示例

 1 int lasttime;
 2 
 3 static void
 4 timeout_cb(int fd, short event, void *arg)
 5 {
 6 struct timeval tv;
 7 struct event *timeout = arg;
 8 int newtime = time(NULL);
 9 
 //printf("%s: called at %d: %d\n", __func__, newtime,
 printf("%s: called at %d: %d\n", "timeout_cb", newtime,
         newtime - lasttime);
 lasttime = newtime;
 
 evutil_timerclear(&tv);
 tv.tv_sec = ;
 //重新注册event
 event_add(timeout, &tv);
 }
 
 int
 main (int argc, char **argv)
 {
 struct event timeout;
 struct timeval tv;
  
 /* Initalize the event library */
 //初始化event环境
 event_init();
 
 /* Initalize one event */
 //设置事件
 evtimer_set(&timeout, timeout_cb, &timeout);
 
 evutil_timerclear(&tv);
 tv.tv_sec = ;
 //注册事件
 event_add(&timeout, &tv);
 
 lasttime = time(NULL);
     
 //等待,分发,处理事件
 event_dispatch();
 
 return ();
 }

这是一个简单的基于libevent的定时器程序,运行结果:

用libevent编程非常简单,只需要调用event_init初始化环境,然后调用event_add注册相应的事件,接着调用event_dispatch等待并处理相应的事件即可。
调用event_add注册事件时,设置其回调函数。Libevent检测到事件发生时,便会调用事件对应的回调用函数,执行相关的业务逻辑。

1.3、源代码结构
Libevent
的源代码虽然都在一层文件夹下面,但是其代码分类还是相当清晰的,主要可分为头文件、内部使用的头文件、辅助功能函数、日志、libevent
框架、对系统 I/O 多路复用机制的封装、信号管理、定时事件管理、缓冲区管理、基本数据结构和基于
libevent的两个实用库等几个部分,有些部分可能就是一个源文件。
(1)头文件
主要就是 event.h:事件宏定义、接口函数声明,主要结构体 event 的声明;
(2)内部头文件
xxx-internal.h:内部数据结构和函数,对外不可见,以达到信息隐藏的目的;
(3)libevent框架
event.c:event 整体框架的代码实现;
(4)对系统 I/O多路复用机制的封装
epoll.c:对 epoll 的封装;
select.c:对 select 的封装;
devpoll.c:对 dev/poll 的封装;
kqueue.c:对kqueue 的封装;
(5)定时事件管理
min-heap.h:其实就是一个以时间作为 key的小根堆结构;
(6)信号管理
signal.c:对信号事件的处理;
(7)辅助功能函数
evutil.h  和 evutil.c:一些辅助功能函数,包括创建 socket pair和一些时间操作函数:加、减和比较等。
(8)日志
log.h和 log.c:log 日志函数
(9)缓冲区管理
evbuffer.c 和buffer.c:libevent 对缓冲区的封装;
(10)基本数据结构
compat\sys 下的两个源文件: queue.h是 libevent 基本数据结构的实现,包括链表,双向链表,队列等;_libevent_time.h:一些用于时间操作的结构体定义、函数和宏定义;
(11)实用网络库
     http 和evdns:是基于 libevent 实现的http 服务器和异步 dns 查询库;

2、核心对象
结构体event和event_base是libevent的两个核心数据结构,前者代表一个事件对象,后者代表整个事件处理框架。
2.1、event(事件)


代码

 1 //event.h
 2 struct event {
 3 TAILQ_ENTRY (event) ev_next;          //已注册事件链表
 4 TAILQ_ENTRY (event) ev_active_next;//就绪事件链表
 5 TAILQ_ENTRY (event) ev_signal_next; //signal链表
 6 unsigned int min_heap_idx;    /* for managing timeouts,事件在堆中的下标 */
 7 
 8 struct event_base *ev_base;
 9 
 int ev_fd;      //对于I/O事件,是绑定的文件描述符;对于signal事件,是绑定的信号
 short ev_events; //event关注的事件类型
 short ev_ncalls; //事件就绪执行时,调用 ev_callback 的次数
 short *ev_pncalls;    /* Allows deletes in callback */
 
 struct timeval ev_timeout;  //timout事件的超时值
 
 int ev_pri;   /* smaller numbers are higher priority,优先级 */
 
 void (*ev_callback)(int, short, void *arg); //回调函数
 void *ev_arg; //回调函数的参数
 
 int ev_res;        /* result passed to event callback */
 int ev_flags; //event的状态
 };
 

Libevent通过event对象将I/O事件、信号事件和定时器事件封装,从而统一处理,这也是libevent的精妙所有。
各个字段的具体含义:
(1) ev_events:event关注的事件类型,它可以是以下3种类型:
I/O事件: EV_WRITE和EV_READ
定时事件:EV_TIMEOUT
信号:    EV_SIGNAL
辅助选项:EV_PERSIST,表明是一个永久事件
libevent中的定义为:
#define EV_TIMEOUT    0x01
#define EV_READ    0x02
#define EV_WRITE    0x04
#define EV_SIGNAL    0x08
#define EV_PERSIST    0x10    /* Persistant event */
(2)ev_next,ev_active_next 和 ev_signal_next 都是双向链表节点指针;它们是 libevent 对不同事件类型和在不同的时期,对事件的管理时使用到的字段。
libevent 使用双向链表保存所有注册的 I/O和 Signal 事件,ev_next 就是该I/O事件在链表中的位置;此链表可以称为“已注册事件链表”;
同样 ev_signal_next 就是 signal 事件在 signal 事件链表中的位置;
ev_active_next:libevent 将所有的激活事件放入到链表 active list 中,然后遍历 active list 执
行调度,ev_active_next就指明了 event 在active list 中的位置;
(3)min_heap_idx 和 ev_timeout,如果是 timeout 事件,它们是 event 在小根堆中的索引和超时值,libevent 使用小根堆来管理定时事件。
(4)ev_base指向事件框架实例。
(5)ev_fd,对于 I/O事件,是绑定的文件描述符;对于 signal 事件,是事件对应的信号;
(6)eb_flags:libevent 用于标记 event信息的字段,表明事件当前的状态,可能的值有:
#define EVLIST_TIMEOUT   0x01 // event在time堆中
#define EVLIST_INSERTED 0x02 // event在已注册事件链表中
#define EVLIST_SIGNAL    0x04 // 未见使用
#define EVLIST_ACTIVE    0x08 // event在激活链表中
#define EVLIST_INTERNAL 0x10 // 内部使用标记
#define EVLIST_INIT      0x80 // event 已被初始化

2.2、event_base(事件处理框架)


代码

 1 //evenet_internal.h
 2 struct event_base {
 3 const struct eventop *evsel; //底层具体I/O demultiplex操作函数集
 4 void *evbase;
 5 int event_count;        /* counts number of total events,总的事件数量 */
 6 int event_count_active;    /* counts number of active events,就绪事件数量 */
 7 
 8 int event_gotterm;        /* Set to terminate loop */
 9 int event_break;        /* Set to terminate loop immediately */
 
 /* active event management */
 //就绪事件链表数组
 struct event_list **activequeues;
 int nactivequeues;//就绪事件队列个数
 
 /* signal handling info */
 struct evsignal_info sig; //用于管理信号
 
 struct event_list eventqueue; //注册事件队列
 struct timeval event_tv;
 
 struct min_heap timeheap; //管理定时器的小根堆
 struct timeval tv_cache; //记录时间缓存
 };

(1)evsel:libevent
支持Linux、Windows等多种平台,也支持epoll、poll、select、kqueue等多种I/O多路复用模型。如果把
event_init、event_add看成高层抽象的统一事件操作接口,则evsel为这些函数在底层具体的I/O
demultiplex的对应的操作函数集。eventop为函数指针的集合:


代码

 1 struct eventop {
 2 const char *name;
 3 void *(*init)(struct event_base *);
 4 int (*add)(void *, struct event *);
 5 int (*del)(void *, struct event *);
 6 int (*dispatch)(struct event_base *, void *, struct timeval *);
 7 void (*dealloc)(struct event_base *, void *);
 8 /* set if we need to reinitialize the event base */
 9 int need_reinit;
 };
 

在初始化函数event_base_new中,libevent将evsel指向全局数组eventops的具体元素:

代码

2.3、主要函数
2.3.1、event_int(初始化libevent实例)
struct event_base *
event_init(void)
初始化事件处理框架实例,内部调用event_base_new。

event_base_new的主要逻辑:


代码

 1 struct event_base *
 2 event_base_new(void)
 3 {
 4 
 5 //初始化小根堆
 6 min_heap_ctor(&base->timeheap);
 7 
 8 //初始化注册事件队列
 9 TAILQ_INIT(&base->eventqueue);
 
 for (i = ; eventops[i] && !base->evbase; i++) {
 //I/O demultiplex机制实例
 base->evsel = eventops[i];
 
 //初始化I/O demultiplex实例(参见win32_init)
 base->evbase = base->evsel->init(base);
 }
 
 //分配1个就绪事件队列
 event_base_priority_init(base, );
 
 }

2.3.2、event_add(注册事件)
//注册事件
int
event_add(struct event *ev, const struct timeval *tv)
该函数主要将事件ev加入到事件框架event_base的注册事件链表base->eventqueue。

2.3.3、event_del(删除事件)
//删除事件
int
event_del(struct event *ev)
该函数主要将事件ev从相应的链表上删除。

2.3.4、event_set(设置事件)


代码

/*设置event对象
**ev:事件对象
**fd:事件对应的文件描述符或信号,对于定时器设为-1
**events:事件类型,比如 EV_READ,EV_PERSIST, EV_WRITE, EV_SIGNAL
**callback:事件的回调函数
**arg:回调函数参数
*/
void
event_set(struct event *ev, int fd, short events,
      void (*callback)(int, short, void *), void *arg)

在将事件注册事件处理框架之前,应该先调用event_set对事件进行相关设置。

2.4、libevent对event的管理
event结构有3个链表结点域和一个小根堆索引,libevent通过3个链表和一个小根堆对I/O事件、signal事件和timer事件进行管理。
对于I/O事件,通过event_add将其加入event_base的注册事件链表eventqueue ;就绪时会加入event_base的就绪链表activequeues[];
对于timer事件,event_add将其加入到event_base的小根堆timeheap;
Signale
事件的管理相对复杂些,event_add将其加入到注册事件链表,同时,event_add内部会调用I/O
demultiplex的add函数(对于I/O事件也一样),比如epoll_add。而add函数又会调用evsignal_add将其加入到
evsignal_info的evsigevents[signo]链表(关于signal,后面会详细介绍)。

3、事件处理框架主循环
Libevent将I/O事件、signal事件和timer事件用统一的模型进行处理,这是非常精妙的。libevent主循环函数不断检测注册事件,如果有事件发生,则将其放入就绪链表,并调用事件的回调函数,完成业务逻辑处理。
3.1、event_dispatch
//事件处理主循环
int
event_dispatch(void)
这是呈现给外部的接口,它的实现很简单,即调用event_loop,而event_loop调用event_base_loop,event_base_loop完成实际的主循环逻辑。

3.2、event_base_loop
主要算法:


 1 done = ;
 2 while (!done) {
 3 
 4 /*如果没有就绪事件,根据timer heap中事件的最小超时时间,计算I/O demultiplex的
 5 **最大等待时间. 相反,如果有就绪事件,则清除tv,即I/O demultiplex不应该等待.
 6 */
 7 if (!base->event_count_active && !(flags & EVLOOP_NONBLOCK)) {
 8     timeout_next(base, &tv_p);
 9 } else {
     /* 
      * if we have active events, we just poll new events
      * without waiting.
      */
     evutil_timerclear(&tv);
 }
 
 /* If we have no events, we just exit */
 //没有事件处理,则退出循环
 if (!event_haveevents(base)) {
     event_debug(("%s: no events registered.", __func__));
     return ();
 }
 
 //tv_p为I/O demultiplex的超时时间
 //处理signal事件和I/O事件
 res = evsel->dispatch(base, evbase, tv_p);
 
 //处理timeout事件,对于超时的事件,将其放到就绪事件链表
 timeout_process(base);
 
 if (base->event_count_active) {
     //处理就绪事件
     event_process_active(base);
     if (!base->event_count_active && (flags & EVLOOP_ONCE))
         done = ;
 } else if (flags & EVLOOP_NONBLOCK)
     done = ;
 
 }//end while
 

3.3、timeout_next


 1 /*根据timer heap中事件的最小超时时间,计算I/O demultiplex的最大等待时间.
 2 **为了及时处理timer事件,I/O demultiplex的最大等待时间不应该超过timer事件中最小的超时时间,
 3 **否则,timer事件就不能得到及时处理
 4 */
 5 static int
 6 timeout_next(struct event_base *base, struct timeval **tv_p)
 7 {
 8     struct timeval now;
 9     struct event *ev;
     struct timeval *tv = *tv_p;
 
     //如果没有timer事件,则直接返回
     if ((ev = min_heap_top(&base->timeheap)) == NULL) {
         /* if no time-based events are active wait for I/O */
         *tv_p = NULL;
         return ();
     }
 
     if (gettime(base, &now) == -)
         return (-);
 
     //如果最小的timer事件已经超时,则清除tv,即I/O demultiplex不应该等待.
     if (evutil_timercmp(&ev->ev_timeout, &now, <=)) {
         evutil_timerclear(tv);
         return ();
     }
 
     //更新I/O demultiplex可以等待的最大时间:ev->ev_timeout - now
     evutil_timersub(&ev->ev_timeout, &now, tv);
     return ();
 }
 

3.4、dispatch函数
调用底层I/O multiplex的dispatch函数,具体的实现可以参见epoll的实现epoll_dispatch。

3.5、event_process_active

4、Timer事件
Timer事件的处理本身比较简单,不再赘述。

5、signal事件
5.1、socket pair
Libevent通过socketpair,将signal事件与I/O事件完美的统一起来。Socketpair,简单的说就一对socket,一端用于写,一端用于读。工作方式如下:

 为了与I/O事件统一起来,libevent内部使用了一个针对read socket的读事件。

5.1.1、Socketpair的创建

信号事件的初始化工作都是在evsignal_init中完成的,而evsignal_init通过调用evutil_socketpair创建
socketpair。对于Unix平台,有socketpair系统调用;对于Windows,则相对复杂一些,具体见
evutil_socketpair函数的实现。

5.2、evsignal_info
在event_base内部有一个evsignal_info类型的字段sig,它是用于管理signal事件的核心数据结构:


代码

 1 //evsignal.h
 2 struct evsignal_info {
 3 struct event ev_signal;    //内部socket读事件
 4 int ev_signal_pair[];  //对应socket pair的两个socket描述符
 5 int ev_signal_added;  //内部socket读事件是否已经加入注册链表
 6 volatile sig_atomic_t evsignal_caught; //是否有信号发生
 7 //信号事件链表数组,evsigevents[signo]表示注册信号signo的事件
 8 struct event_list evsigevents[NSIG]; 
 9 //具体记录每个信号触发的次数,evsigcaught[signo]是记录信号 signo被触发的次数
 sig_atomic_t evsigcaught[NSIG];
     
 //sh_old记录了原来的 signal 处理函数指针,当信号 signo 注册的 event 被清空时,需要重新设置其处理函数
 #ifdef HAVE_SIGACTION
 struct sigaction **sh_old;
 #else
 ev_sighandler_t **sh_old;
 #endif
 int sh_old_max;
 };
 

5.3、主要函数
5.3.1、evsignal_init
主要完成evsignal_info的初始化,主要算法:


代码

 1 int
 2 evsignal_init(struct event_base *base)
 3 {
 4 evutil_socketpair(AF_UNIX, SOCK_STREAM, , base->sig.ev_signal_pair);
 5 base->sig.sh_old = NULL;
 6 base->sig.sh_old_max = ;
 7 
 8 //事件发生次数设为0
 9 base->sig.evsignal_caught = ;
 memset(&base->sig.evsigcaught, , sizeof(sig_atomic_t)*NSIG);
 /* initialize the queues for all events */
 for (i = ; i < NSIG; ++i)
 TAILQ_INIT(&base->sig.evsigevents[i]);
 
 evutil_make_socket_nonblocking(base->sig.ev_signal_pair[]); //写端
 
 //设置内部读事件
 event_set(&base->sig.ev_signal, base->sig.ev_signal_pair[], 
 EV_READ | EV_PERSIST, evsignal_cb, &base->sig.ev_signal); //读端
 base->sig.ev_signal.ev_base = base;
 
 //sig.ev_signal == EV_READ | EV_PERSIST | EVLIST_INTERNAL
 base->sig.ev_signal.ev_flags |= EVLIST_INTERNAL;
 }


函数的关键在于这里会设置libevent用于管理信号事件的内部读事件evsignal_info的ev_signal,并将该事件对应的文件描述符设
为socket pair的读端。该函数由I/O multiplex的init函数调用。注:这里只是设置,而并没有注册socket
pair的读事件(见下一节)。

5.3.2、evsignal_add
当调用event_add注册信号事件时,内部会先调用
I/O
multiplex的add函数,add函数又会调用evsignal_add,将事件加到evsignal_info内部的信号事件链表。然后再
event_queue_insert将其添加到event_base的注册事件链表。

这里有两个地方需要注意,一是调用_evsignal_set_handler设置外部注册信号事件对应的信号的信号处理函数evsignal_handler:


代码

 1 static void
 2 evsignal_handler(int sig)
 3 {
 4     int save_errno = errno;
 5 
 6     //设置信号事件的发生次数
 7     evsignal_base->sig.evsigcaught[sig]++;
 8     evsignal_base->sig.evsignal_caught = ;
 9 
 #ifndef HAVE_SIGACTION
     signal(sig, evsignal_handler);
 #endif
 
     /* Wake up our notification mechanism */
     //向socket pair的写端写数据
     send(evsignal_base->sig.ev_signal_pair[], "a", , );
     errno = save_errno;
 }

当用户注册信号事件对应的信号发生时,OS转到evsignal_handler函数,从而设置sig.evsignal_caught,并向socket pair的写端发送数据。
二是通过调用event_add完成内部socket pair的读事件sig->ev_signal的注册。最后,将(外部)事件添加到信号事件链表。
5.3.2、与主循环结合
信号事件完成了注册,libevent就会在主循环中,等待事件发生,并处理事件。为了理解,来看看具体I/O demultiplex的dispatch函数:


代码

 1 static int
 2 epoll_dispatch(struct event_base *base, void *arg, struct timeval *tv)
 3 {
 4     struct epollop *epollop = arg;
 5     struct epoll_event *events = epollop->events;
 6     struct evepoll *evep;
 7     int i, res, timeout = -;
 8 
 9     if (tv != NULL)
         timeout = tv->tv_sec *  + (tv->tv_usec + ) / ;
 
     if (timeout > MAX_EPOLL_TIMEOUT_MSEC) {
         /* Linux kernels can wait forever if the timeout is too big;
          * see comment on MAX_EPOLL_TIMEOUT_MSEC. */
         timeout = MAX_EPOLL_TIMEOUT_MSEC;
     }
 
     //等待I/O事件
     res = epoll_wait(epollop->epfd, events, epollop->nevents, timeout);
 
     if (res == -) {
         if (errno != EINTR) {
             event_warn("epoll_wait");
             return (-);
         }
         //epoll_wait被信号中断
         evsignal_process(base);
         return ();
     } else if (base->sig.evsignal_caught) {//发生了信号事件
         //处理信号事件
         evsignal_process(base);
     }
 //…
 }

epoll_dispatch函数调用epoll_wait函数等待I/O发生。然后,如果有信号事件发生,则调用evsignal_process处理信号事件,evsignal_process的逻辑比较简单,它只是将事件从注册事件链表转移到就绪事件链表。


记得evsignal_handler函数吗?它是所有(外部)信号事件对应的信号的信号处理函数,将实际的信号发生时,OS会转而执行
evsignal_handler函数,而它便向socket pair的写端写数据,而读端收到数据。而此时,libevent的内部socket
pair读事件已经完成注册。libevent正阻塞在epoll_wait处,当socketp
pair读端收到数据时,libevent便从epoll_wait处返回。总之,signal事件通过socket
pair,与I/O事件实现完美的统一。

Libevent从epoll_wait返回后,它调用evsignal_process处理信号事件,然后调用event_active将I/O事件(包括内部的socket pair读事件)转移到就绪事件链表。

Socket pair的读事件回调函数:


代码

 1 static void
 2 evsignal_cb(int fd, short what, void *arg)
 3 {
 4 static char signals[];
 5 #ifdef WIN32
 6 SSIZE_T n;
 7 #else
 8 ssize_t n;
 9 #endif
 //接收数据
 n = recv(fd, signals, sizeof(signals), );
 if (n == -)
 event_err(, "%s: read", __func__);
 }

6、libevent的应用
Libevent
是一个非常优秀的开源网络库,它被许多其它开源程序使用。Memcache使用libevent作为底层的网络处理组件,并采用主线程(main
thread,单一)+工作线程(work thread,多个)的多线程模型(这将在Memcached的分析中详细介绍)。

libevent源码分析的更多相关文章

  1. 【转】libevent源码分析

    libevent源码分析 转自:http://www.cnblogs.com/hustcat/archive/2010/08/31/1814022.html 这两天没事,看了一下Memcached和l ...

  2. Libevent源码分析 (1) hello-world

    Libevent源码分析 (1) hello-world ⑨月份接触了久闻大名的libevent,当时想读读源码,可是由于事情比较多一直没有时间,现在手头的东西基本告一段落了,我准备读读libeven ...

  3. Libevent源码分析系列【转】

    转自:https://www.cnblogs.com/zxiner/p/6919021.html 1.使用libevent库     源码那么多,该怎么分析从哪分析呢?一个好的方法就是先用起来,会用了 ...

  4. Libevent源码分析系列

    1.使用libevent库     源码那么多,该怎么分析从哪分析呢?一个好的方法就是先用起来,会用了,然后去看底层相应的源码,这样比较有条理,自上向下掌握.下面用libevent库写个程序,每隔1秒 ...

  5. libevent源码分析二--timeout事件响应

    libevent不仅支持io事件,同时还支持timeout事件与signal事件,这篇文件将分析libevent是如何组织timeout事件以及如何响应timeout事件. 1.  min_heap ...

  6. libevent源码分析一--io事件响应

    这篇文章将分析libevent如何组织io事件,如何捕捉事件的发生并进行相应的响应.这里不会详细分析event与event_base的细节,仅描述io事件如何存储与如何响应. 1.  select l ...

  7. Libevent源码分析—event_init()

    下面开始看初始化event_base结构的相关函数.相关源码位于event.c event_init() 首先调用event_init()初始化event_base结构体 struct event_b ...

  8. Libevent源码分析—event, event_base

    event和event_base是libevent的两个核心结构体,分别是反应堆模式中的Event和Reactor.源码分别位于event.h和event-internal.h中 1.event: s ...

  9. Libevent源码分析—event_add()

    接下来就是将已经初始化的event注册到libevent的事件链表上,通过event_add()来实现,源码位于event.c中. event_add() 这个函数主要完成了下面几件事: 1.将eve ...

随机推荐

  1. np基本函数大全

    Numpy是科学计算库,是一个强大的N维数组对象ndarray,是广播功能函数.其整合C/C++.fortran代码的工具 ,更是Scipy.Pandas等的基础 .ndim :维度 .shape : ...

  2. WCF服务引用时错误: 无法导入 wsdl:portType详细信息

    WCF服务发布到IIS后,在客户端或WCFTestClient添加引用的时候报错如下: 错误: 无法导入 wsdl:portType详细信息: 在运行 WSDL 导入扩展时引发异常: System.S ...

  3. codechef Graph on a Table

    codechef Graph on a Table https://www.codechef.com/problems/TBGRAPH 题意 : 一个\(n\times m\)的网格图.\(q\) 个 ...

  4. LeetCode Delete Operation for Two Strings

    原题链接在这里:https://leetcode.com/problems/delete-operation-for-two-strings/description/ 题目: Given two wo ...

  5. Spring IOC容器的初始化—(一)Resource定位

    前言 上一篇博文“ Spring IOC是怎样启动的 ”中提到了refresh()方法,这个就是容器初始化的入口.容器初始化共有三个阶段: 第一阶段:Resource定位 第二阶段:BeanDefin ...

  6. JsQuick--个人封装的Js库

    JsQuick 该库为本人封装的Js库,尚未进行浏览器兼容 /** * 快速框架 版本:1.0.0 * 日期:2015.02.26 * 作者:简楚恩 */ /** * 快速获取控件类 */ var $ ...

  7. Redis 字符串(String)

    Redis 字符串(String) Redis 字符串数据类型的相关命令用于管理 redis 字符串值,基本语法如下: 语法 redis 127.0.0.1:6379> COMMAND KEY_ ...

  8. maven依赖顺序原则

    使用maven的程序员都会遇到一个问题,那就是maven依赖冲突的问题,这会导致ClassNotFound或者MethodNotFound这样的异常.其实只要明白maven依赖的根本性的原则就不怕这样 ...

  9. java代码做repeat次运算,从键盘输入几个数,比最值

    总结:今天这个题目有点灵活,因为它不但要求输出结果,还要进行几次相同的输入,不退出循环 import java.util.Scanner; //从键盘一次输入更多的数,然后把每一次的数进行---可比较 ...

  10. Annotation之三:自定义注解示例,利用反射进行解析

    @Retention定义了该Annotation被保留的时间长短有3中RetentionPolicy.SOURCE源文件有效,RetentionPolicy.CLASS:在class文件中有效,Ret ...