卖不甜枣 发表于 2024-11-6 14:20:12

数据库系统 第11节 事件

在计算机科学中,特别是在数据库管理中,“事件”(Transaction)是一个非常重要的概念。事件是用来管理对数据库的一系列操作的逻辑单元,这些操作要么全部乐成完成,要么一个也不做。事件包管了数据的同等性和可靠性,尤其在并发情况中,这一点尤为重要。
事件通常遵循ACID特性:


[*]原子性(Atomicity):整个事件的所有操作要么全部完成,要么都不执行。
[*]同等性(Consistency):事件的执行结果必须使数据库从一个同等性状态转变到另一个同等性状态。
[*]隔离性(Isolation):多个并发事件之间的操作是相互独立的,不会相互干扰。
[*]持久性(Durability):一旦事件完成,它对数据库所做的更改就是永久性的。
案例分析:银行转账

假设我们有一个简单的银行业务场景,用户A必要向用户B转账100元。这个操作涉及到两个账户:用户A的账户余额淘汰100元,用户B的账户余额增长100元。
步调说明:


[*]开始事件:首先启动一个事件来处理这笔转账业务。
[*]减款操作:从用户A的账户中扣除100元。
[*]加款操作:向用户B的账户中增长100元。
[*]提交事件:假如上述两步操作都乐成,则提交事件,否则回滚事件。
大概的题目及办理方案:



[*] 题目1:假如在第二步和第三步之间系统崩溃,会发生什么?

[*]办理方案:通过事件的回滚机制确保在系统崩溃后能够规复到转账前的状态,即用户A的账户不被扣款,用户B的账户也没有增长金额。

[*] 题目2:假如有多个转账哀求同时发生,怎样避免数据冲突?

[*]办理方案:使用锁定机制来包管数据的同等性。例如,在减款操作之前锁定用户A的账户,在加款操作之前锁定用户B的账户。这样可以确保在一个事件未完成之前,其他事件不能访问这些账户。

实现细节:

在实际应用中,比如使用SQL语句实现上述转账过程,大概会像这样:
BEGIN TRANSACTION; -- 开始事务
UPDATE accounts SET balance = balance - 100 WHERE id = 'A';
UPDATE accounts SET balance = balance + 100 WHERE id = 'B';
COMMIT; -- 提交事务
假如在这个过程中发生了任何错误,如余额不敷或者数据库连接丢失等,可以通过ROLLBACK命令来取消所有已经举行的操作,以保持数据的同等性。
通过这个例子,我们可以看到事件是怎样确保数据完整性和同等性的。在设计和开辟涉及数据安全的应用步调时,正确地使用事件是非常关键的。
让我们继续使用银行业务作为背景,但这次我们将讨论更复杂的场景——跨行转账。这个案例将涉及到两个不同的银行系统,这增长了事件处理的复杂性。
案例分析:跨行转账

假设用户A在Bank A开户,用户B在Bank B开户。用户A必要向用户B转账100元。为了简化讨论,我们假设这两个银行之间有预先建立好的合作机制,允许它们举行这种跨行生意业务。
步调说明:


[*]开始事件:在Bank A的系统中启动一个事件来处理转账。
[*]减款操作:从用户A的账户中扣除100元,并将这笔生意业务记录在Bank A的系统中。
[*]发送消息:Bank A通过一个安全的通讯渠道向Bank B发送一条消息,指示Bank B向用户B的账户添加100元。
[*]接收消息:Bank B接收到消息并验证其合法性。
[*]加款操作:Bank B在其系统中为用户B的账户增长100元,并记录这笔生意业务。
[*]确认消息:Bank B向Bank A发送确认消息,表示转账已完成。
[*]竣事事件:Bank A接收到确认消息后,提交事件并完成转账流程。
大概的题目及办理方案:



[*] 题目1:假如在步调3和步调5之间Bank B的系统崩溃,会发生什么?

[*]办理方案:Bank B应该能够检测到未完成的事件,并在系统重启后自动重试或哀求Bank A重新发送指令。

[*] 题目2:假如在步调5之后、步调6之前Bank B的系统崩溃,怎样处理?

[*]办理方案:Bank B必要在加款操作完成后立即记录这一操作,而且在系统崩溃后能够根据记录规复到最新状态。此外,Bank A必要有超时机制来检测长时间未收到确认的情况,并采取相应的措施,例如重新发送指令或取消事件。

实现细节:

在实际应用中,跨行转账大概涉及到更为复杂的协议和标准,例如SWIFT或类似的金融信息交换协议。下面是一个简化的示例,展示怎样在两个银行系统之间和谐这样的转账操作。
-- Bank A系统中的操作
BEGIN TRANSACTION; -- 开始事务
UPDATE accounts SET balance = balance - 100 WHERE id = 'A';
INSERT INTO transactions (account_id, amount, status) VALUES ('A', -100, 'pending'); -- 记录减款交易
CALL send_transfer_message('B', 100); -- 发送转账消息给Bank B
WAIT FOR confirmation_message; -- 等待确认消息
IF confirmation_received THEN
    COMMIT; -- 提交事务
