我在互联网上找到了例子,但这并没有让我完全理解。使用 WebFlux 时的标准 CRUD。
路由器:
@Configuration
public class PersonRouter {
@Bean
public RouterFunction<ServerResponse> route(PersonHandler handler) {
return RouterFunctions
.route(GET("/getAllPersons").and(accept(MediaType.APPLICATION_JSON)), handler::findAll)
.andRoute(GET("/getPerson/{id}").and(accept(MediaType.APPLICATION_STREAM_JSON)), handler::findById)
.andRoute(POST("/createPerson").and(accept(MediaType.APPLICATION_JSON)), handler::save)
.andRoute(DELETE("/deletePerson/{id}").and(accept(MediaType.APPLICATION_JSON)), handler::delete);
}
}
处理程序:
@Component
public class PersonHandler {
private final PersonService personService;
public PersonHandler(PersonService personService) {
this.personService = personService;
}
public Mono<ServerResponse> findById(ServerRequest request) {
String id = request.pathVariable("id");
return ok()
.contentType(MediaType.APPLICATION_JSON)
.body(personService.getById(id), Person.class);
}
public Mono<ServerResponse> findAll(ServerRequest request) {
return ok()
.contentType(MediaType.APPLICATION_JSON)
.body(personService.getAll(), Person.class);
}
public Mono<ServerResponse> save(ServerRequest request) {
final Mono<Person> person = request.bodyToMono(Person.class);
return ok()
.contentType(MediaType.APPLICATION_JSON)
.body(fromPublisher(person.flatMap(personService::save), Person.class));
}
public Mono<ServerResponse> delete(ServerRequest request) {
String id = request.pathVariable("id");
return ok()
.contentType(MediaType.APPLICATION_JSON)
.body(personService.delete(id), Void.class);
}
}
存储库:
@Repository
public interface PersonRepository extends ReactiveMongoRepository<Person, String> {
}
服务:
@Service
@Transactional
@AllArgsConstructor
public class PersonService {
private final PersonRepository personRepository;
public Flux<Person> getAll() {
return personRepository.findAll().switchIfEmpty(Flux.empty());
}
public Mono<Person> getById(final String id) {
return personRepository.findById(id);
}
public Mono update(final String id, final Person person) {
return personRepository.save(person);
}
public Mono save(final Person person) {
return personRepository.save(person);
}
public Mono delete(final String id) {
final Mono<Person> dbPerson = getById(id);
if (Objects.isNull(dbPerson)) {
return Mono.empty();
}
return getById(id).switchIfEmpty(Mono.empty()).filter(Objects::nonNull).flatMap(personToBeDeleted -> personRepository
.delete(personToBeDeleted).then(Mono.just(personToBeDeleted)));
}
}
我了解除了save
和update
方法之外的所有内容。我不明白为什么我们flatMap
在这种情况下使用。为什么会这样,以及如何在我的Handler中编写update方法的实现。
更新
让我们看看Handler 中的方法save()
public Mono<ServerResponse> save(ServerRequest request) {
final Mono<Person> person = request.bodyToMono(Person.class);
return ok()
.contentType(MediaType.APPLICATION_JSON)
.body(fromPublisher(person.flatMap(personService::save), Person.class));
}
我认为事实是我们已经收到:
final Mono<Person> person = request.bodyToMono(Person.class);
然后我们做:
personService::save
结果,我们得到 Mono< Mono< Person>>
flatMap 就像 map 一样,除了它会解包给定的 lambda 的返回值,如果该值本身包含在 a 中Publisher<T>
。在我们的例子中,该personService.save(T)
方法返回一个Mono<T>
. 如果我们使用 map 而不是flatMap(T)
,我们就会有一个Mono< Mono< T>>
, 而我们真正想要的是一个Mono<T>
. 我们可以使用 flatMap 干净地解决这个问题。
我是对的还是这个说法是错的?