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优缺点

相关推荐
蓝倾22 分钟前
淘宝获取商品分类接口操作指南
前端·后端·fastapi
小希爸爸27 分钟前
curl 网络测试常用方法
后端
星星电灯猴1 小时前
iOS WebView 调试实战 页面跳转失效与历史记录错乱的排查路径
后端
重楼七叶一枝花2 小时前
MySQL的在线模式学习笔记
后端·mysql
代码男孩2 小时前
python包管理工具uv的使用
后端
CodeWolf2 小时前
关于端口号配置优先级的问题
后端
C182981825752 小时前
Ribbon轮询实现原理
后端·spring cloud·ribbon
lemon_sjdk2 小时前
LWJGL教程(2)——游戏循环
java·人工智能·算法·游戏
鹿鹿的布丁2 小时前
freeswitch通过编译方式安装
后端
JavaDog程序狗2 小时前
【软件环境】Windows安装JDK21
后端