ELSE
    ROLLBACK; -- 回滚事务
END IF;
在Bank B系统中,也会有一个类似的事件处理流程,用于接收消息、更新账户余额以及发送确认消息。
额外考虑因素:



[*]安全性:确保转账消息的安全传输非常重要。这通常涉及到加密技能和数字签名。
[*]日志和审计:每个步调都应该被记录下来,以便于审计和故障排查。
[*]错误处理:必要有详细的错误处理机制来应对各种非常情况。
通过这个案例,我们可以看到在复杂的系统交互中,事件不仅必要在单个数据库层面得到妥善处理,还必要在不同的系统之间举行和谐,以确保整体流程的顺遂举行。
让我们继续扩展跨行转账的案例,考虑一些更具体的细节和技能挑衅。
案例扩展:跨行转账与分布式事件

在跨行转账的例子中,我们必要考虑怎样处理分布在不同银行系统中的事件。这引入了一个新的概念——分布式事件。分布式事件是指超过多个节点或系统的事件,此中每个节点或系统大概有自己的事件管理器。在我们的例子中,Bank A和Bank B各自有自己的事件管理系统。
技能挑衅:


[*] 同等性:确保两个银行系统中的数据同等性是一个巨大的挑衅。当一个系统中的操作乐成,而另一个系统中的操作失败时,怎样保持同等性?
[*] 网络延伸:由于两个银行系统大概位于不同的地理位置,因此网络延伸会影响事件的性能。
[*] 故障规复:当此中一个系统出现故障时,怎样规复数据的同等性?
[*] 安全性:确保数据传输的安全性,防止数据被篡改或窃取。
办理方案:



[*] 两阶段提交协议(Two-Phase Commit, 2PC):这是一种经典的分布式事件和谐协议,用来办理上述题目。它分为两个阶段:

[*]准备阶段(Prepare Phase):主事件管理器扣问所有参与节点是否准备好提交事件。
[*]提交阶段(Commit Phase):根据所有参与节点的回答决定提交还是回滚事件。

[*] 赔偿事件(Compensating Transactions):假如某个操作失败,可以通过执行相反的操作(赔偿操作)来回滚整个事件。
示例扩展:

假设用户A在Bank A开户,用户B在Bank B开户,现在用户A必要向用户B转账100元。

[*]开始事件:在Bank A的系统中启动一个事件来处理转账。
[*]减款操作:从用户A的账户中扣除100元,并将这笔生意业务记录在Bank A的系统中。
[*]发送消息:Bank A通过一个安全的通讯渠道向Bank B发送一条消息,指示Bank B向用户B的账户添加100元。
[*]接收消息:Bank B接收到消息并验证其合法性。
[*]准备阶段:Bank B复兴Bank A表示已经准备好提交事件。
[*]加款操作:Bank B在其系统中为用户B的账户增长100元,并记录这笔生意业务。
[*]提交阶段:Bank A接收到Bank B的准备消息后,正式提交事件。
[*]确认消息:Bank B向Bank A发送确认消息,表示转账已完成。
[*]竣事事件:Bank A接收到确认消息后,提交事件并完成转账流程。
处理故障:



[*]故障1:假如在步调5之前Bank B的系统崩溃,那么Bank B必要在重启后检测到未完成的事件,并实行重新到场准备阶段。
[*]故障2:假如在步调7之前Bank A的系统崩溃,那么Bank A必要在重启后检测到未完成的事件,并实行重新发送提交或回滚命令。
[*]故障3:假如在步调8之前Bank B的系统崩溃,那么Bank B必要在重启后检测到未完成的事件,并实行重新发送确认消息。
安全性考虑:



[*]加密:使用加密技能保护数据传输,例如使用TLS/SSL协议。
[*]数字签名:使用数字签名确保消息的完整性和真实性。
日志和审计:



[*]日志记录:记录每一步操作的日志,包括事件状态的变革、消息发送和接收的时间戳等。
[*]审计追踪:为每个事件创建审计日志,记录转账过程中的所有操作。
实现细节:

在实际应用中,分布式事件的处分析更加复杂,大概涉及到更高级别的协议和技能。以下是一个简化的示例,展示了怎样在Bank A的系统中使用两阶段提交协议来处理跨行转账。
-- Bank A系统中的操作
BEGIN DISTRIBUTED TRANSACTION; -- 开始分布式事务
UPDATE accounts SET balance = balance - 100 WHERE id = 'A';
INSERT INTO transactions (account_id, amount, status) VALUES ('A', -100, 'pending'); -- 记录减款交易
CALL send_transfer_message('B', 100); -- 发送转账消息给Bank B
WAIT FOR prepare_response FROM Bank B; -- 等待准备响应
IF prepare_response THEN
    CALL commit_transaction(); -- 提交事务
    WAIT FOR commit_confirmation FROM Bank B; -- 等待提交确认
    IF commit_confirmation THEN
      COMMIT; -- 提交分布式事务
    ELSE
      ROLLBACK; -- 回滚分布式事务
    END IF;
ELSE
    ROLLBACK; -- 回滚分布式事务
