解決springboot整合cxf-jaxrs中json轉換的問題
前言
我在將項目用boot重構時, 關於cxf的使用出瞭一些問題, 主要在實體類和json轉換這一方面。
在看瞭一些晚上的相關答案後, 瞭解到jaxb默認支持xml格式, 而實現對象轉json是需要額外的轉換器的,然後在stackoverflow上找到一個解決方法是聲明一個bean,註入JsonProvider,但我發現這個可以解決服務端將對象轉為json的問題,
而客戶端還是會報一個異常:
No message body reader has been found for class ……, ContentType: application/json
後面在cxf的WebClient類的源碼中發現:
create()方法有很多重載方法,其中有一個是可以指定provider來轉換格式,最後通過這個重載方法解決瞭客戶端json格式轉換問題。
最後的解決方案:
在單獨使用cxf的基礎上做出改動,主要有兩方面
1. 服務端:在啟動類上聲明一個bean, 註入JacksonJaxbJsonProvider
2. 客戶端:在WebClient調用create()方法時,指定轉json的provider
下面是一個簡單的demo:
一、webservice服務端(生產者)
1.maven依賴
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-jpa</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <!--cxf-jaxrs-starter--> <dependency> <groupId>org.apache.cxf</groupId> <artifactId>cxf-spring-boot-starter-jaxrs</artifactId> <version>3.2.0</version> </dependency> <!--jaxrs轉json工具--> <dependency> <groupId>com.fasterxml.jackson.jaxrs</groupId> <artifactId>jackson-jaxrs-json-provider</artifactId> <version>2.8.5</version> </dependency>
2.application.yml配置文件
配置cxf路徑和包掃描
server: port: 9001 cxf: path: /services servlet.init: service-list-path: /info jaxrs: component-scan: true
3.boot應用啟動類配置
在啟動類中聲明一個bean,自動註入JacksonJaxbJsonProvider 對象,這樣cxf在將對象轉為json時會自動使用這個對象
@SpringBootApplication public class CxfServerApplication { public static void main(String[] args) { SpringApplication.run(CxfServerApplication.class, args); } // 配置一個對象與json轉換的工具 @Bean public JacksonJaxbJsonProvider jacksonJaxbJsonProvider() { return new JacksonJaxbJsonProvider(); } }
4.客戶服務接口
關於cxf的路徑註解,請參照其他cxf資料
@Path("/customerService") public interface CustomerService { /** * 客戶服務:根據id查詢客戶 */ @Path("/findById") @GET @Produces({"application/xml", "application/json"}) Customer findById(@QueryParam("id")Integer id); }
5.客戶服務實現類
一個簡單的實現類, 不需要加額外註解, 註入dao從數據庫查詢數據返回(dao層代碼未貼出, 可自行實現)。
@Service @Transactional public class CustomerServiceImpl implements CustomerService { @Autowired private CustomerDao customerDao; @Override public Customer findById(Integer id) { // 調用dao, 從數據庫查詢客戶 return customerDao.findById(id); } }
二、webservice客戶端(消費者)
1.maven依賴
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!--cxf-jaxrs-starter--> <dependency> <groupId>org.apache.cxf</groupId> <artifactId>cxf-spring-boot-starter-jaxrs</artifactId> <version>3.2.0</version> </dependency> <!--jaxrs轉json工具--> <dependency> <groupId>com.fasterxml.jackson.jaxrs</groupId> <artifactId>jackson-jaxrs-json-provider</artifactId> <version>2.8.5</version> </dependency>
2.配置轉json工具
由於WebClient的create()方法需要的是List<Provider>形式的參數,所以創建一個繼承ArrayList類的JsonProvider,在構造方法中添加JacksonJaxbJsonProvider對象元素
@Component public class JsonProvider extends ArrayList<JacksonJaxbJsonProvider> { // 在構造方法中, 添加JacksonJaxbJsonProvider public JsonProvider(){ this.add(new JacksonJaxbJsonProvider()); } }
3.使用WebClient調用webservice服務
在Controller內註入上面創建的自定義的JsonProvider,並在WebClient調用create()方法時,作為方法參數註入,以此達到手動指定json轉換器的目的
@Controller public class CustomerController { // 註入配置的轉json工具 @Autowired private List<JacksonJaxbJsonProvider> jsonProvider; @RequestMapping("/customer_findById") @ResponseBody public List<Customer> findById(Integer id) { //調用webservice獲取查詢數據 Customer customer = WebClient .create("http://localhost:9001/services/customerService/findById?id="+id, jsonProvider) .accept(MediaType.APPLICATION_JSON).get(Customer.class); return customer; } }
三、其他註意
1.需要用xml/json格式轉換後傳輸的實體類要在類名上加一個註解
@XmlRootElement(name = "xxx")
2.上面demo使用的cxf-spring-boot-starter-jaxrs版本為3.2.0
在3.2.1以後的版本需要手動配置ViewResolver
否則會報錯:
@ConditionalOnProperty(spring.mvc.locale) did not find property ‘locale’ (OnPropertyCondition)
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- 一篇文章帶你瞭解Java SpringBoot四大核心組件
- springboot log4j2不能打印框架錯誤日志的解決方案
- 五分鐘解鎖springboot admin監控新技巧
- SpringBoot的依賴管理配置
- SpringBoot Admin 如何實現Actuator端點可視化監控