成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

高效的并發管理:房間預訂 API 的樂觀鎖和消息隊列

開發 前端
想象一下這樣一個場景:多名旅行者同時嘗試預訂熱門目的地的最后一個可用房間。如果沒有適當的并發控制機制,這種情況很快就會變成競爭狀態,導致房間超額預訂和客戶沮喪。

想象一下這樣一個場景:多名旅行者同時嘗試預訂熱門目的地的最后一個可用房間。如果沒有適當的并發控制機制,這種情況很快就會變成競爭狀態,導致房間超額預訂和客戶沮喪。

我們將深入研究用于應對這些挑戰的兩種關鍵策略的復雜性:樂觀鎖定和消息隊列。

想象一下您正在使用一個在線酒店預訂平臺,類似于 Booking.com 或 Expedia 等知名平臺。以下是同步和異步流程如何發揮作用:

同步流程:

預訂房間(同步):

  • 您訪問酒店預訂網站并選擇您的目的地、入住和退房日期以及其他偏好。
  • 您點擊“立即預訂”按鈕即可預訂房間。
  • 該網站使用基于 HTTP 的同步協議(如 REST 或 SOAP)將您的請求發送到酒店的預訂系統。
  • 酒店的系統會立即同步處理您的請求。它檢查房間可用性,為您預訂房間,并生成預訂號碼。
  • 預訂號碼將發送回您的瀏覽器,并在幾秒鐘內顯示在網站上。
  • 您可以立即獲得預訂號碼,然后可以放心地繼續您的旅行計劃。

創建房間實體

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;


@Entity
public class Room {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String roomType;
    private boolean isAvailable;


    // getters and setters
}

創建房間存儲庫

import org.springframework.data.jpa.repository.JpaRepository;


public interface RoomRepository extends JpaRepository<Room, Long> {
    Room findByRoomType(String roomType);
}

創建客房預訂請求 DTO

import java.time.LocalDate;


public class RoomBookingRequest {
    private String roomType;
    private LocalDate checkInDate;
    private LocalDate checkOutDate;


    // getters and setters
}

創建客房預訂響應 DTO

public class RoomBookingResponse {
    private String reservationNumber;


    // getters and setters
}

創建客房服務

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;


import java.util.UUID;


@Service
public class RoomService {


    @Autowired
    private RoomRepository roomRepository;


    public RoomBookingResponse bookRoom(RoomBookingRequest bookingRequest) {
        String roomType = bookingRequest.getRoomType();
        LocalDate checkInDate = bookingRequest.getCheckInDate();
        LocalDate checkOutDate = bookingRequest.getCheckOutDate();


        Room room = roomRepository.findByRoomType(roomType);


        if (room != null && room.isAvailable()) {
            // Add validation to check availability based on check-in and check-out dates here.


            // For simplicity, we'll assume the room is available.
            room.setAvailable(false);
            roomRepository.save(room);


            // Generate a reservation number (you can implement your logic here).
            String reservationNumber = generateReservationNumber();


            return new RoomBookingResponse(reservationNumber);
        } else {
            throw new RoomNotAvailableException();
        }
    }


    private String generateReservationNumber() {
        // Generate a unique reservation number (you can implement your logic here).
        return UUID.randomUUID().toString();
    }
}

創建房間控制器

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;


@RestController
@RequestMapping("/api/rooms")
public class RoomController {


    @Autowired
    private RoomService roomService;


    // Book a room
    @PostMapping("/book")
    public RoomBookingResponse bookRoom(@RequestBody RoomBookingRequest bookingRequest) {
        return roomService.bookRoom(bookingRequest);
    }
}

定義自定義異常

import org.springframework.http.HttpStatus;
import org.springframework.web.bind.annotation.ResponseStatus;


@ResponseStatus(HttpStatus.BAD_REQUEST)
public class RoomNotAvailableException extends RuntimeException {
    public RoomNotAvailableException() {
        super("The requested room is not available.");
    }
}

測試API

您可以使用 Postman 或 cURL 等工具來測試您的 API。要預訂房間,請http://localhost:8080/api/rooms/book使用包含房間類型、入住日期和退房日期的 JSON 正文發出 POST 請求:

{ 
  "roomType" :  "Standard" , 
  "checkInDate" :  "2023-10-01" , 
  "checkOutDate" :  "2023-10-05" 
}

如果房間可用,API 將返回帶有預訂編號的 JSON 響應。您可以根據您的課堂需求自定義預訂邏輯和預訂號碼生成RoomService。

異步流程

當多個用戶同時調用Booking API時

當多個并發呼叫在系統中搜索同一房間時,可能存在潛在的缺點和挑戰:

競爭條件:當多個請求嘗試同時預訂同一房間時,可能會出現競爭條件。如果處理不當,這可能會導致超額預訂,即系統允許的預訂數量超過了可用房間的數量。

如何解決并發問題?

樂觀鎖定是一種數據庫級技術,可防止多個用戶同時嘗試更新同一資源時發生數據沖突。

另一方面,消息隊列是異步通信工具,可確保請求的有序、可靠處理,使其成為分布式系統中處理并發請求的理想選擇。