END IF;
在Bank B系统中,也会有一个类似的事件处理流程,用于接收消息、更新账户余额以及发送准备相应和提交确认消息。
通过这种方式,即使在分布式情况下,也能确保跨行转账的安全性和同等性。
这次我们来看一个电子商务网站中的购物车结算流程,这是一个典型的必要事件支持的场景。在这个案例中,我们将关注怎样处理用户下单、库存淘汰、支付确认等多个环节的事件处理。
案例分析:电子商务网站购物车结算

假设有一个电子商务网站,用户可以在购物车中选择商品并完成购买。该过程包括几个关键步调:检查库存、淘汰库存、天生订单、支付确认等。
步调说明:


[*]开始事件:在用户点击“提交订单”按钮时启动一个事件。
[*]检查库存:查询数据库以确保所选商品有足够的库存。
[*]淘汰库存:假如库存充足,则淘汰相应数目的商品库存。
[*]天生订单:在订单表中插入一条新记录,包括用户信息、商品信息和总价等。
[*]支付确认:期待支付系统简直认信号。
[*]竣事事件:假如支付乐成,则提交事件;假如支付失败,则回滚事件。
大概的题目及办理方案:



[*] 题目1:假如在淘汰库存后,支付系统出现故障,导致无法确认支付状态,怎么办?

[*]办理方案:在这种情况下,事件应该包罗一个超时机制,假如在肯定时间内没有收到支付确认,就回滚事件,以确保库存不会被错误地淘汰。

[*] 题目2:假如有多个用户同时实行购买同一款商品,怎样避免库存超卖?

[*]办理方案:在检查库存和淘汰库存的过程中使用锁定机制,确保在淘汰库存之前锁定商品的库存记录,直到事件完成。

实现细节:

下面是一个简化的SQL脚本示例,展示了怎样在数据库层面上处理这个过程。
BEGIN TRANSACTION; -- 开始事务

-- 检查库存
SELECT * FROM products WHERE product_id = '123' FOR UPDATE;

-- 减少库存
UPDATE products SET stock_quantity = stock_quantity - 1 WHERE product_id = '123';

-- 生成订单
INSERT INTO orders (user_id, product_id, quantity, total_price)
VALUES ('user123', '123', 1, 100);

-- 模拟支付确认
WAIT FOR payment_confirmation; -- 假设这是通过某种方式等待支付确认的代码

IF payment_confirmed THEN
    COMMIT; -- 如果支付确认,则提交事务
ELSE
    ROLLBACK; -- 否则回滚事务
END IF;
扩展案例:分布式事件处理

在实际应用中,支付确认大概涉及外部支付服务提供商,这就必要处理分布式事件。这里我们使用两阶段提交协议(Two-Phase Commit, 2PC)来处理这种跨系统的事件。
步调说明:


[*]开始事件:在用户点击“提交订单”按钮时启动一个事件。
[*]检查库存:查询数据库以确保所选商品有足够的库存。
[*]淘汰库存:假如库存充足,则淘汰相应数目的商品库存。
[*]天生订单:在订单表中插入一条新记录,包括用户信息、商品信息和总价等。
[*]发送支付哀求:向支付服务提供商发送支付哀求。
[*]支付服务准备阶段:支付服务提供商准备执行支付,并复兴准备确认。
[*]提交事件:假如支付服务提供商确认准备,则提交事件。
[*]支付服务提交阶段:支付服务提供商完成支付,并发送支付确认。
[*]竣事事件:假如支付确认,则提交事件;假如支付失败,则回滚事件。
实现细节:

在实际应用中,支付哀求和支付确认大概通过API调用或消息队列等方式举行。下面是一个简化的示例,展示怎样使用两阶段提交协议处理跨系统的事件。
-- 电子商务系统中的操作
BEGIN DISTRIBUTED TRANSACTION; -- 开始分布式事务

-- 检查库存
SELECT * FROM products WHERE product_id = '123' FOR UPDATE;

-- 减少库存
UPDATE products SET stock_quantity = stock_quantity - 1 WHERE product_id = '123';

-- 生成订单
INSERT INTO orders (user_id, product_id, quantity, total_price)
VALUES ('user123', '123', 1, 100);

-- 发送支付请求
CALL send_payment_request('payment_service', 100); -- 发送支付请求给支付服务提供商

-- 等待支付服务提供商的准备确认
WAIT FOR payment_prepare_confirmation FROM payment_service;

IF payment_prepare_confirmation THEN
    CALL commit_transaction(); -- 提交事务
    WAIT FOR payment_commit_confirmation FROM payment_service; -- 等待支付服务提供商的提交确认
    IF payment_commit_confirmation THEN
      COMMIT; -- 如果支付确认,则提交分布式事务
    ELSE
      ROLLBACK; -- 否则回滚分布式事务
    END IF;
ELSE
    ROLLBACK; -- 如果支付服务提供商未能准备,则回滚分布式事务
END IF;
在支付服务提供商系统中,也会有一个类似的事件处理流程,用于处理支付哀求、发送准备确认和提交确认。
通过这种方式,即使在分布式情况下,也能确保购物车结算过程的安全性和同等性。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: 数据库系统 第11节 事件