溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

發(fā)布時間:2022-05-23 09:30:38 來源:億速云 閱讀:562 作者:zzz 欄目:開發(fā)技術

這篇文章主要介紹“Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)”,在日常操作中,相信很多人在Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

1 Scenario 場景

電商系統(tǒng)的促銷手段(Electronic Commerce Systems):

  • 優(yōu)惠券

  • 拼團

  • 砍價

  • 老帶新

優(yōu)惠券的種類:

  • 滿減券

  • 直減券

  • 折扣券

優(yōu)惠券系統(tǒng)的核心流程:

發(fā)券

發(fā)券的方式: 同步發(fā)送 or 異步發(fā)送

領券

  • 誰能領?

  • 所有用戶 or 指定的用戶

  • 領取上限

  • 一個優(yōu)惠券最多能領取多少張?

  • 領取方式

  • 用戶主動領取 or 自動發(fā)放被動領取

用券

  • 作用范圍

  • 商品、商戶、類目

  • 計算方式

  • 是否互斥、是否達到門檻等

需求拆解

商家側:

  • 創(chuàng)建優(yōu)惠券

  • 發(fā)送優(yōu)惠券

用戶側:

  • 領取優(yōu)惠券

  • 下單

  • 使用優(yōu)惠券

  • 支付

2 Service 服務

2.1 服務結構設計

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

2.2 優(yōu)惠券系統(tǒng)難點

券的分布式事務,使用券的過程會出現(xiàn)的分布式問題分析

如何防止超發(fā)

如何大批量給用戶發(fā)券

如何限制券的使用條件

如何防止用戶重復領券

3 Storage存儲

模型的設計

優(yōu)惠券系統(tǒng) Coupon System 模型定義

優(yōu)惠券系統(tǒng)的難點

3.1 表單設計

券批次(券模板),coupon_batch

指一批優(yōu)惠券的抽象、模板,包含優(yōu)惠券的大部分屬性。

如商家創(chuàng)建了一批優(yōu)惠券,共1000張,使用時間為2022-11-11 00:00:00 ~ 2022-11-11 23:59:59,規(guī)定只有數(shù)碼類目商品才能使用,滿100減50。

發(fā)放到用戶的一個實體,已與用戶綁定。

如將某批次的優(yōu)惠券中的一張發(fā)送給某個用戶,此時優(yōu)惠券屬于用戶。

規(guī)則

優(yōu)惠券的使用有規(guī)則和條件限制,比如滿100減50券,需要達到門檻金額100元才能使用。

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

券批次表 coupon_batch

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

規(guī)則表 rule:

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

規(guī)則內容:

 {
   threshold: 5.01 // 使用門檻
   amount: 5 // 優(yōu)惠金額
   use_range: 3 // 使用范圍,0—全場,1—商家,2—類別,3—商品 
   commodity_id: 10 // 商品 id
   receive_count: 1 // 每個用戶可以領取的數(shù)量
   is_mutex: true // 是否互斥,true 表示互斥,false 表示不互斥
   receive_started_at: 2020-11-1 00:08:00 // 領取開始時間
   receive_ended_at: 2020-11-6 00:08:00 // 領取結束時間
   use_started_at: 2020-11-1 00:00:00 // 使用開始時間
   use_ended_at: 2020-11-11 11:59:59 // 使用結束時間
 }

優(yōu)惠券表 coupon:

 create table t_coupon
 (
     coupon_id     int          null comment '券ID,主鍵',
     user_id       int          null comment '用戶ID',
     batch_id      int          null comment '批次ID',
     status        int          null comment '0-未使用、1-已使用、2-已過期、3-凍結',
     order_id      varchar(255) null comment '對應訂單ID',
     received_time datetime     null comment '領取時間',
     validat_time  datetime     null comment '有效日期',
     used_time     datetime     null comment '使用時間'
 );

3.2 優(yōu)惠券系統(tǒng)

建券:

1、新建規(guī)則

 INSERT INTO rule (name, type, rule_content)
 VALUES(“滿減規(guī)則”, 0, '{
                         threshold: 100
                         amount: 10
                         ......
                       }');

2、新建優(yōu)惠券批次

 INSERT INTO coupon_batch (coupon_name, rule_id, total_count ) 
 VALUES(“勞斯萊斯5元代金券”, 1010, 10000);