方法一:實現消息隊列響應并發請求

消息隊列確保請求按照接收順序進行處理,從而防止競爭條件和超量預訂。

  • 個客戶端向端點發出 POST 請求/api/rooms/book以同時預訂酒店房間。
  • 處理RoomController傳入的預訂請求。
  • 該roomService.bookRoom方法接收預訂請求。
  • 它使用該方法將預訂請求發送到名為“room-booking”的 RabbitMQ 消息隊列rabbitTemplate.convertAndSend。
  • 它向客戶端返回初步響應,其中包含一條消息,表明預訂請求已發送,客戶端應等待確認。
  • 預訂請求被放入“房間預訂”隊列中。消息隊列系統(在本例中為 RabbitMQ)確保每個預訂請求都按照收到的順序進行處理,以防止競爭情況。
  • 監聽RoomBookingMessageConsumer“房間預訂”隊列。
  • processBookingRequest當預訂請求出隊時,將調用消費者的方法。在該方法中,您通常會實現以下邏輯:
  • 根據請求的房型、入住日期和退房日期檢查客房供應情況。
  • 如果房間可用,則生成預訂號碼。
  • 更新數據庫中的房間可用性,將其標記為不可用,以防止重復預訂。
  • 通過RabbitMQ向客戶端發送包含預約號的響應消息

8. 在 中RoomBookingMessageConsumer,處理預訂請求并生成預訂號碼后,您可以使用傳統的 HTTP 客戶端(例如RestTemplate、HttpClient)將確認響應直接發送到客戶端的回調 URL 端點(該端點在請求中發送)。

執行:

創建客房預訂請求和響應 DTO

import java.time.LocalDate;


public class RoomBookingRequest {
    private String roomType;
    private LocalDate checkInDate;
    private LocalDate checkOutDate;
    private String clientCallbackUrl; // Added to specify the client's callback URL


    // getters and setters
}


public class RoomBookingResponse {
    private String reservationNumber;


    // getters and setters
}

修改控制器

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;


@RestController
@RequestMapping("/api/rooms")
public class RoomController {


    @Autowired
    private RoomService roomService;


    @PostMapping("/book")
    public RoomBookingResponse bookRoom(@RequestBody RoomBookingRequest bookingRequest) {
        return roomService.bookRoom(bookingRequest);
    }
}

創建客房預訂服務(生產者)

import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.http.ResponseEntity;
import org.springframework.web.client.RestTemplate;


@Service
public class RoomService {


    @Autowired
    private RoomRepository roomRepository;


    @Autowired
    private RabbitTemplate rabbitTemplate;


    private RestTemplate restTemplate = new RestTemplate();


    public RoomBookingResponse bookRoom(RoomBookingRequest bookingRequest) {
        String roomType = bookingRequest.getRoomType();


        // Send the booking request to the message queue
        rabbitTemplate.convertAndSend("room-booking-exchange", "room-booking", bookingRequest);


        return new RoomBookingResponse("Booking request sent. Please wait for confirmation.");
    }


    // This method sends the response to the client's callback URL
    public void sendResponseToClient(RoomBookingResponse response, String clientCallbackUrl) {
        ResponseEntity<Void> result = restTemplate.postForEntity(clientCallbackUrl, response, Void.class);
        if (result.getStatusCode().is2xxSuccessful()) {
            // Handle a successful response sent to the client
        } else {
            // Handle the case when the response to the client failed
        }
    }
}

創建消息消費者

import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;


@Component
public class RoomBookingMessageConsumer {


    @Autowired
    private RoomService roomService;


    @RabbitListener(queues = "room-booking-queue")
    public void processBookingRequest(RoomBookingRequest bookingRequest) {
        // Process the booking request
        RoomBookingResponse response = processBookingLogic(bookingRequest);


        // Send the confirmation response to the client's callback URL
        roomService.sendResponseToClient(response, bookingRequest.getClientCallbackUrl());
    }


    private RoomBookingResponse processBookingLogic(RoomBookingRequest bookingRequest) {
        // Implement your booking logic here, e.g., checking room availability and generating a reservation number
        // Update room availability in the database
        // Send a response message to confirm the booking or indicate unavailability


        // For simplicity, we'll assume the room is available and generate a reservation number.
        String reservationNumber = generateReservationNumber();


        return new RoomBookingResponse(reservationNumber);
    }


    private String generateReservationNumber() {
        // Generate a unique reservation number (you can implement your logic here).
        return "RES-" + System.currentTimeMillis();
    }
}

方法二:實現樂觀鎖來處理并發請求

您可以修改代碼以使用同步方法和 JPA 樂觀鎖定。

步驟1:修改Room實體:@Version向實體添加一個字段Room以啟用樂觀鎖定:

import javax.persistence.*;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;


@Entity
public class Room {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String roomType;
    private boolean isAvailable;
    
    @Version
    private Long version;


    // getters and setters
}
步驟2:修改客房服務對每個房間使用ReentrantLock來同步訪問房間預訂操作
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;


import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;


@Service
public class RoomService {


    @Autowired
    private RoomRepository roomRepository;


    private final ConcurrentHashMap<Long, Lock> roomLocks = new ConcurrentHashMap<>();


    public RoomBookingResponse bookRoom(RoomBookingRequest bookingRequest) {
        String roomType = bookingRequest.getRoomType();
        LocalDate checkInDate = bookingRequest.getCheckInDate();
        LocalDate checkOutDate = bookingRequest.getCheckOutDate();


        Room room = roomRepository.findByRoomType(roomType);


        if (room != null) {
            Lock roomLock = roomLocks.computeIfAbsent(room.getId(), id -> new ReentrantLock());


            roomLock.lock();
            try {
                if (room.isAvailable()) {
                    // Add validation to check availability based on check-in and check-out dates here.


                    // For simplicity, we'll assume the room is available.
                    room.setAvailable(false);
                    roomRepository.save(room);


                    // Generate a reservation number (you can implement your logic here).
                    String reservationNumber = generateReservationNumber();


                    return new RoomBookingResponse(reservationNumber);
                }
            } finally {
                roomLock.unlock();
            }
        }


        throw new RoomNotAvailableException();
    }


    private String generateReservationNumber() {
        // Generate a unique reservation number (you can implement your logic here).
        return UUID.randomUUID().toString();
    }
}

詳細工作原理:

并發請求&ConcurrentHashMap:當同一房間收到多個并發預訂請求時,它們可能同時到達并可能導致競爭條件。的引入ConcurrentHashMap確保每個房間都有自己的鎖。這ConcurrentHashMap是一個線程安全的映射,可以由多個線程同時安全地訪問。

通過鎖定并發更新房間可用性:如果兩個線程同時嘗試預訂同一個房間,則只有其中一個線程會使用 成功獲取鎖roomLock.lock(),而另一個線程將暫時阻塞,直到第一個線程釋放鎖。

釋放鎖以供其他線程更新:一旦線程獲取了鎖并成功修改了房間的可用性,它就會使用 釋放鎖roomLock.unlock(),從而允許其他線程繼續預訂其他房間。

樂觀鎖防止數據庫級別的競爭條件:在代碼中,實體中的字段啟用數據庫級別的樂觀鎖。更新房間時,JPA 在允許更新之前會根據實體中的版本字段檢查數據庫中的版本字段。@VersionRoom

  • 如果兩個事務同時嘗試更新同一個房間,根據版本號的比較,只有其中一個會成功,從而防止數據庫級別的數據沖突。
  • 因此 2 個不同的事務無法同時更新數據庫中的一個房間
責任編輯:華軒 來源: HELLO程序員
相關推薦

2021-01-15 05:12:14

Java并發樂觀鎖

2023-10-13 00:00:00

并發樂觀鎖CAS

2023-07-05 08:18:54

Atomic類樂觀鎖悲觀鎖

2020-07-06 08:03:32

Java悲觀鎖樂觀鎖

2024-09-03 15:14:42

2025-04-23 08:45:00

悲觀鎖樂觀鎖并發控制機制

2024-01-29 01:08:01

悲觀鎖遞歸鎖讀寫鎖

2025-02-26 09:55:59

Linux內核并發

2023-11-07 10:01:34

2023-08-17 14:10:11

Java開發前端

2010-08-18 09:00:38

數據庫

2019-07-19 07:56:13

消息隊列消息代理消息中間件

2021-03-30 09:45:11

悲觀鎖樂觀鎖Optimistic

2024-05-16 08:10:17

RabbitMQ軟件通信機制

2016-11-28 14:40:00

MQTT消息協議

2022-09-22 11:36:31

隊列Python

2024-05-17 09:33:22

樂觀鎖CASversion

2009-09-24 14:43:53

Hibernate樂觀

2023-04-14 12:23:15

2024-07-25 09:01:22

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久青| h视频免费观看 | 婷婷久久一区 | 欧美 日韩 亚洲91麻豆精品 | 成人午夜电影在线观看 | 高清视频一区二区三区 | 国产精品一区二区日韩 | 一级做a爰片性色毛片16 | 日本不卡一区 | 国产露脸对白88av | 精产嫩模国品一二三区 | 黑人精品欧美一区二区蜜桃 | 国产日韩精品在线 | 久国产精品 | 久久精品国产免费高清 | 97精品超碰一区二区三区 | 国产精品一区在线观看你懂的 | 日本在线一二 | 亚洲午夜精品在线观看 | 国产成人精品综合 | a天堂在线 | 日本电影免费完整观看 | 精品亚洲一区二区 | 午夜视频精品 | 亚洲精品欧美精品 | 欧美一级网站 | 国产二区在线播放 | 欧美成人一区二区三区 | 久久精品欧美一区二区三区不卡 | 亚洲精品久久久久久下一站 | 成人免费视频一区 | 久久国产区 | 国产视频第一页 | 国家一级黄色片 | 99精品99久久久久久宅男 | 国产粉嫩尤物极品99综合精品 | 真人毛片 | 91久久国产综合久久91精品网站 | 国产9 9在线 | 中文 | 精品国产91亚洲一区二区三区www | 美女视频黄色片 |