Redis quickList
QuickList briefly
-
List objects in Redis before version 3.2, the underlying encoding of the list was implemented by Ziplist and LinkedList, but after version 3.2, a QuickList data structure was reintroduced, and the underlying encoding of the list was implemented by QuickList.
-
In earlier designs, ziplist was used for small or small numbers of elements in a list object, and two-way linkedList was used for large or large numbers of elements in a list object.
The advantages and disadvantages of these two storage methods
- The bidirectional linkedList is convenient for push and pop operations at both ends of the table. It has low complexity in inserting nodes, but it has a large memory overhead. First, it holds two Pointers on each node in addition to data; Secondly, each node of bidirectional linked list is a separate memory block, the address is not continuous, more nodes will easily generate memory fragmentation.
- Ziplist stores on a contiguous piece of memory, so storage is efficient. However, it is not conducive to modification operations, as insert and delete operations require frequent memory requisition and freeing. Especially if the Ziplist is long, one realloc may result in a large number of data copies.
The basic structure
Parameter configuration
-
list-max-ziplist-size 2 -
Copy the codeA positive value limits the ziplist length on each QuickList node according to the number of data items. For example, when this parameter is set to 5, the ziplist of each QuickList node contains a maximum of five items.
When negative, the ziplist length on each QuickList node is limited by the number of bytes consumed. In this case, it can only take five values from -1 to -5. Each value has the following meanings:
-
-5: The ziplist size of each QuickList node cannot exceed 64 Kb. (note: 1kb => 1024 bytes)
-
-4: The ziplist size of each QuickList node cannot exceed 32 Kb.
-
-3: The ziplist size of each QuickList node cannot exceed 16 Kb.
-
-2: The ziplist size of each QuickList node cannot exceed 8 Kb. (-2 is the default value given by Redis)
-
-1: The ziplist size of each QuickList node cannot exceed 4 Kb.
-
list-compress-depth 0
Copy the codeThis parameter represents the number of uncompressed nodes at both ends of a QuickList. Note: The number of nodes here refers to the number of nodes in quickList, not the number of items in Ziplist. In fact, a Ziplist on a QuickList node, if compressed, is compressed as a whole.
The list-compressed-depth parameter has the following meanings:
-
0: this is a special value, indicating that none is compressed. This is the default value for Redis.
-
1: indicates that one node at both ends of the QuickList is not compressed, and the middle node is compressed.
-
2: indicates that two nodes at both ends of the QuickList are not compressed, and the middle node is compressed.
-
3: indicates that three nodes at both ends of the QuickList are not compressed, and the middle node is compressed.
-
And so on…
Since 0 is a special value, it is easy to see that the head and tail of the QuickList are always uncompressed for quick access at both ends of the table.
Redis uses LZF, a lossless compression algorithm, to compress quickList internal nodes.
Data structure definition for QuickList
typedef struct quicklistNode {
struct quicklistNode *prev;
struct quicklistNode *next;
unsigned char *zl;
unsigned int sz; /* ziplist size in bytes */
unsigned int count : 16; /* count of items in ziplist */
unsigned int encoding : 2; /* RAW==1 or LZF==2 */
unsigned int container : 2; /* NONE==1 or ZIPLIST==2 */
unsigned int recompress : 1; /* was this node previous compressed? * /
unsigned int attempted_compress : 1; /* node can't compress; too small */
unsigned int extra : 10; /* more bits to steal for future usage */
} quicklistNode;
typedef struct quicklistLZF {
unsigned int sz; /* LZF size in bytes*/
char compressed[];
} quicklistLZF;
typedef struct quicklist {
quicklistNode *head;
quicklistNode *tail;
unsigned long count; /* total count of all entries in all ziplists */
unsigned int len; /* number of quicklistNodes */
int fill : 16; /* fill factor for individual nodes */
unsigned int compress : 16; /* depth of end nodes not to compress; 0=off */
} quicklist;
Copy the code
The quicklistNode structure represents a quickList node, where the fields have the following meanings:
- Prev: pointer to the previous node in the list.
- Next: a pointer to the next node in the list.
- Zl: data pointer. If the current node’s data is not compressed, it points to a Ziplist structure; Otherwise, it points to a quicklistLZF structure.
- Sz: represents the total size of the Ziplist that zL points to (including
zlbytes
.zltail
.zllen
.zlend
And individual data items). Note that if ziplist is compressed, the sZ value is still the ziplist size before compression. - Count: indicates the number of data items contained in ziplist. This field is only 16 bits long. We’ll figure out later if that’s enough.
- Encoding: Indicates whether ziplist is compressed (and which compression algorithm is used). Currently, there are only two values: 2 means compressed (using LZF compression) and 1 means not compressed.
- Container: Indicates a reserved field. It is designed to indicate whether a QuickList node is used to store data directly, ziplist, or some other structure (used as a data container, hence container). However, in the current implementation, this value is a fixed value of 2, indicating the use of ziplist as a data container.
- Recompress: When you want to extract the compressed data using a command like lindex, set recompress=1 and wait for the opportunity to compress the data.
- Attempted_compress: This value is only useful to Redis automated test programs. We don’t have to worry about that.
- Extra: other extended fields. It is not currently used in the Redis implementation.