發(fā)券:

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

如何給大量用戶發(fā)券?

異步發(fā)送

觸達系統(tǒng)
  • 短信、郵件

  • 可通過調用第三方接口的方式實現(xiàn)

  • 站內信

  • 通過數(shù)據庫插入記錄來實現(xiàn)

信息表 message

 create table t_message
 (
     id         int null comment '信息ID',
     send_id    int null comment '發(fā)送者id',
     rec_id     int null comment '接受者id',
     content    vachar(255) comment '站內信內容',
     is_read    int null comment '是否已讀',
     send_time  datetime comment '發(fā)送時間'
 )
 comment '信息表';

先考慮用戶量很少的情況,商家要給所有人發(fā)站內信,則先遍歷用戶表,再按照用戶表中的所有用戶依次將站內信插入到 message 表中。這樣,如果有100個用戶,則群發(fā)一條站內信要執(zhí)行100個插入操作。

系統(tǒng)用戶數(shù)增加到萬級

發(fā)一條站內信,就得重復插入上萬條數(shù)據。而且這上萬條數(shù)據的 content 一樣!假設一條站內信占100K,發(fā)一次站內信就要消耗十幾M。對此,可將原來的表拆成兩個表:

信息表 message

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

信息內容表 message_content

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

發(fā)一封站內信的步驟
  • 往 message_content 插入站內信的內容

  • 在 message 表中,給所有用戶插入一條記錄,標識有一封站內信

千w級用戶數(shù)

這就有【非活躍用戶】的問題,假設注冊用戶一千萬,根據二八原則,其中活躍用戶占20%。若采用上面拆成兩個表的情況,發(fā)一封“站內信”,得執(zhí)行一千萬個插入操作??赡苁O?0%用戶基本都不會再登錄,其實只需對其中20%用戶插入數(shù)據。

信息表 message:

 create table t_message
 (
     id         int null comment '信息 ID',
     # send_id    int null comment '發(fā)送者 id', 去除該字段
     rec_id     int null comment '接受者 id',
     message_id int null comment '外鍵,信息內容',
     is_read    int null comment '是否已讀'
 )
     comment '信息表';
 create table t_message_content
 (
     id        int          null comment '信息內容id',
     send_id    int         null comment '發(fā)送者id',
     content   varchar(255) null comment '內容',
     send_time datetime     null comment '發(fā)送時間'
 );
用戶側操作

登錄后,首先查詢 message_content 中的那些沒有在 message 中有記錄的數(shù)據,表示是未讀的站內信。在查閱站內信的內容時,再將相關的記錄插入 message。

系統(tǒng)側操作

發(fā)站內信時:

  • 只在 message_content 插入站內信的主體內容

  • message 不插入記錄

假設商家要給 10W 用戶發(fā)券:

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

有什么問題?重復消費,導致超發(fā)!

  • 運營提供滿足條件的用戶文件,上傳到發(fā)券管理后臺并選擇要發(fā)送的優(yōu)惠券

  • 管理服務器根據【用戶ID】、【券批次ID】生成消息,發(fā)送到MQ

  • 優(yōu)惠券服務器消費消息

 # 記住使用事務哦!
 INSERT INTO coupon (user_id, coupon_id,batch_id)
   VALUES(1001, 66889, 1111);
 
 UPDATE coupon_batch SET total_count = total_count - 1,
                           assign_count = assign_count + 1
                       WHERE batch_id = 1111 AND total_count > 0;
領券

步驟:

  • 校驗優(yōu)惠券余量

 SELECT total_count FROM coupon_batch 
   WHERE batch_id = 1111;
  • 新增優(yōu)惠券用戶表,扣減余量

 # 注意事務!
 INSERT INTO coupon (user_id, coupon_id,batch_id)
   VALUES(1001, 66889, 1111); 
 
 UPDATE coupon_batch SET total_count = total_count - 1,
                           assign_count = assign_count + 1
                       WHERE batch_id = 1111 AND total_count > 0;

用戶領券過程中,其實也會出現(xiàn)類似秒殺場景。秒殺場景下會有哪些問題,如何解決?

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

解決用戶重復領取或多領:

