Java Spring Cloud 與響應式微服務(三)客戶服務建立

gung123發表於2020-03-09

新建一個基本的 Spring Boot 工程,命名為  cloud-customer,POM 依賴和之前的  cloud-account 的一模一樣。瞭解springcloud架構可以加求求:三五三六二四七二五九

配置檔案如下,僅是改了服務名和埠號:

spring:
  application:
    name: cloud-customer
server:
  port: 8200
eureka:
  client:
    service-url:
      defaultZone: 

建立一個 Customer 的實體類

@Data
@AllArgsConstructor
@NoArgsConstructor
@Document(collection = "customers")
public class Customer {
    @Id
    private String id;
    private String name;
    private String mobile;
}

資料訪問層直接繼承  ReactiveCrudRepository  我們便有了基本的 CRUD 能力

public interface CustomerMongoReactiveRepository extends ReactiveCrudRepository<Customer, String> {
}

因為我們只是示例,不做複雜的業務邏輯,所以省略了 Service 層,在 Controller 裡邊直接將 CRUD 的操作代理給了 Repository。

@RestController
@RequestMapping("/customer")
public class CustomerController {
    @Autowired private CustomerMongoReactiveRepository repository;
    @Autowired private WebClient.Builder webClientBuilder;
    @GetMapping("")
    public Flux<Customer> list() {
        return repository.findAll();
    }
    @GetMapping("/{id}")
    public Mono<Customer> get(@PathVariable String id) {
        return repository.findById(id);
    }
    @PostMapping("")
    public Mono<Customer> create(@RequestBody Customer customer) {
        return repository.save(customer);
    }
    @PutMapping("/{id}")
    public Mono<Customer> update(@PathVariable("id") String id, @RequestBody Customer customer) {
        customer.setId(id);
        return repository.save(customer);
    }
    @DeleteMapping("/{id}")
    public Mono<Void> delete(@PathVariable String id) {
        return repository.deleteById(id);
    }
}

到這裡,我們的服務註冊中心和兩個微服務就都好了。但是,這兩個微服務之間還是完全獨立的,沒有相互間的服務呼叫。現在我們來實現之前說的需求:客戶服務與帳戶服務可以相互通訊,以獲取客戶的所有帳戶,並透過客戶服務 API 方法返回。

首先建立一個 Java Config,這裡我們不再使用  RestTemplate  來呼叫服務,而是  WebClient 。這個配置看起來和註冊  RestTemplate  時差不多,但是要注意這裡註冊的 Bean 是  WebClient.Builder

@Configuration
public class WebClientConfig {
    @Bean
    @LoadBalanced
    public WebClient.Builder loadBalancedWebClientBuilder() {
        return WebClient.builder();
    }
}

除了這種寫法,還有一種寫法是

public class MyClass {
    @Autowired
    private LoadBalancerExchangeFilterFunction lbFunction;
    public Mono<String> doOtherStuff() {
        return WebClient.builder().baseUrl(")
            .filter(lbFunction)
            .build()
            .get()
            .uri("")
            .retrieve()
            .bodyToMono(String.class);
    }
}

下邊的是錯誤的寫法,會丟擲異常

@Bean
@LoadBalanced
public WebClient loadBalancedWebClient() {
    return WebClient.builder().baseUrl(").build();
}

然後在  CustomerController  實現這個端點:

@GetMapping("/{id}/account")
public Flux<Account> getAllAccounts(@PathVariable String id) {
    return webClientBuilder.baseUrl(").build()
        .get().uri("/customer/" + id)
        .retrieve()
        .bodyToFlux(Account.class);
}

這裡需要在  cloud-customer  裡建立一個 DTO  Account ,因為和  cloud-account  裡的完全一樣,就省略了。

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69952307/viewspace-2679180/,如需轉載,請註明出處,否則將追究法律責任。

相關文章