背景
在计算机专业面试中,业务逻辑BUG的识别和修复是一个常见的。这个旨在考察者对业务流程的理解、对代码的细致分析能力以及解决的技巧。是一个典型的面试题,以及相应的解答过程。
面试题
假设你正在参与一个在线购物平台的开发,该平台有一个功能是用户下单后,系统会自动计算订单的总金额,并从用户的账户余额中扣除相应的金额。是一个简化的代码片段,用于处理订单金额的计算和扣除:
python
def process_order(user_id, order_items):
# 假设order_items是一个列表,每个元素是一个字典,包含item_id和price
total_amount = 0
for item in order_items:
total_amount += item['price']
if total_amount > 1000:
total_amount *= 1.1 # 应用10%的折扣
if user_id == 1:
total_amount *= 0.9 # 用户ID为1的用户享受9折优惠
return total_amount
def deduct_amount(user_id, total_amount):
# 假设这个函数从数据库中获取用户账户余额
user_balance = get_user_balance(user_id)
if total_amount <= user_balance:
update_user_balance(user_id, user_balance – total_amount)
return True
else:
return False
# 假设的用户数据
user_id = 1
order_items = [{'item_id': 101, 'price': 200}, {'item_id': 102, 'price': 500}]
# 处理订单并扣除金额
total_amount = process_order(user_id, order_items)
if deduct_amount(user_id, total_amount):
print("Order processed and amount deducted successfully.")
else:
print("Failed to deduct amount, insufficient balance.")
在上述代码中,存在一个业务逻辑上的BUG。请找出这个BUG并解释为什么它是BUG,给出修复BUG的代码。
解答过程
我们需要运行上述代码来观察其行为。在运行代码之前,我们可以先对代码进行分析。
1. `process_order` 函数计算订单的总金额,并应用了两个折扣条件:总金额超过1000,则应用10%的折扣;用户ID为1,则应用9折优惠。
2. `deduct_amount` 函数尝试从用户的账户余额中扣除订单总金额。扣除成功,则返回True;余额不足,则返回False。
让我们运行代码并观察结果。
python
# 假设的用户数据
user_id = 1
order_items = [{'item_id': 101, 'price': 200}, {'item_id': 102, 'price': 500}]
# 处理订单并扣除金额
total_amount = process_order(user_id, order_items)
if deduct_amount(user_id, total_amount):
print("Order processed and amount deducted successfully.")
else:
print("Failed to deduct amount, insufficient balance.")
运行结果可能是:
Order processed and amount deducted successfully.
这里存在一个BUG。用户ID为1,订单中有一个价格超过1000的商品,该用户将获得额外的折扣,因为代码中先计算了超过1000的折扣,又应用了用户ID为1的折扣。
BUG解释
BUG在于折扣的应用顺序。代码检查了总金额是否超过1000,超过,则应用10%的折扣。代码检查用户ID是否为1,是,则应用9折优惠。这意味着用户ID为1,且订单中有一个价格超过1000的商品,实际折扣将是19%(10% + 9%),而不是预期的9%。
修复BUG的代码
为了修复这个BUG,我们需要调整折扣的应用顺序。我们可以先应用用户ID为1的9折优惠,再检查总金额是否超过1000,并相应地应用10%的折扣。
python
def process_order(user_id, order_items):
# 假设order_items是一个列表,每个元素是一个字典,包含item_id和price
total_amount = 0
for item in order_items:
total_amount += item['price']
if user_id == 1:
total_amount *= 0.9 # 用户ID为1的用户享受9折优惠
if total_amount > 1000:
total_amount *= 1.1 # 应用10%的折扣
return total_amount
# 其余代码不变
运行代码,我们应该得到正确的结果。
python
# 假设的用户数据
user_id = 1
order_items = [{'item_id': 101, 'price': 200}, {'item_id': 102, 'price': 500}]
# 处理订单并扣除金额
total_amount = process_order(user_id, order_items)
if deduct_amount(user_id, total_amount):
print("Order processed and amount deducted successfully.")
else:
print("Failed to deduct amount, insufficient balance.")
运行结果应该是:
Failed to deduct amount, insufficient balance.
这是因为虽然用户ID为1享受了9折优惠,但订单总金额为890(200 + 500 * 0.9),小于1000,没有额外的10%折扣。这样,代码的逻辑就更加清晰和正确了。
还没有评论呢,快来抢沙发~