Redis 數(shù)據校驗!

  • 領券前,先查緩存

 # 判斷成員元素是否是集合的成員
 SISMEMBER KEY VALUE
 SISMEMBER batch_id:1111:user_id 1001
  • 領券

  • 領券后,更新緩存

 # 將一或多個成員元素加入到集合中,已經存在于集合的成員元素將被忽略 
 SADD KEY VALUE1......VALUEN
 SADD batch_id:1111:user_id 1001
用券

何時校驗優(yōu)惠券使用規(guī)則?

  • 確認訂單(√)

  • 提交訂單

  • 立即付款

確認訂單頁,對優(yōu)惠券進行校驗:

  • 判斷是否過期

  • 判斷適用范圍

  • 判斷是否達到門檻

  • 判斷是否互斥

返回可用券

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

 SELECT batch_id FROM coupon WHERE user_id = 1001 AND status = 0;
 
 SELECT rule_id FROM coupon_batch WHERE batch_id = 1111;
 
 SELECT name, type, rule_content FROM rule WHERE rule_id = 1010;
選擇可用券,并返回結果

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

同時操作多個服務,如何保證一致性?

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

表設計

優(yōu)惠券操作記錄表 Coupon_opt_record

 create table t_coupon_opt_record
 (
     user_id     int      null comment '用戶id',
     coupon_id   int      null comment '優(yōu)惠券id',
     operating   int      null comment '操作,0-鎖定、1-核銷、2-解鎖',
     operated_at datetime null comment '操作時間'
 );

TCC,Try-Confirm-Cancel,目前分布式事務主流解決方案。

階段一:Try

對資源進行凍結,預留業(yè)務資源

創(chuàng)建訂單時,將優(yōu)惠券狀態(tài)改為 “凍結”

階段二:Confirm

確認執(zhí)行業(yè)務操作,做真正提交,將第一步Try中凍結的資源,真正扣減

訂單支付成功,將優(yōu)惠券狀態(tài)改為 “已使用”

階段三:Cancel

取消執(zhí)行業(yè)務操作,取消Try階段預留的業(yè)務資源

支付失敗/超時或訂單關閉情況,將優(yōu)惠券狀態(tài)改為 “未使用”

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

Scale擴展

快過期券提醒:

定時掃券表:

缺點:掃描數(shù)據量太大,隨著歷史數(shù)據越來越多,會影響線上主業(yè)務,最終導致慢SQL。

延時消息:

缺點:有些券的有效時間太長了(30天)以上,有可能造成大量 MQ 積壓

新增通知表:

優(yōu)點:掃描的數(shù)據量小,效率高。刪除無用的已通知的數(shù)據記錄

通知信息表(notify_msg)設計
 create table t_notify_msg
 (
     id          bigint auto_increment comment '自增主鍵',
     coupon_id   bigint       null comment '券id',
     user_id     bigint       null comment '用戶id',
     notify_day  varchar(255) null comment '需要執(zhí)行通知的日期',
     notify_type int          null comment '通知類型,1-過期提醒',
     notif_time  timestamp    null comment '通知的時間,在該時間戳所在天內通知',
     status      int          null comment '通知狀態(tài),0-初始狀態(tài)、1-成功、2-失敗',
     constraint t_notify_msg_id_uindex
         unique (id)
 );
 alter table t_notify_msg
     add primary key (id);

過期券提醒:

  • 在創(chuàng)建優(yōu)惠券的時候就將需要提醒的記錄插入提醒表中notify_msg

  • 把用戶ID+批次ID+通知日期作為唯一索引,防止同一個批次有重復的記錄通知,保證每天只會被通知一次

  • 建立notify_time,通知時間索引,每日的通知掃描通過該索引列查詢,通過索引列來提高查詢效率

  • 通知完成后該表中的數(shù)據變失去了意義,通過定時任務將該數(shù)據刪除

數(shù)據庫層面優(yōu)化 - 索引

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

發(fā)券接口,限流保護

前端限流:

點擊一次后,按鈕短時間內置灰

Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)

后端限流:

部分請求直接跳轉到【繁忙頁】

到此,關于“Java+MySQL如何實現(xiàn)設計優(yōu)惠券系統(tǒng)”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續(xù)學習更多相關知識,請繼續(xù)關注億速云網站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI