node之中连接redis使用的redis模块,虽然好用,但是有些地方还是需要注意。

npm install redis

redis client 行为:
1、客户端执行过程中断网的情况
  由于原本连接正常,断网后socket无法主动检测到,因此TCP进入不断的重传,ubuntu系统大概在1000秒左右返回ETIMEOUT。
  由于redis客户端没有等待回复超时时间,所以会等待到TCP超时才最终导致超时。
2、重连接机制:当服务端异常断开会导致重连接,而若是客户端主动断开则不会重连接
  如下两种情况表示客户端主动断开:
    1)向服务端发送quit命令
    2)调用客户端类的end函数
  重连接选项:
    max_attempts:重试次数,默认无限制
    retry_max_delay:两次重连接之间有一个延迟,默认按照指数避退原则不断增加,这个数值设置延迟的最大值
    connect_timeout:连接的超时时间(包括重连接),也就是若设置这个参数,达到这个值后就不会重连接(在这个时间内会按照重连接规则进行多次尝试,直到时间用尽)
3、设置client.stream.setTimeout这种方式,对于不断向服务器发送命令的应用无效
  因为sock的timeout表示空闲超时,只有在发送与接收都没有数据时才计入空闲时间,
  而由于应用不断向服务端发送命令,也就是在发送上是处于忙状态,不能记录为空闲。

4、关于end事件
  使用redisClient.quit()向服务端发送quit命令,服务端关闭连接,redis客户端能够收到end事件通知。
  使用redisClient.end()函数主动关闭连接时,redis客户端不会收到end事件通知。
  原因:
    redisClient.end()函数体中有一条语句
    this.stream._events = {}; //将socket连接上注册的所有事件清空
    由于redisClient上的事件是基于socket事件的,当socket事件清空后,redisClient本身也就无法触发任何事件了。
    延伸:socket的end事件触发时机:当读取到一个EOF时触发,因此若没有注册data事件(监控数据可读),也就不会有end事件。
    socket的close在被关闭时触发,因此只要不是极端的清除所有的注册事件,都可以收到该事件通知。
5、关于redisClient的命令队列与离线队列
  命令队列offline_queue:
  当前连接可用时客户端会将命令不断发出去,每出去一个在队列末尾添加一项,收到一个回复从队列头部删除一项。因此当应用使用多个连接时可能会出现,A连接执行set命令,B连接去查询为空,那是由于A连接上的set命令还在队列里等待redis执行。
  长度查询:client.command_queue.length
  离线队列offline_queue:
  当前连接不可用时,客户端会将命令放置到离线队列之中,同时重连接规则尝试连接服务器,每次尝试失败都会清空离线队列。
  长度查询:client.offline_queue.length
  可使用enable_offline_queue:false选项关闭该功能。
6、错误时的回调
  若在执行命令中设置了回调函数,当有异常时这个回调会先被调用,然后触发error事件。

05-13 19:20