1. Introduction to Redis
1.为什么要用Nosql
In order to solve a series of problems such as high concurrency, high availability, high expansion, multi-data storage and database solutions, is Nosql
2. Nosql (Not only SQL)
Relational databases: include databases such as mysql and Oracle, and contain tables, rows, and columns
Nosql stands for not just SQL, but non-relational databases,
3. No characteristics
1, easy to expand (there is no relationship between data, good expansion)
2, large amount of data performance (Redis write 80,000 times a second, read 110,000 times, Nosql cache record level, is a fine-grained cache, high performance)
3. Diversity of data types (no need to design the database beforehand)
4. Distinguish between traditional RDBMS and Nosql
The traditional DBMS
-
— Structured organization
-
— SQL
-
Data and relationships are stored in separate tables
-
The language in which data definitions are manipulated
-
— Strict consistency
-
— Basic transactions
Nosql
-
— It’s not just data
-
— There is no set language
-
— Key-value pair storage, column storage, document storage, graph database (social relationship)
-
— Final consistency
-
— CAP theorem and BASE
-
— High performance, high availability, and high scalability
5. The 3V in the era of big data: It mainly describes problems
-
Massive Velume
-
Multiple segments
-
Real-time Velocity
6, the era of big data 3 high: mainly on the requirements of the program
-
High concurrency
-
High scalability
-
A high performance
Real practice in the company: NoSQL + RDBMS together is the strongest.
The four categories of Nosql
1. Key-value key value pairs
-
Sina: Redis
-
Meituan: Redis + Tair
-
Alibaba, Baidu: Redis+meme’cache
2. Document database (BSON format is the same as JSON)
- MongoDB
MongoDB is a database based on distributed file storage, written in C++, mainly used to process a large number of documents! MongoDB is a product in between relational databases and non-relational data! MongoDB is the most functional of the non-relational data libraries, the most like a relational database!Copy the code
- ConthDB
3. Column storage database
- HBase
- Distributed file system
4. Graph relational database
- What we store is not graph, but relationship, such as social network of friends circle, advertisement recommendation and so on
- Neo4h, InfoGrid
Comparison of four Nosql categories
2. Introduction to Redis
1. What is Redis?
Redis (Remote Dictionary Server)
Redis is a free, open source, network-enabled, memory-based and persistent logging, key-value database written in ANSIC, and provides multiple language apis.
Redis periodically writes the updated data to disk or the modified operation to the appended record file, and implements master-slave synchronization on this basis.
2. What can Redis do?
-
1. Memory storage, persistence, memory power is lost, so persistence is very important (RDB, AOF) 2. 3. Publish and subscribe system 4. Map information analysis 5. Timers, counters 6........Copy the code
3, features,
-
1. Multiple data types 2. Persistence 3. Cluster 4.Copy the code
3, Redis installation
1. Official website: redis. IO /
Chinese version: www.redis.cn/
3. Download address: download from the official website
Note: The Windows version is available for download on GitHub, but has been stopped for much longer
Redis recommends building on a Linux server, and currently I am learning based on Linux
1. Windows installation
1. Download the installation package: github.com/dmajkic/red…
2. Get the compressed package after downloading:
Decompression to their own computer environment directory can be! Redis is very small, only 5M
4. Click Redis-server. exe to run
2. Linux installation
1. Download the installation package redis-5.0.8.tar.gz
2. Run tar -zxvf redis-5.0.8.tar.gz to decompress the installation package to the /opt directory
3. Access the decompressed redis file and view the redis configuration file
4. Installation of Redis basic environment
1. yum install gcc-c++
2. make
3. make
Copy the code
The default Redis installation path is /usr/local/bin
6. Copy the redis configuration file redisConfig to the redisConfig file
Daemonize no = yes = yes = yes = yes = yes
8. Start the redis service redisConfig/redis.conf
9. Use Redis-CLI for connection test
10. Check whether the Redis process is started
3, Redis built-in test performance tools
Redis-benchmark is an official redis stress test tool
Redis-benchmark command parameters are as follows:
Just test it out
Analysis:
Redis basic syntax:
Flushdb # flushhall # flushall 1. Keys * # check all key pairs of redis set get check keys exists check keys move remove current key expire check keys expireCopy the code
Redis is single-threaded
Redis is based on memory operation, CPU is not Redis performance bottleneck, Redis bottleneck is based on the machine’s memory and network bandwidth, since it can be implemented with a single thread, with a single thread implementation
Redis is written in C language, and the official data provided is 100000+ QPS, not worse than Memecache which also uses key-vale!
Question: Why is Redis single thread so fast?
High performance servers are not necessarily multithreaded (Myth 1: High performance servers are not necessarily multithreaded?)
2. Multithreading is not necessarily more efficient than single threading (Myth 2: Multithreading (CPU context switches!) Must be more efficient than a single thread!)
3. The CPU speed is higher than the memory speed, and the memory speed is higher than the hard disk CPU> Memory > hard disk
4, core point: Redis is to put all the data in memory, for Redis to use a single thread to operate is the most efficient, multithreading (CPU context switch: For memory systems, if there is no CPU context switch, the efficiency is the highest, multiple reads and writes are in the same CPU, in memory, this is the best solution.
5. Five data types of Redis
The official documentation
Redis is an open source (BSD-licensed), in-memory data structure storage system that can be used as a database, cache, and messaging middleware. It supports many types of data structures, such as strings (hashes), lists (Lists), sets (sets), sorted sets (sorted sets) and range queries, bitmaps, Hyperloglogs and Geospatial index radius query. Redis is built with replication, LUA scripting, LRU eviction, transactions and different levels of disk persistence, And provides high availability through Redis Sentinel and Automated partitioning (Cluster).
1, Redis – key
Redis basic syntax:
Flushdb # flushdb # keys # flushdb # flushdb # keys * # flushdb # flushdb # keys * # flushdb # flushdb # keys * # flushdb # flushdb # keys * # flushdb # flushdb Move # Delete current key expire # set current key expire type # check current key type TTL # check current key'Copy the code
String is used in a similar way: value can be our number as well as our String! Counter counts the number of multiple units the number of fans object cache storage!
Redis case insensitive command
2. String (String)
127.0.0.1:6379> set key1 v1 # set value OK 127.0.0.1:6379> get key1 # set value "v1 127.0.0.1:6379> keys * # check all values 1) "key1" 127.0.0.1:6379> EXISTS key1 # Check whether the value EXISTS (integer) 1 127.0.0.1:6379> APPEND key1 "helloQ" # APPEND the string if the current key does not exist, Equivalent to setKey (INTEGER) 8 127.0.0.1:6379> get key1" v1HelloQ "127.0.0.1:6379> keys * 1) "key1" 127.0.0.1:6379> STRLEN key1 8 127.0.0.1:6379> APPEND key1 (integer) 17Copy the code
i++
Step I + =
127.0.0.1:6379> set views 0 # get views "0" 127.0.0.1:6379> incr views 1 127.0.0.1:6379> INCr views (integer) 2 127.0.0.1:6379> get views "2" 127.0.0.1:6379> decr views 1 127.0.0.1:6379> Decr views (INTEGER) 0 127.0.0.1:6379> DECr views (integer) -1 127.0.0.1:6379> get views "-1" 127.0.0.1:6379> INCRBY views 10 # (INTEGER) 9 127.0.0.1:6379> INCRBY views 10 (integer) 19 127.0.0.1:6379> DECRBY views 5 (integer) 14Copy the code
String range
127.0.0.1:6379> set key1 "hello,kuangshen" # set key1 to OK 127.0.0.1:6379> get key1 "hello,kuangshen" 127.0.0.1:6379> GETRANGE key1 0 3 # intercept string [0,3] "hell" 127.0.0.1:6379> GETRANGE key1 0 -1Copy the code
Replace!
127.0.0.1:6379> set key2 abcdefg OK 127.0.0.1:6379> get key2 "abcdefg" 127.0.0.1:6379> SETRANGE key2 1 xx # Replaces the string at the beginning of the specified position! (INTEGER) 7 127.0.0.1:6379> get key2 "axxdefg"Copy the code
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
Setex (set with expire) # set expiration time
Setnx (set if not exist) # does not exist in Settings
127.0.0.1:6379> setex key3 30 "hello" # set key3 to hello and expire after 30 seconds OK 127.0.0.1:6379> TTL key3 (integer) 26 127.0.0.1:6379> Get key3 "hello" 127.0.0.1:6379> setnx mykey "redis" # Create mykey (integer) 1 127.0.0.1:6379> keys * 1) "key2" 2) "mykey" 3) "key1" 127.0.0.1:6379> TTL key3 (integer) -2 127.0.0.1:6379> setnx mykey "MongoDB" (INTEGER) 0 127.0.0.1:6379> get mykey "redis"Copy the code
########################################################################## mset mget
127.0.0.1:6379> mset k1 v1 k2 v2 k3 v3 # set multiple values OK 127.0.0.1:6379> keys * 1) "k1" 2) "k2" 3) "k3" 127.0.0.1:6379> mget 1) "v1" 2) "v2" 3) "v3"Copy the code
String is used in a similar way: value can be our number as well as our String! Counter counts the number of multiple units the number of fans object cache storage! List Basic data type, List
127.0.0.1:6379> msetnx k1 v1 k4 v4 # msetnx is an atomic operation that either succeeds together or fails together! (integer) 0 127.0.0.1:6379> get k4 (nil)Copy the code
# object
Set user:1 {name:zhangsan,age:3} # set user:1 {name:zhangsan,age:3} # set user:1 {name:zhangsan,age:3}
The key is a clever design: user:{id}:{filed}, so the design is completely OK in Redis!
127.0.0.1:6379> mset user:1:name zhangsan user:1:age 2
OK
127.0.0.1:6379> mget user:1:name user:1:age
"zhangsan"
"2"
Copy the code
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # getset to get first and then in the set
127.0.0.1:6379> getset db redis # 127.0.0.1:6379> get db "redis 127.0.0.1:6379> getset db mongodb # Redis 127.0.0.1:6379> get db" mongodb"Copy the code
It’s all the same data structure it’s all key-value
String has many similar scenarios: value can be our number as well as our string!
-
counter
-
Count the number of multiple units
-
Number of fans
-
Object cache storage!
-
.
3, the List
List is the basic data type, List
In Redis, lists can become stacks, queues, blocking queues, and so on
List basic syntax
Remember: all list commands start with L
127.0.0.1:6379> LPUSH list one Insert into the list header (left) (INTEGER) 1 127.0.0.1:6379> LPUSH list two (integer) 2 127.0.0.1:6379> LPUSH list three (integer) 3 127.0.0.1:6379> LRANGE list 0 -1 1) "three" 2) "two" 3) "one" 127.0.0.1:6379> LRANGE list 0 1 1) "three" 2) "two" 127.0.0.1:6379> Rpush list righr # (INTEGER) 4 127.0.0.1:6379> LRANGE list 0-1 1) "three" 2) "two" 3) "one" 4) "righr"Copy the code
LPOP and Rpop
127.0.0.1:6379> LRANGE list 0-1 1) "three" 2) "two" 3) "one" 127.0.0.1:6379> LPOP list # remove first element "three" 127.0.0.1:6379> LRANGE list 0-1 1) "two" 2) "one" 127.0.0.1:6379> RPOP list # remove the last element of list "one" 127.0.0.1:6379> LRANGE List 0-1 1) "two" 127.0.0.1:6379>Copy the code
Lindex
127.0.0.1:6379> LRANGE list 0-1 1) "two" 2) "one" 127.0.0.1:6379> lindex list 1 "One" 127.0.0.1:6379 > lindex list 0 "two" # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # Llen 127.0.0.1:6379 > Lpush list one (integer) 1 127.0.0.1:6379> Lpush list two bilibili: Java(integer) 2 127.0.0.1:6379> Lpush list three (integer) 3 127.0.0.1:6379> Llen listCopy the code
Removes the specified value! Take off the uid Lrem
127.0.0.1:6379> LRANGE list 0 -1 1) "three" 2) "three" 3) "two" 4) "one" 127.0.0.1:6379> lrem list 1 one # Remove the specified number of values from the list. Exact matching (INTEGER) 1 127.0.0.1:6379> LRANGE list 0-1 1) "three" 2) "three" 3) "two" 127.0.0.1:6379> lrem list 1 three (integer) 1 127.0.0.1:6379> LRANGE list 0-1 1) "three" 2) "two" 127.0.0.1:6379> Lpush list three (integer) 3 127.0.0.1:6379> lrem list 2 three (integer) 2 127.0.0.1:6379> LRANGE list 0-1 1) "two"Copy the code
Trim the trim. ; Cutting the list!
127.0.0.1:6379> keys * (empty list or set) 127.0.0.1:6379> Rpush myList "hello" (integer) 1 127.0.0.1:6379> Rpush myList "Hello1" (integer) 2 127.0.0.1:6379> Rpush myList "hello2" (integer) 3 127.0.0.1:6379> Rpush myList "hello3" (integer) 3 127.0.0.1:6379> Rpush myList "hello3" (integer) 4 127.0.0.1:6379> ltrim myList 12 # Truncated by subscript, truncated only truncated elements are left! OK 127.0.0.1:6379> LRANGE myList 0-1 1) "hello1"Copy the code
rpoplpush
Rpoplpush # remove the last element of the list and move it to the new list! 127.0.0.1:6379> rpush myList "hello" bilibili: Before Node after, left, and right can insert values. If the key does not exist, create a new list. New Content (integer) 1 127.0.0.1:6379> rpush myList "hello1" (integer) 2 127.0.0.1:6379> rpush myList "hello2" (integer) 3 127.0.0.1:6379> rpoplpush myList myotherList # remove the last element of the list and move it to the new list! "Hello2" 127.0.0.1:6379> lrange myList 0-1 # Check the original list 1) "hello" 2) "hello1" 127.0.0.1:6379> lrange myOtherList 0-1 # Check the target list, there is a change! 1) "hello2"Copy the code
Lset replaces the index value in the list with another value, an update operation
127.0.0.1:6379> EXISTS list # Check whether the list EXISTS (integer) 0 127.0.0.1:6379> lset list 0 item # If the list does not exist, we will error ERR no Such key 127.0.0.1:6379> lpush list value1 (integer) 1 127.0.0.1:6379> LRANGE list 0 0 1) "value1" 127.0.0.1:6379> lset 127.0.0.1:6379> LRANGE list 0 0 1) "item" 127.0.0.1:6379> lset list 1 An error will be reported! (error) ERR index out of rangeCopy the code
Linsert # inserts a specific value into the column before or after an element in your column!
127.0.0.1:6379> Rpush myList "hello" (integer) 1 127.0.0.1:6379> Rpush myList "world" (integer) 2 127.0.0.1:6379> LINSERT myList before "world" "other" (integer) 3 127.0.0.1:6379> LRANGE myList 0-1 1) "hello" 2) "other" 3) "world" 127.0.0.1:6379> LINSERT myList after world new (integer) 4 127.0.0.1:6379> LRANGE myList 0-1 1) "Hello" 2) "other" 3) "world" 4) "new"Copy the code
Summary:
-
List is actually a linked list. Before, None, after, left, and right can all insert values
-
If the key does not exist, a new list is created
-
If the key exists, add content
-
If you remove all the values, it’s an empty list, and it doesn’t exist
-
It’s most efficient to insert or change the values on both sides, and the middle element is a little bit less efficient
Message queue! Message queue (Lpush Rpop), stack (Lpush Lpop)!
4, the set
All commands in set begin with s, and the values in set cannot repeat REM
127.0.0.1:6379> sadd myset "hello" # set Add uniform speed (integer) 1 127.0.0.1:6379> sadd myset "kuangshen" (integer) 1 127.0.0.1:6379> sadd myset "loveKuangshen" (integer) 1 127.0.0.1:6379> SMEMBERS myset "Lovekuangshen" 3) "kuangshen" 127.0.0.1:6379> SISMEMBER myset hello # (integer) 1 127.0.0.1:6379> SISMEMBER MySet World (integer) 0 # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # 127.0.0.1:6379 > scard myset # Get the number of content elements in the set! (integer) 4Copy the code
1 127.0.0.1:6379> scard mySet (integer) 3 127.0.0.1:6379> SMEMBERS myset 1) "lovekuangshen2" 2) "lovekuangshen" 3) "kuangshen"Copy the code
Set unordered non-repeating collection. Take out random!
127.0.0.1:6379> SMEMBERS myset 1) "kuangshen 2" 2) "kuangshen" 3) "kuangshen" 127.0.0.1:6379> SRANDMEMBER myset # 127.0.0.1:6379> SRANDMEMBER myset "kuangshen" 127.0.0.1:6379> SRANDMEMBER myset "kuangshen" 127.0.0.1:6379> SRANDMEMBER myset "kuangshen" 127.0.0.1:6379> SRANDMEMBER myset 127.0.0.1:6379> SRANDMEMBER myset 1) "loveKuangshen" 2) "loveKuangshen2" 127.0.0.1:6379> SRANDMEMBER myset # "lovekuangshen2"Copy the code
Delete certain key, randomly delete key!
127.0.0.1:6379> SMEMBERS myset 1) "lovekuangshen2" 2) "lovekuangshen" 3) "kuangshen" 127.0.0.1:6379> spop myset # Randomly remove some elements from the set! 127.0.0.1:6379> SMEMBERS myset 1) "kuangshen"Copy the code
Move a specified value to another set!
127.0.0.1:6379> sadd myset "Hello" (integer) 1 127.0.0.1:6379> sadd myset "world" (integer) 1 127.0.0.1:6379> sadd myset "Kuangshen" (integer) 1 127.0.0.1:6379> sadd myset2 "set2" (integer) 1 127.0.0.1:6379> smove myset myset2 "kuangshen" # Moves a specified value to another set. (integer) 1 127.0.0.1:6379> SMEMBERS myset 1) "world" 2) "hello" 127.0.0.1:6379> SMEMBERS myset2 1) "kuangshen" 2) "set2"Copy the code
Micro-blog, B station, common concern! (union) Number set class:
-
Difference set SDIFF
127.0.0.1:6379> SDIFF key1 key2 # difference set 1) "b" 2) "a" 127.0.0.1:6379> SINTER key1 key2 # intersection common friends can be done like this 127.0.0.1:6379> SUNION key1 key2 # union 1) "b" 2) "c" 3) "e" 4) "a"Copy the code
Bilibili: The crazy god says that in Java microblogging, A user puts all his followers into A set! Put its fans in a collection too! Common attention, common interest, second friends, recommend friends! (Six degrees of segmentation theory)
-
intersection
-
And set
5. Hash
Key-map this value is a Map set! Not much different from string, but a simple key-value
set myhash field haspmap
127.0.0.1:6379> hset hash field1 test # set a specific key-value (integer) 1 127.0.0.1:6379> hget hash field (nil) 127.0.0.1:6379> hmset hash field1 Hello field2 world #set multiple key-values OK 127.0.0.1:6379> hmget hash field1 field2 # Retrieve multiple field values 1) "hello" 2) "world" 127.0.0.1:6379> hget hash (error) ERR wrong Number of arguments for 'hget' command 127.0.0.1:6379> hgetall Hash # Fetch all data 1) "field1" 2) "hello" 3) "field2" 4) "World" 127.0.0.1:6379> hdel hash field1 (integer) 1 127.0.0.1:6379> hgetall Hash 1) "field2" 2) "world" 127.0.0.1:6379>Copy the code
hlen
127.0.0.1:6379> hmset myhash field1 hello field2 world OK 127.0.0.1:6379> HGETALL myhash 1) "field2" 2) "world" 3) "Field1" 4) "hello" 127.0.0.1:6379> hlen myHash # (integer) 2Copy the code
myhash
127.0.0.1:6379> HEXISTS myhash field1 (integer) 1 127.0.0.1:6379> HEXISTS myHash Field3 (integer) 0Copy the code
Only get all fields
Just get all the values
127.0.0.1:6379> hkeys myhash # delete value 1) "world" 2) "hello"Copy the code
incr decr
127.0.0.1:6379> hset MyHash field3 5 # (integer) 1 127.0.0.1:6379> HINCRBY MyHash Field3 1 (integer) 6 127.0.0.1:6379> HINCRBY MyHash Field3-1 (integer) 5 127.0.0.1:6379> hsetnx myHash field4 Hello # If it does not exist (integer) 1 127.0.0.1:6379> hsetnx myHash field4 World # If it exists, (integer) 0 cannot be setCopy the code
Summary:
Hash changed data users, names, and ages, especially user information, change frequently. Hash is more suitable for storing objects, and String is more suitable for storing strings
6. Zset (ordered)
Set k1 v1 zset k1 score1 v1
127.0.0.1:6379> zadd myset 1 one # add a value (integer) 1 127.0.0.1:6379> zadd myset 2 two three 127.0.0.1:6379> ZRANGE myset 0 -1 1) "one" 2) "two" 3) "three"Copy the code
How to implement sorting
127.0.0.1:6379> zadd Salary 5000 zhangsan (integer) 1 127.0.0.1:6379> zhangsan (integer) 1 127.0.0.1:6379> zadd salary 500 kaungshen (integer) 1 # ZRANGEBYSCORE key min Max 127.0.0.1:6379> ZRANGEBYSCORE salary -inf + INF # display all users from small to large! 1) "kaungshen" 2) "xiaohong" 3) "Zhangsan" 127.0.0.1:6379> ZREVRANGE salary 0-1 # 1) "zhangsan" 2) "kaungshen" 127.0.0.1:6379> ZRANGEBYSCORE salary - INF + INF withscores # 2) "500" 3) "xiaohong" 4) "2500" 5) "zhangsan" 6) "5000" 127.0.0.1:6379> ZRANGEBYSCORE salary - INF 2500 withscores # Display the ascending order of employees with salary less than 2500! 1) "kaungshen" 2) "500" 3) "xiaohong" 4) "2500"Copy the code
Remove elements from REM
127.0.0.1:6379> zrange salary 0-1 1) "kaungshen" 2) "xiaohong" 3) "Zhangsan" 127.0.0.1:6379> Zrem salary 0-1 1) "kaungshen" 2) "Xiaohong" 3) "Zhangsan" 127.0.0.1:6379> Zrem salary 0-1 Remove the specified element from the ordered set (integer) 1 127.0.0.1:6379> zrange salary 0-1 1) "kaungshen" 2) "Zhangsan" 127.0.0.1:6379> zcard salary # Gets the number of ordered sets (integer) 2Copy the code
127.0.0.1:6379> zadd myset 1 Hello (integer) 1 127.0.0.1:6379> zadd myset 2 world 3 kuangshen (integer) 2 127.0.0.1:6379> zcount myset 1 (integer) 3 127.0.0.1:6379> zcount myset 12 (integer) 2Copy the code
summary
Mainly learn to read official documents,
Set sort store class grade table, salary table sort
Common message 1. Important message 2. Judge with weight
Ranking application implementation, to topN test!
6. Three special data types
1. Geospatial
-
Add the specified address space (longitude, latitude, name) to the specified key, and the data will be stored in the sorted set. The purpose of this is to facilitate the use of GEORADIUS or GEORADIUSBYMEMBER command for radius query of data and other operations
-
The command takes arguments x,y in standard format, so longitude must precede latitude. The limits of these coordinates can be indexed, and the area can be close to the pole but not indexed. There are also partial restrictions on longitude and latitude:
-
Effective longitude ranges from -180 degrees to 180 degrees
-
Valid latitudes range from -85.05112878 degrees to 85.05112878 degrees
-
An error is returned when the coordinate position is outside the range described above
Related commands:
-
geoadd
-
geodist
-
geohash
-
geopos
-
georadius
-
GEORADIUSBYMEMBER
getadd
Getadd adds the geographic location
Rule: the two levels can not be directly added, we will generally download the city data, directly through the Java program one-time import!
- Effective longitude ranges from -180 degrees to 180 degrees.
- Valid latitudes range from -85.05112878 degrees to 85.05112878 degrees.
- This command returns an error when the coordinates are outside the range specified above.
# 127.0.0.1:6379> Geoadd China :city 39.90 116.40 Beijin (error) ERR invalid longitude, Latitude pair 39.900000,116.400000 # key value () 127.0.0.1:6379> geoAdd China :city 116.40 39.90 Beijing (integer) 1 127.0.0.1:6379> geoadd China :city 121.47 31.23 Shanghai (INTEGER) 1 127.0.0.1:6379> Geoadd China: City 106.50 29.53 CHONGqi 114.05 22.52 shengzhen (INTEGER) 2 127.0.0.1:6379> Geoadd China: City 120.16 30.24 Hangzhou 108.96 34.26 Xian (integer) 2Copy the code
getpos
Gets current location: must be a coordinate value
127.0.0.1:6379> GEOPOS China: City Beijing 1) 1) "GEOPOS China: City Beijing Chongqi 1) 1) 1) 2)" "116.39999896287918091" 2) "39.90000009167092543" 2) 1) "106.49999767541885376" 2) "29.52999957900659211"Copy the code
getdist
The distance between two people
Unit: m
M is in meters. Km is expressed in kilometers. Mi means miles. Ft is in feet.
127.0.0.1:6379> GEODIST China :city Beijing Shanghai km # Beijing chongqi km # check the linear distance from Chongqing to Beijing "1464.0708"Copy the code
georadius
Given latitude and longitude, find an element within the radius
127.0.0.1:6379> GEORADIUS China: City 110 30 1000 km # Find a city within 1000km 1) "Chongqi" 2) "xian" 3) "shengzhen" 4) "Hangzhou" 127.0.0.1:6379> GEORADIUS China: City 110 30 500 km 1) "chongqi" 2) "xian" 127.0.0.1:6379> GEORADIUS China: City 110 30 500 km withdist # display to middle distance location 1) 1) "chongqi" 2) "341.9374" 2) 1) "xian" 2) "483.8340" 127.0.0.1:6379> GEORADIUS China: City 110 30 500 km withcoord # show other's location info 1) 1) "Chongqi "2) 1) 106.49999767541885376" 2) "29.52999957900659211" 2) 1) "xian" 2) 1) "108.96000176668167114" GEORADIUS China: City 110 30 500 km withdist withcoord count 1 # 1) 1) "chongqi" 2) "341.9374" 3) 1) "106.49999767541885376" 2) "29.52999957900659211" 127.0.0.1:6379> GEORADIUS China: City 110 30 500 km Withdist Withcoord Count 2 1) 1) "chongqi" 2) "341.9374" 3) 1) "106.49999767541885376" 2) "29.52999957900659211" 2) 1) "2) 2) "2) "2) "2)"Copy the code
georadiusbymember
Find other elements around the specified element! 127.0.0.1:6379> GEORADIUSBYMEMBER China: City Beijing 1000 km 1) "Beijing" 2) "xian" 127.0.0.1:6379> GEORADIUSBYMEMBER china:city shanghai 400 km 1) "hangzhou" 2) "shanghaCopy the code
geohash
Returns the geoHash element at one or more locations
## returns a string of 11 characters
Convert the latitude and longitude from two dimensions to a one-dimensional string. The closer the two strings are, the closer they are! 127.0.1:6379 > Geohash China: City Beijing Chongqi 1) "wx4fbpb ke0" 2) "wm5xzrybty0"Copy the code
2, Hyperloglog
1. What is the cardinal number?
{1,2,3,4,5, 6} b:{1,2,3,4,5}
Cardinality (non-repeatable elements) =6, error is acceptable
2. Redis Hyperloglog cardinal statistics algorithm
Advantages: The memory footprint is fixed, 2*64 different elements of the technology, only cost 12KB of memory! Hyperloglog is preferred for memory comparison
UV of web pages (one person visits a site multiple times, but still counts as one person!) Traditionally, a set stores the user’s ID, and then counts the number of elements in the set as a standard judgment! This way, if you save a large number of user ids, will be more troublesome! Our purpose is to count, not save user ids; 0.81% error rate! Statistical UV task, can be ignored!
127.0.0.1:6379> PFadd mykey a b c d e f g h I j # create mykey (integer) 1 127.0.0.1:6379> PFCOUNT mykey Number of cardinality elements (integer) 10 127.0.0.1:6379> PFadd mykey2 I j z x c v b n m # Create the second group of elements mykey2 (integer) 1 127.0.0.1:6379> PFCOUNT Mykey2 (integer) 9 127.0.0.1:6379> PFMERGE mykey3 mykey mykey2 => merge mykey3 mykey2 OK 127.0.0.1:6379> PFCOUNT mykey3 # look at the number of union! (integer) 15Copy the code
3, Bitmaps
A storage
Statistics user information, active, inactive! Logged in, not logged in! Punch in, 365 punch in! You can use Bitmaps for both! Bitmap, data structure! Both operate on binary bits to record, there are only 0 and 1 states! 365 days = 365 bits 1 byte = 8bits 46 bytes!
Use bitmap to record your clocking from Monday to Sunday! Monday: 1 Tuesday: 0 Wednesday: 0 Thursday: 1……
Check to see if you clock in on a particular day
127.0.0.1:6379> getbit sign 3 (integer) 0 127.0.0.1:6379> getbit sign 4 (integer) 1 127.0.0.1:6379> Number of days for collecting statistics on card punching 127.0.0.1:6379> BITCOUNT sign # 127.0.0.1:6379>Copy the code
7. Redis transactions
The essence of Redis transactions: a collection of commands! All commands in a transaction are serialized and executed sequentially during the execution of the transaction
Redis transactions have the characteristics of consistency, sequence and exclusivity
— Queue set set Set Run —-
Redis transactions have no concept of isolation levels
All commands in the transaction are not executed directly, and only when the execution command is initiated will the exec be executed
Redis single commands are guaranteed atomicity, but transactions are not
1. Redis transaction order
-
Start transaction (MULTI)
-
Order in (…)
-
Execute transaction (Exec)
2. Normal transaction execution
127.0.0.1:6379> set k1 v1 QUEUED 127.0.0.1:6379> set k2 v2 QUEUED 127.0.0.1:6379> Get k1 QUEUED 127.0.0.1:6379> set k3 v3 QUEUED 127.0.0.1:6379> EXEC # EXEC 1) OK 2) OK 3) OK 4) OK 127.0.0.1:6379>Copy the code
3. Abort transaction execution
127.0.0.1:6379> MULTI OK 127.0.0.1:6379> set k1 v1 QUEUED 127.0.0.1:6379> set k2 v2 QUEUED 127.0.0.1:6379> get k2 QUEUED 127.0.0.1:6379> DISCARDCopy the code
None of the commands will be executed
127.0.0.1:6379> MULTI OK 127.0.0.1:6379> set K1 v1 QUEUED 127.0.0.1:6379> set k2 v2 QUEUED 127.0.0.1:6379> set K3 v3 QUEUED 127.0.0.1:6379> get k1 k2 k3 # Error ERR wrong number of arguments for 'get' command 127.0.0.1:6379> get K1 QUEUED 127.0.0.1:6379> get k2 QUEUED 127.0.0.1:6379> set k4 v4 QUEUED 127.0.0.1:6379> exec # (error) EXECABORT Transaction discarded because of previous errors.Copy the code
Run time exception (1/0) If there is syntax in the transaction queue, other commands can be executed normally when the command is executed, but the error command throws an exception
127.0.0.1:6379> set k1 "v1" OK 127.0.0.1:6379> multi OK 127.0.0.1:6379> incr k1 # QUEUED 127.0.0.1:6379> set k2 v2 QUEUED 127.0.0.1:6379> set K3 v3 QUEUED 127.0.0.1:6379> get K3 QUEUED 127.0.0.1:6379> Exec 1) (error) ERR value is not an integer or out of range 2) OK 3) OK 4) "v3" 127.0.0.1:6379> get k2 "v2" 127.0.0.1:6379> get k3 "v3"Copy the code
8. Monitor (watch) optimistic and pessimistic locks
Pessimistic locking
Pessimistic locking, thinking that everything will go wrong at any time, and will be locked no matter what you do
Optimistic locking
-
I don’t lock it because I think it’s never going to get rid of the problem, but when I update the data it’s going to say, has anyone changed this process in the meantime
-
Access to the version
-
Compare version when updating
Redis monitors tests
Normal execution
127.0.0.1:6379> set money 100 OK 127.0.0.1:6379> set out 0 OK 127.0.0.1:6379> watch money # 127.0.0.1:6379> multi # The transaction ends without any changes to the data. OK 127.0.0.1:6379> DECRBY money 20 QUEUED 127.0.0.1:6379> INCRBY out 20 QUEUED 127.0.0.1:6379> EXEC 1) (integer) 80 2) (integer) 20 127.0.0.1:6379>Copy the code
Test multithreaded modification value, using Watch can be used as redis optimistic lock operation!
127.0.0.1:6379> watch money # Monitor money OK 127.0.0.1:6379> multi OK 127.0.0.1:6379> DECRBY money 10 QUEUED 127.0.0.1:6379> INCRBY out 10 QUEUED 127.0.0.1:6379> exec # (nil)Copy the code
If the modification fails, just get the latest value
9 Jedis.
1. What is Jedis
Jedis is an official Redis recommended Java connection development tool that uses Java to operate on Redis middleware.
1, test,
- Import the corresponding Maven dependencies
<! -- Import jedis package -->
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>3.2.0</version><! -- The version number can be filled in according to actual situation -->
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.62</version>
</dependency>
Copy the code
2. Coding tests
-
Connecting to a Database
-
Operation command
-
disconnect
public class TestPing {
public static void main(String[] args) {
// 1, new Jedis object
Jedis jedis = new Jedis("127.0.0.1".6379);
// All jedis commands are all the commands we learned before! So prior instruction learning is very important!System.out.println(jedis.ping()); }}Copy the code
2. Common apis
Stirng, list, set, hash, zset
The transaction
public class testJedis {
public static void main(String[] args) {
// Import the jedis package
Jedis jedis = new Jedis("127.0.0.1".6379);
/ / jedis command
//1. Test whether the connection is successful
System.out.println( jedis.ping());
JSONObject jsonObject=new JSONObject();
jsonObject.put("hello"."world");
jsonObject.put("hello1"."world1");
String result = jsonObject.toJSONString();
// Start the transaction
Transaction multi = jedis.multi();
try {
multi.set("user1",result);
multi.set("user2",result);
multi.set("user3",result);
multi.exec();
}catch (Exception e){
// Discard the transaction
multi.discard();
e.printStackTrace();
}finally {
System.out.println(jedis.get("user1"));
System.out.println(jedis.get("user2"));
// Close the connectionjedis.close(); }}}Copy the code
10. Springboot integration
SpringBoot operation data: Spring-data JPA JDBC mongodb, redis, etc
SprinData and SpringBoot
Note: After springboot2.x, the original jedis is replaced with the lettuce!
Jedis: The bottom layer uses direct connection, multiple threads operate jedis, is not safe, if you want to avoid unsafe threads, you should use jedis pool connection pool! Equivalent to BIO mode
Lettuce: With netty connection, instances can be shared between multiple threads, no thread insecurity! Equivalent to NIO mode
Integration testing
1. Source code analysis
@Bean
@ConditionalOnMissingBean(name = "redisTemplate") // We can define one ourselvesRedisTemplate to replace the default!public RedisTemplate redisTemplate(RedisConnectionFactory redisConnectionFactory)
throws UnknownHostException {
// The default RedisTemplate does not have too many Settings, redis objects are serialized!
// Both generics are Object types, which need to be cast later
RedisTemplate template = new RedisTemplate<>();
template.setConnectionFactory(redisConnectionFactory);
return template;
}
@Bean
@ConditionalOnMissingBean // Since String is the most commonly used type in Redis, it is a separate typeA bean!public StringRedisTemplate stringRedisTemplate(RedisConnectionFactory redisConnectionFactory)
throws UnknownHostException {
StringRedisTemplate template = new StringRedisTemplate();
template.setConnectionFactory(redisConnectionFactory);
return template;
}
Copy the code
2. Import dependencies
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
Copy the code
3. Configure the connection
spring:
redis:
host: 127.0. 01.
port: 6379
Copy the code
4, test,
@SpringBootTest
class Redis02SpringbootApplicationTests {
@Autowired
private RedisTemplate redisTemplate;
@Test
void contextLoads(a) {
// redisTemplate operates on different data types. The API is the same as our instructions
// opsForValue The operation String is similar to String
// opsForList operates like List
// opsForSet
// opsForHash
// opsForZSet
// opsForGeo
// opsForHyperLogLog
// In addition to the import operation, we can use the redisTemplate operation directly, such as transactions, and basic
CRUD
// Get the connection object of redis
// RedisConnection connection =
redisTemplate.getConnectionFactory().getConnection();
// connection.flushDb();
// connection.flushAll();
redisTemplate.opsForValue().set("mykey"."Attention crazy god said public account");
System.out.println(redisTemplate.opsForValue().get("mykey")); }}Copy the code
About saving objects
5. Customize RedisTemplate
@Configuration
public class RedisConfig {
@Bean
public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) throws UnknownHostException {
RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
redisTemplate.setConnectionFactory(redisConnectionFactory);
/ / json serialization
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
/ / string serialization
// Serialization of String
StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
// Key uses String serialization
redisTemplate.setKeySerializer(stringRedisSerializer);
// The hash key also uses String serialization
redisTemplate.setHashKeySerializer(stringRedisSerializer);
// Value serialization uses Jackson
redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
// The hash value serialization method uses Jackson
redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);
redisTemplate.afterPropertiesSet();
returnredisTemplate; }}Copy the code
6. General Redis tool class
7, summary
All redis operations are very simple for developers, but it is more important to understand the redis philosophy and the uses and scenarios of each data structure
11. Redis. conf configuration in redis
When Redis starts, it starts from a configuration file
1. Unit:
1. Configuration file unit Unit is case-insensitive (case insensitive)
2, contains
Same as import and include in Java
3, the network
Bind 127.0.0.1 # protected-mode yes # port 6379 # port SettingsCopy the code
4. General
Daemonize yes # Daemonize yes # Daemonize yes # Daemonize yes # Pidfile /var/run/redis_6379.pid Specify server verbosity level. # This can be one of: # debug (a lot of information, useful for development/testing) # verbose (many rarely useful info, but not a mess like the debug level) # notice (moderately verbose, What you want in production probably) Production environment # Warning (only very important/critical messages are logged) loglevel Notice logfile "" # databases 1 # Number of databases 2 # always-show-logo yes # Check whether the logo is displayedCopy the code
5, the snapshot
Persisting, how many operations are performed at a specified time, will persist to a file,.rdb.aof
Redis is an in-memory database. If it is not persisted, the data will be lost to power
Save 900 1 # If at least 1 key has been changed in 900 seconds, save 300 1 # If at least 10 key has been changed in 300 seconds, save 300 10 # If in 60 seconds, If at least 10000 key changes are made, we will persist the test and save 60 10000. Stop-writing-on-bgsave-error yes # Rdbcompression yes # RDBcompression yes # rDBcompression yes # RDBcompression Rdbchecksum yes # select * from rdbchecksum; Dir./ # RDB file save directory!Copy the code
6. REPLICATION
7
127.0.0.1:6379> ping PONG 127.0.0.1:6379> config get requirepass # Config set requirepass "123456" # set redis password OK 127.0.0.1:6379> config get requirepass # error NOAUTH Authentication required. 127.0.0.1:6379> ping (error) NOAUTH Authentication required. 127.0.0.1:6379> auth 123456 # Log in with your password! OK 127.0.0.1:6379> config get requirepass 1) "requirepass" 2) "123456"Copy the code
8. Clients limits
Maxclients 10000 # Specifies the maximum number of clients that can connect to Redis. Maxmemory # Redis specifies the maximum memory capacity. Maxmemory-policy noeviction # specifies the processing policy for memory Max Volatile - lRU: lRU only for keys whose expiration time is set (default value) 2. Allkeys-lru: delete keys of lRU algorithm 3. Volatile -random: delete keys that are about to expire randomly 4. Volatile - TTL: delete items that are about to expire. Noeviction: never expire, return errorCopy the code
9. Append only mode AOF configuration
Appendonly no # do not enable aof mode by default, persist using RDB, RDB is sufficient in most cases! Appendfilename "appendonly. Aof "# the name of the persistent file # appendfsync always # Every change will be synchronized. Appendfsync everysec # Execute sync once per second and may lose data for 1s! # appendfsync no # do not perform sync, this is the fastest time for the operating system to synchronize data.Copy the code
Redis persistence
Redis is based on memory database, if you do not save the database state in memory to disk, then the server process exits, the server database state will also message, so Redis provides the function of persistence!
1, the RDB (Redis DataBase)
What is the RDB: in master/slave replication, the RDB is the standby on the slave machineCopy the code
Writes a snapshot of an in-memory data set to disk at a specified time interval. When it recovers, the snapshot file is read directly into memory.
Redis forks a separate subprocess for persistence, writing data to a temporary file and replacing it with a temporary file when the persistence process is complete. During the whole process, the main process does not perform any I/O operations. High performance is guaranteed. RDB method is more efficient than AOF method if large-scale data restoration is required and the integrity of data restoration is not very sensitive. Disadvantages of RDB: Data may be lost after the last persistence. The default RDB, however, does not need to be modified, but in some production environments, files can be backed up before modification
== The file saved by RDB is dump. RDB ==, which is configured in the snapshot of the configuration file
1. RDB trigger mechanism
- If the save rule is met, the RDB rule is automatically triggered
- The flushall command also triggers the RDB rule
- Exiting Redis also generates RDB files
- The backup automatically generates a dump.rdb
2, how to restore RDB file
Dump. RDB: Dump. RDB: dump. RDB: dump.
2. Check the location of the RDB
127.0.0.1:6379> config get dir 1) "dir" 2) "/usr/local/bin" # if there is a dumpCopy the code
3. Advantages of RDB
1. Suitable for large-scale data recovery
2. Low requirements on data integrity
4. Disadvantages of RDB
1, need a certain time interval process operation, if Redis unexpectedly down, the last modification of the data will not be
2. The fork process occupies some memory space
2, AOF (Append Only File)
What is aOF? Record all of our commands, history, and execute the file when you restore it. See the following figure for detailsCopy the code
Record every write operation in the form of log, record all instructions executed by Redis (read operation is not recorded), only append files but cannot rewrite files, Redis will read the file to rebuild data at the beginning of each startup. When redis is restarted, write instructions are executed from front to back according to the contents of the log file to complete data recovery
Aof holds the appendone.aof file
Note: AOF is not enabled by default, we need to configure it manually. We only need to change appendOnly to yes to enable aOF, which will take effect after the restart
Rewriting rule Specification
Aof defaults to unlimited appending of files, and files get bigger and bigger
If the aOF file is larger than 64MB, which is relatively large, fork a new process to rewrite the AOF file
Appendonly no # do not enable aof mode by default, persist using RDB, RDB is sufficient in most cases! Appendfilename "appendonly. Aof "# the name of the persistent file appendfsync always # Every change will be sync. Appendfsync everysec # Execute sync once per second and may lose data for 1s! Appendfsync no # do not perform sync, this is the fastest time for the operating system to synchronize data. Rewrite rewritten,Copy the code
Advantages:
1, every change is synchronized, the integrity of the file is better
2. If the data is synchronized once every second, one second of data may be lost
3, never synchronized, the highest efficiency
Disadvantages: 1, compared to the data file, AOF is far larger than RDB, repair speed is far less than RDB, 2, AOF is slower than RDB, so redis default configuration is RDB persistence
13. Extensions to RDB and AOF
1, RDB persistent way to within the specified time interval for a snapshot of your data storage way 2, AOF persistent record to the operation of the server to write every time, when the server restart to perform these commands to restore the original data, AOF command to Redis protocol additional save each write operation to the end of the file, Redis can also write AOF files in the background so that the size of AOF files is not too large. In this case, when Redis restarts, the AOF file will be loaded first to recover the original data. AOF files usually hold more complete data sets than RDB files do. RDB data is not real-time, and the server will only look for AOF files when both are used, so should only use AOF? The authors advise against it, as RDB is better for backing up databases (AOF is not easy to back up constantly), restarts quickly, and there are no potential AOF bugs left as a back-up measure. 5. Performance Suggestion Since RDB files are only used for backup purposes, it is recommended to persist RDB files only on Slave and backup them only once every 15 minutes. Only save 9001 is retained. If you Enable AOF, you will lose no more than two seconds of data in the worst case. The startup script is simpler and only loads your own AOF files. The second is the end of AOF rewrite. The blocking of writing new data to new files in the rewrite process is almost inevitable. The default base size of AOF rewrite (64M) should be set to more than 5G. The default base size of AOF rewrite should be set to more than 100% of the original size. If AOF is not enabled, high availability can be achieved by master-slave Repllcation alone, saving a lot of I/O and reducing the amount of system volatility that rewrite might cause. The price is that if the Master/Slave is dropped at the same time, the data will be lost for more than ten minutes. The startup script also compares the RDB files in the two Master/Slave files and loads the newer one.Copy the code
14. Redis publishes subscriptions
Redis publishing schedule (PUB/SUB) is a == message communication mode == : the sender (PUB) sends the message and the subscriber (sub) receives the message. Suitable for wechat, Weibo, attention, etc
==Redis clients can subscribe to any number of channels ==
First message sender, second: channel, third message subscriber
The subscribe/publish message graph looks like this:
The following figure shows channel Channel1 and the relationship between the three clients that subscribe to this channel — Client2, Client5, and Client1:When a new message is sent to channel Channel1 via PUBLISH, the message is sent to the three clients that subscribed to it:
Redis commandCopy the code
testCopy the code
Subscription period:
[root@iZbp13u71xuet7lb86xtzhZ redisConfig]# redis-cli -p 6379 127.0.0.1:6379> SUBSCRIBE test # SUBSCRIBE a channel test Reading messages... (press ctrl-c to quit) 1) "subscribe" 2) "test" 3) (integer) 1) "message" # 2) "test" # 3) "hello" 1) "message" 2) "test" 3) "hello"Copy the code
The sender:
127.0.0.1:6379> PUBLISH test hello1 (integer) 1 127.0.0.1:6379> PUBLISH test hello1 (integer) 1 127.0.0.1:6379 >Copy the code
The principle of
Redis is realized by C. By analyzing the pubsub. C file in Redis source code, we can understand the underlying implementation of the publishing and subscription mechanism and deepen our understanding of Redis. Redis implements publish and subscribe functions through commands such as publish, subscribe and psubscribeCopy the code
WeChat:
1, Redis-server maintains a dictionary after subscribing to a channel with the SUBSCRIBE command. The key of the dictionary is a channel! , and the dictionary value is a linked list of all clients that subscribe to the channel. The key to the SUBSCRIBE command is to add a client to the subscription list for a given channel. 2. PUBLISH a message to subscribers. Using a given channel as a key, redis-server looks up the list of all clients that subscribe to the channel in the channel dictionary it maintains, iterates through the list, and publishes the message to all subscribers. Pub/Sub: Publish and Subscribe. In Redis, you can Publish and Subscribe messages to a key value. When a key value is published, all clients subscribing to it will receive corresponding messages. The most obvious use of this feature is as a real-time messaging system, such as normal im, group chat, etc.Copy the code
Redis usage scenarios
Real time messaging system 2. Fact chat (channel as chat room, displaying information to everyone) 3. Subscription and following systems can be used for more complex scenarios we use messaging middleware (MQ)Copy the code
Redis primary/secondary replication
Concept:Copy the code
== Primary/secondary replication refers to the replication of data from one Redis server to other Redis databases. The former is the master/leader node, and the latter is the slave/folloer node ==
== Data replication is one-way and can only be performed from the primary node to the secondary node. By default, each Redis server is the master node. A master node can have multiple slave nodes, but a slave node can only have one master node ==
The role of master-slave replication
1. Data redundancy: Master/slave replication implements data backup, which is a data redundancy mode other than persistence. 2. In effect, this is redundancy of a service. 3. Load balancing: on the basis of master/slave replication, with read/write separation, the master node can provide write service, and the slave node can provide read service (that is, when Redis data is written, the application connects to the master node and when Redis data is read, the application connects to the slave node) to share the load of the server. Especially in the scenario of less write and more read, the concurrency of the Redis server can be greatly increased by sharing the read load with multiple slave nodes. High availability (clustering) cornerstone: In addition to the above functions, master-slave replication is the basis on which sentry and clustering can be implemented, so master-slave replication is the foundation of High availability of Redis.Copy the code
Each project does not know the reason for a Redis server
1. Structurally, a single Redis server will have a single point of failure, and one server needs to handle all the request loads, resulting in great pressure; 2, from the capacity, a single Redis server memory capacity is limited, even if a Redis server memory capacity of 256G, also can not use all memory as Redis storage memory, generally speaking, a single Redis maximum use of memory should not exceed 20G. Included: the goods on the e-commerce website are generally uploaded once and viewed for countless times, which is also "read more and write less".Copy the code
The master/slave replication diagram is shown below
== Primary/secondary replication, read/write separation, to ease the server pressure, often use a primary/secondary ==
1. Environment configuration
==redis only configures the slave library, not the master library, because: redis default master library ==
127.0.0.1:6379> info replication # Check the current library information # Replication role:master # Role connected_SLAVES :0 # No slave Master_replid: 1 d5fcded821f63a2f5633551b5a411844b4d9a6a 127.0.0.1:6379 >Copy the code
Copy the three configuration files and modify the corresponding information
Pid file 3. Name of log file 4. Name of dumpCopy the code
After the completion of the start, can use ps – ef | grep redis command to see whether the start-up success
One master and two subordinate
== By default, each Redis service is a host. Generally, only the slave server is configured
One master (79) and two slave (80,81)
127.0.0.1:6380> SLAVEOF 127.0.0.1 6379 # Set host address and port OK 127.0.0.1:6380> info replication # replication role:slave # Master_port :6379 master_link_status:up master_last_io_seconds_ago:1 master_sync_in_progress:0 slave_repl_offset:14 slave_priority:100 slave_read_only:1 connected_slaves:0 master_replid:4f5c321e9fe398f47fd90f58a50782a464eff1fa master_replid2:0000000000000000000000000000000000000000 master_repl_offset:14 second_repl_offset:-1 repl_backlog_active:1 repl_backlog_size:1048576 repl_backlog_first_byte_offset:1 repl_backlog_histlen:14Copy the code
The host query
127.0.0.1:6379> Info replication # Replication role: Master Connected_Slaves: Slave0: IP =127.0.0.1,port=6380,state=online,offset=28,lag=0 master_replid:4f5c321e9fe398f47fd90f58a50782a464eff1fa master_replid2:0000000000000000000000000000000000000000 master_repl_offset:28 second_repl_offset:-1 repl_backlog_active:1 repl_backlog_size:1048576 repl_backlog_first_byte_offset:1 repl_backlog_histlen:28Copy the code
== The master/slave configuration should be configured in the configuration file so that it is permanent
Note:
The host can write data, but the slave machine can only read data. All information and data on the host are automatically saved by the slave machineCopy the code
Host:
From the machine
Test: host disconnected, from the machine is still connected to the host, but there is no write operation, this time, the host if back, from the machine can still directly get the host write information! If it is the use of command line, to configure the master and slave, this time if the restart, will change back to the host! As soon as it becomes slave, it will get the value from the host!Copy the code
Replication principle:
After connecting to the master, the Slave sends a sync command. The master receives the command to start the background saving process and collects all the commands received for modifying the data set. After the background process is complete, the master sends the entire data file to the Slave. And complete a full synchronization. Full replication: After receiving the database file data, the slave service saves it and loads it to the memory. Incremental replication: The Master continues to send all the new changes to the slaves in turn to complete the synchronization, but as soon as the Master is reconnected, a full synchronization (full replication) is automatically performed! Our data must be visible from the machine!Copy the code
Layer upon layer link
The last M links to the next S,
The specific process is shown in the figure
If the host is disconnected, we can use SLAVEOF No One to make ourselves a host! Other nodes can be manually connected to the latest master node (manually)! If the boss fixes it at this point, reconnect! = =
3. Sentinel mode
An overview of theCopy the code
The method of the master-slave switchover technology is: when the master server is down, you need to manually switch a slave server to the master server, which requires manual intervention, laborious, and will cause service unavailability for a period of time. This is not a recommended approach, and more often than not, we prefer sentinel mode. Redis has officially provided Sentinel architecture since 2.8 to solve this problem.
== Sentinel mode mainly monitors whether the host is faulty in the background. If the host is faulty, the slave library will be automatically converted to the master library according to the number of votes
Sentry mode is a special mode. First, Redis provides the command of sentry, which is a process and can run independently as a process.
== the sentry monitors multiple instances of Redis running == by sending commands and waiting for the Redis server to respond
The sentinel here serves two purposes - by sending commands to the Redis server back to monitor its running status, both master and slave. - When the sentry detects that the master is down, it will automatically switch the slave to master and notify the other slave servers through the ** publish/subscribe mode ** to modify the configuration file and let them switch hosts. However, one sentry process monitoring the Redis server can cause problems, so we can use multiple sentries to monitor. Each sentry is also monitored, creating a multi-sentry mode.Copy the code
Describe the process of failover in words. If the primary server is down and Sentry 1 detects this result first, the system does not perform a failover immediately, but sentry 1 thinks that the primary server is unavailable, and this phenomenon becomes a subjective offline. When the sentries at the back also detect that the primary server is unavailable and the number reaches a certain value, a vote is conducted between the sentries. The result of the vote is initiated by one sentry and failover is performed. After the switch is successful, each sentinel will switch the host from the secondary server monitored by himself through the publish and subscribe mode, which is called objective offline process. This makes everything transparent to the client.Copy the code
== Multiple sentry modes also listen on each other ==
1. Redis configures sentinel mode
Service type | Is it the primary server? | The IP address | port |
---|---|---|---|
Redis | is | 127.0.0.1 | 6379 |
Redis | no | 120.0.0.1 | 6380 |
Redis | no | 127.0.01 | 6381 |
Sentinel | 127.0.0.1 | ||
Sentinel | 127.0.0.1 | ||
Sentinel | 127.0.0.1 |
Testing:
1. Configure the sentinel configuration file sentinel.conf
# sentinel monitor host port 1 sentinel monitor myredis 127.0.0.1 6379 1 Will become the host!Copy the code
2. Start the sentry
[root @ bin] # redis - sentinel redisConfig/sentinel config # start the guard mode 23006: X 17 Apr 2021 16:26:21. 398 # oO0OoO0OoO0Oo redis Is starting oo0oo0o0oo0o023006 :X 17 Apr 2021 16:26:21.398 # Redis version=5.0.8, bits=64, commit=00000000, modified=0, pid=23006, Just started # redis version of 23006: X 17 Apr 2021 16:26:21. 398 # Configuration. The loaded _ _ _. - ` ` __ '- _ _. - ` ` `. ` _.' '- _ redis 5.0.8 (00000000/0), 64 - bit. - ` `. - ` ` `. ` ` ` \ / _, _ '-. _ ('. - ` | `,) Running in sentinel mode | ` - _ ` -... - ` __... -.``-._|'` _.-'| Port: 26379 | `-._ `._ / _.-' | PID: 23006 `-._ `-._ `-./ _.-' _.-' |`-._`-._ `-.__.-' _.-'_.-'| | `-._`-._ _.-'_.-' | http://redis.io `-._ `-._`-.__.-'_.-' _. - '| ` - _ ` - _ ` - __. -' _. - '_. -' | | ` - _ ` - _ _. - '_. -' | ` - _ ` - _ ` - __. - '_. -' _. - '` - _ ` -. __. -' _. - '` -) _ _) -' ` - __. - ' 23006:X 17 Apr 2021 16:26:21.399 # WARNING: The TCP backlog setting of 511 cannot be enforced because /proc/sys/net/core/somaxconn is set to the lower value of 128. 23006: X 17 Apr 2021 16:26:21. 400 # 3 d1613fd46f08691944287af3fec83af1a75853a Sentinel ID is: 23006 X 17 Apr 2021 16:26:21.400 # +monitor master myredis 127.0.0.1 6379 quorum 1 # host 23006:X 17 Apr 2021 16:26:21.401 * +slave slave 127.0.0.1:6380 127.0.0.1 6380 @myredis 127.0.0.1 6379 # from machine 1 23006:X 17 Apr 2021 16:26:21.402 * +slave 127.0.0.1:6381 127.0.0.1 6381 @myredis 127.0.0.1 6379 # slave 2Copy the code
If the Master node is disconnected, a server is randomly selected from the Master node
The guard log
== If the host returns, it can only merge to the new host, when slave, this is the sentinel rule ==
3. Advantages and disadvantages of sentinel mode
Advantages:
1. Sentinel cluster, based on the master-slave replication mode, all the advantages of the master-slave configuration, sentry also has 2. The master and slave can be switched, the fault can be transferred, the system availability will be better, high availability 3Copy the code
Disadvantages:
1. It is difficult to expand the capacity of redis online. Once the cluster capacity reaches online, it is very troublesome to expand the capacity onlineCopy the code
4, Sentry default all configuration
# Example sentinel.conf # Port 26379 port 26379 # Working directory dir/TMP # IP of the redis primary node monitored by Sentinel sentinel Port # master-name Specifies the name of the master node that can be named by itself. The name consists of only letters A-z, digits 0-9, and ".-_". # sentinel monitor <master-name> < IP > <redis-port> <quorum> Sentinel Monitor myMaster 127.0.0.1 6379 2 # When requirePass Foobared is enabled in the Redis instance, all clients connected to the Redis instance must provide the password # Sentinel auth-pass <master-name> <password> Sentinel auth-pass mymaster MySUPER--secret-0123passw0rd # specifies how many milliseconds after the primary node does not respond to sentinel <master-name> <milliseconds> sentinel down-after-milliseconds mymaster 30000 # This configuration item specifies the maximum number of slaves that can synchronize the new master at the same time when a failover occurs. The smaller the number is, the longer it takes to complete the failover. However, if the number is larger, This means that the more slaves become unavailable due to Replication. Setting this value to 1 ensures that only one slave at a time is unable to process command requests. # sentinel parallel-syncs <master-name> < numSlaves > Sentinel parallel-syncs mymaster 1 It can be used in the following ways: The interval between two failover operations for the same sentinel and the same master. #2. Start time counting when a slave synchronizes data from an incorrect master. Until the slave is corrected to synchronize data to the correct master. #3. The time required to cancel an ongoing failover. #4. Maximum time required to configure all Slaves to point to the new Master when performing failover. However, even after this timeout, slaves will still be configured correctly to point to the Master, [info] [info] [info] [info] [info] [info] [info] [info] [info Mymaster 180000 # SCRIPTS EXECUTION # Configure the SCRIPTS that need to be executed when an event occurs. You can use SCRIPTS to notify administrators, for example, by sending emails when the system is not working properly. # If the script returns 1 after execution, the script will be executed again later. The current default is 10. # If the script returns 2 after execution, or a value higher than 2, the script will not be executed again. If the script is aborted during execution due to a system interrupt signal, it behaves the same as if the value is 1. The maximum execution time of a script is 60 seconds. If this time is exceeded, the script will be terminated by a SIGKILL signal and executed again. # Notification script: This script will be called when sentinel has any warning level events (such as subjective and objective failures of redis instances, etc.). In this case, this script should be used to notify the system administrator of abnormal system performance via email, SMS, etc. When the script is called, it is passed two parameters, the type of the event and the description of the event. If the script path is configured in the sentinel.conf configuration file, the script must exist in the path and be executable, otherwise sentinel cannot start successfully. Sentinel notification-script <master-name> <script-path> sentinel notification-script mymaster When a master changes due to a failover, this script will be called to inform the client that the master address has changed. The following arguments will be passed to the script when it is called: # <master-name> <role> <state> <from-ip> <from-port> <to-ip> <to-port> # <role> is either "leader" or "observer". # from-ip, from-port, to-ip, to-port are used to communicate with the old master and the new master(i.e., the old slave) # This script should be generic and can be called multiple times, not specific. # sentinel client-reconfig-script <master-name> <script-path> sentinel client-reconfig-script mymaster /var/redis/reconfig.shCopy the code
16. Redis cache penetration and avalanche (common)
The use of Redis cache greatly improves the performance and efficiency of applications, especially data query. But at the same time, it also brings some problems. Among them, the most crucial problem, it is the consistent problem of data, tell from strict sense, this problem does not have solution. If the consistency of the data is high, you can’t use caching. = =
Typical problems are cache penetration, cache avalanche, and cache breakdown. At present, the industry also has a more popular solution.
== If there is none in the cache, go back to the database and query for ==
1. Cache penetration
1. Concept:
Cache penetration is: the user wants to query a data and finds that the Redis in-memory database is not hit, that is, the cache is not hit, so the user queries to the persistent layer database, and finds that there is no hit, then the query fails. When the user visits the Redis in-memory database many times and does not hit, they request the persistent layer database. This puts a lot of pressure on the persistence layer database, which is the equivalent of cache penetrationCopy the code
2. Solutions
1. Bloom filter A bloom filter is a data structure that stores all possible query parameters in the hash format. If the parameters do not meet the requirements, they are discarded at the control layer, avoiding the query pressure on the underlying storage systemCopy the code
2, cache empty objects when the storage layer does not match, even the empty object will be cached, and set an expiration time, after accessing the data will be obtained from the cache, protecting the backend data source, but this method will have two problems: 1. If null values can be cached, this means that the cache needs more space to store more keys, because there may be a lot of null keys; 2. Even if the expiration time is set for null values, data at the cache layer and storage layer will be inconsistent for a period of time, which will affect services that need to maintain consistency.Copy the code
2. Cache breakdown (heavy traffic, cache expiration)
1, concept,
Cache breakdown refers to a key is very hot, in the continuous carrying large concurrency, large concurrency focused on this point to access, when the key in the moment of failure, continuous large concurrency will break through the cache, directly request the database, just like in a barrier to cut a hole. When a key expires, a large number of concurrent requests are generated. Such data is usually hot data. Due to the expiration of the cache, the database is accessed to query the latest data and the cache is written back.Copy the code
2. Solutions
1. Set the hotspot data to never expire
From the cache level, the expiration time is not set, so there are no problems caused by hot key expiration.Copy the code
2. Add mutex
Distributed lock: The distributed lock ensures that only one thread queries back-end services for each key. Other threads do not have the permission to obtain the distributed lock, so they only need to wait. This approach shifts the pressure of high concurrency to distributed locks, so the challenge for distributed locks is great.Copy the code
Cache avalanche
1, concept,
A cache avalanche is when a cache set expires at some point in time. Redis downtime! One of the reasons for the avalanche, for example, at the time of writing this article, it will soon be double Twelve zero, a wave of buying, this wave of goods in a relatively concentrated time cache, let's say one hour cache. So by one o 'clock in the morning, the cache of these items will have expired. And the access to this batch of goods, all fell on the database, for the database, will produce periodic pressure peaks. Then all the requests will reach the storage layer, and the amount of calls to the storage layer will explode, causing the storage layer to also fail. In fact, centralized expiration is not so deadly, but a more deadly cache avalanche is when a node of the cache server goes down or is disconnected. Because of the natural cache avalanche, the cache must be created at a time when the database can withstand the pressure. It's just a periodic strain on the database. The breakdown of the cache service node, the pressure on the database server is unpredictable, it is likely to crush the database in an instant.Copy the code
2. Solutions
1. Redis is highly available
This idea means that since redis may fail, I will add more Redis, so that the others can continue to work after the failure of one, which is actually a cluster built. (Live long distance!)Copy the code
2. Current limiting degradation
The idea behind this solution is to control the number of threads that read the database write cache by locking or queuing after the cache is invalidated. For example, only one thread is allowed to query data and write to the cache for a key, while the other threads wait.Copy the code
3. Data preheating
The idea of data heating is that I pre-access all possible data prior to deployment, so that some of the potentially heavily accessed data is loaded into the cache. Before large concurrent access occurs, manually trigger the loading of different cache keys, set different expiration times, and make the cache failure time as even as possible.Copy the code