preface
In the previous article, I shared with you the use of locks in a single architecture through the example of seckilling in the e-marketplace. However, many applications today are quite large, and many are microservice architectures, so how can we solve concurrency in such a cross-JVM scenario?
Limitations of single application locks
Before entering the actual combat, briefly talk with you about the architecture evolution of the Internet system.
At the beginning of the development of the Internet system, the consumption of resources and the number of users were relatively small, so we could only deploy one Tomcat application to meet the needs. A Tomcat can be thought of as a JVM process. When a large number of concurrent requests arrive in the system, all the requests fall on a single Tomcat. If some request methods need to be locked, such as the second kill inventory scenario mentioned in the previous article, this can be satisfied. However, as the number of visits increased, it was difficult to support one Tomcat. At this time, we needed to deploy Tomcat in a cluster and use multiple Tomcats to support the system.
After the simple evolution shown in the figure above, we deployed two Tomcat to support the system together. When a request arrives on the system, it first passes through NGINx, which acts as a load balancer and forwards the request to one of the Tomcat servers based on its own load balancing configuration policy. When a large number of requests are accessed concurrently, the two Tomcats share all the traffic. After that, when we also use the single application lock for the second kill inventory reduction, can we still meet the demand?
This type of lock works on a single JVM. When there are two or more tomcat requests, a large number of concurrent requests are distributed to different Tomcats. concurrency can be prevented in each Tomcat, but in multiple Tomcats, the lock request is obtained in each Tomcat, and concurrency is generated. Thus the problem of inventory deduction still exists. This is the limitation of singleton locking. So what if we solve this problem? Now I’m going to share distributed locks with you.
A distributed lock
What is distributed locking?
What is distributed locking? Before we talk about distributed locking, we saw that singleton application locks are valid within a JVM, but not across JVMS or processes. So a less official definition, a distributed lock is a lock that spans multiple JVMS, spans multiple processes, and locks like this are distributed locks.
Design ideas
Since Tomcat is started in Java, each Tomcat can be treated as a SINGLE JVM, and locks within the JVM cannot span multiple processes. So we implement distributed locks, we have to look outside these JVMS and implement distributed locks through other components.
The two Tomcats above implement distributed locking across JVMS and processes using third-party components. This is the idea of distributed locking.
implementation
So what third-party components are currently available to implement this? Currently more popular are the following:
- Database, through the database can achieve distributed lock, but the high concurrency of the situation on the database pressure is relatively large, so rarely used.
- Redis, with the help of Redis can achieve distributed locks, and Redis Java client types, so the use of the method is not the same.
- Zookeeper can also implement distributed locking, and ZK also has many Java clients, which are used in different ways.
For the above implementation, the old cat or through specific code examples to demonstrate one by one.
Database based distributed locking
Database pessimistic lock to achieve distributed lock, with the main is select… For update. select … For update is to lock the data when the query is performed. When the user performs this behavior operation, other threads are forbidden to modify or delete the data, and must wait for the last thread to complete the operation after release, so as to achieve the lock effect.
Implementation: We are still based on the example of oversold e-commerce and share the code with you.
Let’s take a look at a new table called distribute_lock, which provides database locks.Since we are simulating an oversold order scenario, we have the lock data of an order in the figure above.
We will modify the code in the previous article to extract a Controller and request the invocation through Postman. Of course, in the background, we will start two JVMS for operation, namely port 8080 and port 8081. The finished code looks like this:
/ * * *@author [email protected]
* @date2021/1/3 he *@descPublic account "programmer old cat" */
@Service
@Slf4j
public class MySQLOrderService {
@Resource
private KdOrderMapper orderMapper;
@Resource
private KdOrderItemMapper orderItemMapper;
@Resource
private KdProductMapper productMapper;
@Resource
private DistributeLockMapper distributeLockMapper;
// Purchase item ID
private int purchaseProductId = 100100;
// Number of items purchased
private int purchaseProductNum = 1;
@Transactional(propagation = Propagation.REQUIRED)
public Integer createOrder(a) throws Exception{
log.info("Into the method.");
DistributeLock lock = distributeLockMapper.selectDistributeLock("order");
if(lock == null) throw new Exception("The service distributed lock is not configured");
log.info("Got the lock.");
// In order to demonstrate concurrency manually, we will hibernate here for 1 minute
Thread.sleep(60000);
KdProduct product = productMapper.selectByPrimaryKey(purchaseProductId);
if (product==null) {throw new Exception("Purchase goods:"+purchaseProductId+"Doesn't exist.");
}
// Current inventory of goods
Integer currentCount = product.getCount();
log.info(Thread.currentThread().getName()+"Inventory number"+currentCount);
// Check inventory
if (purchaseProductNum > currentCount){
throw new Exception("Goods"+purchaseProductId+"Only"+currentCount+"Piece, cannot be purchased.");
}
// Complete the decrement in the database
productMapper.updateProductCount(purchaseProductNum,"kd".new Date(),product.getId());
// Generate the order. The source code can be downloaded at Github.//github.com/maoba/kd-distribute
returnorder.getId(); }}Copy the code
SQL is written as follows:
select
*
from distribute_lock
where business_code = #{business_code,jdbcType=VARCHAR}
for update
Copy the code
The above is the main implementation logic, points to note in the code:
- The createOrder method must have a transaction because the select for Update lock can only be triggered if a transaction exists.
- The code must determine the existence of the current lock, if the case is empty, an exception will be reported
Let’s take a look at the final run, starting with the console log,
8080 console logs:
11:49:41 INFO 16360 --- [nio-8080-exec-2] c.k.d.service.MySQLOrderService : Entered the method 11:49:41 INFO - 16360 [nio - 8080 - exec - 2] C.K.D.S ervice. MySQLOrderService: got the lockCopy the code
Console logs of 8081:
11:49:48 INFO - 17640 [nio - 8081 - exec - 2] C.K.D.S ervice. MySQLOrderService: into the methodCopy the code
In the log case, two different JVMS, since the first request to 8080 received the lock first, the 8081 request is waiting for the lock to be released before being executed, indicating that our distributed lock is in effect.
Take a look at the log after the full execution:
8080 request:
11:58:01 INFO 15380 --- [nio-8080-exec-1] c.k.d.service.MySQLOrderService : Entered the method 11:58:01 INFO - 15380 [nio - 8080 - exec - 1] C.K.D.S ervice. MySQLOrderService: Got the lock 11:58:07 INFO - 15380 [nio - 8080 - exec - 1] C.K.D.S ervice. MySQLOrderService: HTTP - nio inventory number 1-8080 - exec - 1Copy the code
Request for 8081:
11:58:03 INFO 16276 --- [nio-8081-exec-1] c.k.d.service.MySQLOrderService : Entered the method 11:58:08 INFO - 16276 [nio - 8081 - exec - 1] C.K.D.S ervice. MySQLOrderService: Got the lock 11:58:14 INFO - 16276 [nio - 8081 - exec - 1] C.K.D.S ervice. MySQLOrderService: ERROR 16276 -- [NIo-8081-exec-1] O.A.C.C.C. [.[.[/].[dispatcherServlet] : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; Nested exception is java.lang. exception: Commodity 100100 only 0 pieces left, cannot be purchased] with root cause java.lang. exception: Goods only 0 100100, unable to buy the at com. The kd. Distribute. Service. MySQLOrderService. CreateOrder (MySQLOrderService. Java: 61) ~ / classes / : naCopy the code
Obviously, due to the lack of inventory in the second request, the final purchase failed. Of course, this scenario is also in line with our normal business scenario. Finally, our database looks like this:
It is clear that we have the correct inventory and order quantity to this database. To this we based on the database of distributed lock actual combat demonstration completed, we will sum up the following if the use of this lock, what advantages and disadvantages.
- Advantages: simple and convenient, easy to understand, easy to operate.
- Disadvantages: large amount of concurrency when the pressure on the database will be relatively large.
- Suggestion: Separate the lock database from the service database.
Write in the last
For the above database distributed lock, in fact, in our daily development with relatively little. The locks based on Redis and ZK are used more, originally the old cat wanted to share the redis lock and ZK lock in this article, but it would be too long to write in the same article, so this article will share with you this distributed lock. You can download the source code on github. The address is github.com/maoba/kd-di…