
很多团队在做外卖跑腿系统初期,往往只服务一个城市,商户数量有限,订单规模不大,使用单体应用加单库结构就可以顺利运行。

但当业务开始扩张后,问题会迅速出现。
城市数量增加,数据混在一起,查询变慢 商户变多,权限混乱,数据隔离困难 订单暴涨,数据库压力过大,系统频繁卡顿
这些问题本质上都来自一个原因:底层架构没有提前为多城市和多商户做设计。
一套真正可长期运营的外卖跑腿系统,必须从一开始就考虑三个核心能力:
多城市部署能力 多商户隔离能力 高并发处理能力
下面结合实际开发经验,从架构和代码层面拆解具体实现方案。
推荐采用分层加微服务的结构,将核心能力拆分成独立服务,例如:
网关层 订单服务 商户服务 骑手调度服务 支付结算服务 城市管理服务 缓存与消息队列层
所有终端(用户端、骑手端、商家端、后台)统一通过网关访问后端服务。
这样拆分的好处是:
各模块职责清晰 可以独立扩容 单个服务故障不影响整体 更容易支持多城市部署
当业务覆盖多个城市时,最关键的是数据隔离。
如果所有城市共用一套数据库,订单量上来之后查询效率会急剧下降,同时也不利于独立扩容。
更合理的方式是按城市分库。
例如:
北京一个数据库 上海一个数据库 广州一个数据库
这样可以做到:
单城市独立维护 查询性能更高 扩容成本更低 故障互不影响
在 SpringBoot 项目中,可以通过动态数据源实现自动路由。
第一步,使用 ThreadLocal 保存当前城市标识。
public class CityContextHolder {
private static final ThreadLocal<String> CITY = new ThreadLocal<>();
public static void set(String city){
CITY.set(city);
}
public static String get(){
return CITY.get();
}
public static void clear(){
CITY.remove();
}
}第二步,请求进入时通过拦截器识别城市。
@Component
public class CityInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
String cityCode = request.getHeader("city-code");
CityContextHolder.set(cityCode);
return true;
}
}第三步,实现动态数据源切换。
public class DynamicDataSource extends AbstractRoutingDataSource {
@Override
protected Object determineCurrentLookupKey() {
return CityContextHolder.get();
}
}这样一来,同一套代码即可根据不同城市自动访问不同数据库,实现真正的多城市部署。

外卖跑腿平台本质是多商户入驻模式。
每个商家都需要独立管理自己的订单、商品、财务数据,因此必须采用多租户设计。
最简单可靠的方法是字段隔离,也就是每张核心业务表都带上 merchant_id。
例如订单表:
CREATE TABLE orders (
id BIGINT PRIMARY KEY,
merchant_id BIGINT,
user_id BIGINT,
amount DECIMAL(10,2),
status TINYINT,
create_time DATETIME
);所有查询都必须携带 merchant_id 条件。
可以使用 AOP 在请求进入时自动写入商户上下文。
public class MerchantContext {
private static final ThreadLocal<Long> HOLDER = new ThreadLocal<>();
public static void set(Long id){
HOLDER.set(id);
}
public static Long get(){
return HOLDER.get();
}
}@Aspect
@Component
public class MerchantAspect {
@Before("execution(* com.xxx.service..*(..))")
public void setMerchant() {
MerchantContext.set(LoginUser.getMerchantId());
}
}查询时统一带入:
@Select("select * from orders where merchant_id = #{merchantId}")
List<Order> list(Long merchantId);这样可以确保商户之间数据完全隔离,权限模型也更加简单清晰。
当订单量上来之后,数据库往往成为最大瓶颈。
正确做法不是提升硬件,而是引入缓存和消息队列削峰。
推荐组合:
Redis 负责缓存和库存控制 MQ 负责异步处理订单 数据库专注持久化
典型下单流程如下:
用户提交订单 Redis 校验库存 发送消息队列 异步创建订单 调度骑手
示例代码如下。
发送消息:
rabbitTemplate.convertAndSend(
"order.exchange",
"order.create",
orderDTO
);消费消息:
@RabbitListener(queues = "order.queue")
public void createOrder(OrderDTO dto){
orderService.create(dto);
}这种方式可以把瞬时高峰流量变成平滑流量,极大提高系统稳定性。
如果希望系统具备长期可运营能力,建议从一开始就采用以下方案:
城市分库部署 商户多租户隔离 微服务拆分架构 Redis缓存加速 MQ异步削峰 容器化部署支持横向扩容
这样设计后:
新增城市只需增加数据库 新增商户无需改动架构 订单增长可直接扩容服务
系统可以稳定支撑长期发展,而不用反复重构。

外卖跑腿系统真正的竞争力并不是界面,而是底层架构是否能支撑规模化运营。
一套支持多城市、多商户和高并发的系统,才能帮助平台持续扩张和盈利。
如果你正在搭建或选型外卖跑腿系统源码,优先考虑是否具备以上架构能力,这比单纯的功能数量更重要。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。