Spring Boot基于AOP的本地/远程调用动态路由实践

01为什么我们要"无缝切换"

• MVP 阶段:单体最快,写完就可以上线。

• 流量暴涨后:微服务才能横向扩展,但把本地 userService.findById() 改成 feignClient.findById() 真要命------全链路改一遍,风险高、工期长。

• 理想状态:同一段业务代码,今天跑本地,明天改一行配置就跑远程,零代码侵入。

02核心设计一张图

java 复制代码
┌---------------┐
│  业务代码      │  ← 只认接口 UserService
└-------┬-------┘
        │ 注入
┌-------┴----------------┐
│  统一抽象层             │
│  根据配置动态选择实现    │
├------------┬-----------┤
│ UserServiceLocalImpl │ UserServiceRemoteImpl │
└------------┴-----------┘

03一步步落地

先画好契约------接口层

java 复制代码
// **统一接口** = 本地实现 + 远程 Feign 共用
public interface UserService {
    User getUserById(Long id);
    List<User> listAllUsers();
    User saveUser(User u);
    void updateUser(User u);
    void deleteUser(Long id);
}

本地实现

java 复制代码
@Service
@ConditionalOnProperty(name = "service.mode", havingValue = "local", matchIfMissing = true)
public class UserServiceLocalImpl implements UserService {

    @Autowired
    private UserRepository repo;   // 直连数据库,不走网络

    @Override
    public User getUserById(Long id) {
        return repo.findById(id).orElse(null);
    }

    @Override
    public List<User> listAllUsers() {
        return repo.findAll();
    }

    /* 其余方法省略 */
}

注解

  1. @ConditionalOnProperty:Spring Boot 的条件装配神器,配置文件里写 local 就激活。
  2. 直接依赖 DAO,零网络损耗,单元测试也能秒起。

远程实现------Feign

java 复制代码
// 1. 声明式 HTTP 客户端
@FeignClient(name = "user-service", fallback = UserServiceFallback.class)
public interface UserServiceFeignClient {

    @GetMapping("/api/users/{id}")
    User getUserById(@PathVariable("id") Long id);

    @GetMapping("/api/users")
    List<User> listAllUsers();

    @PostMapping("/api/users")
    User saveUser(@RequestBody User u);

    @PutMapping("/api/users")
    void updateUser(@RequestBody User u);

    @DeleteMapping("/api/users/{id}")
    void deleteUser(@PathVariable("id") Long id);
}
@Service
@ConditionalOnProperty(name = "service.mode", havingValue = "remote")
public class UserServiceRemoteImpl implements UserService {

    @Autowired
    private UserServiceFeignClient feignClient;   // 代理,真正发 HTTP

    @Override
    public User getUserById(Long id) {
        return feignClient.getUserById(id);
    }

    /* 其余方法省略 */
}

注解

  1. @FeignClient:Ribbon + Hystrix 自动集成,负载均衡、熔断降级开箱即用。
  2. fallback:远程挂了直接走兜底逻辑,雪崩不存在的。

自动配置

java 复制代码
@Configuration
@EnableFeignClients(basePackages = "com.example.feign")
public class ServiceAutoConfiguration {

    @Bean
    @ConditionalOnProperty(name = "service.mode", havingValue = "remote")
    public UserService userServiceRemote(UserServiceFeignClient client) {
        return new UserServiceRemoteImpl(client);
    }

    @Bean
    @ConditionalOnProperty(name = "service.mode", havingValue = "local", matchIfMissing = true)
    public UserService userServiceLocal(UserRepository repo) {
        return new UserServiceLocalImpl(repo);
    }
}

配置

java 复制代码
# application.yml
service:
  mode: local   # 改成 remote 秒变微服务

进阶玩法:细粒度路由 + 智能负载
按模块单独开关
service:
  user: local
  order: remote
  product: local

AOP 动态选路(伪代码)

java 复制代码
@Aspect
@Component
public class SmartRoutingAspect {
    @Around("@annotation(SmartRouting)")
    public Object route(ProceedingJoinPoint pjp) {
        // 统计 RT、错误率,实时计算 local vs remote 分值
        boolean goLocal = loadBalancingService.shouldGoLocal(pjp.getSignature());
        return goLocal ? pjp.proceed() : feignInvoke(pjp);
    }
}

05优缺点

相关推荐
泡海椒4 分钟前
jquickexcel 全功能指南:从数据导入到精美导出的完整流程
后端
一只爱撸猫的程序猿16 分钟前
创建一个使用Spring AI框架构建RAG(Retrieval-Augmented Generation)系统的案例
spring boot·aigc·ai编程
iOS开发上架哦16 分钟前
移动端网页调试实战,键盘弹出与视口错位问题的定位与优化
后端
百度Geek说20 分钟前
PaddleMIX推出扩散模型推理加速Fast-Diffusers:自研蒸馏加速方法FLUX-Lightning实现4步图像生成
后端
gopher_looklook28 分钟前
Go并发实战:singleflight 源码解读与二次封装
数据结构·后端·go
用户8338102512236 分钟前
我为什么做PmMock:让接口设计不再头疼
前端·后端
congvee40 分钟前
springboot学习第11期 - @HttpExchange
spring boot
二闹43 分钟前
IService 和 BaseMapper:CRUD 操作的选择指南
后端
dylan_QAQ44 分钟前
【附录】Spring AOP 基础知识及应用
后端·spring
Java中文社群1 小时前
抱歉!Java面试标准答案最不重要
java·后端·面试