Spring Boot 发起 HTTP 请求

起步

新年目标Spring Cloud开始实施,打开慕课网。html

clipboard.png

刚学了一章,大致就是调用中国天气网的api,使用Spring Boot构建本身的天气预报系统,而后使用Spring Cloud,一步一步使用微服务的思想来演进架构。java

小目标spring

昨天去百度抢了新年红包,感叹百度的高并发作的也是如此优秀。apache

阿里的双十一,百度的新年。(发现了两者的共同点,可能也是解决并发的一种思路,并发的时候只容许增长数据。)json

感叹归感叹,指望着学习完Spring Cloud也能设计出优秀的架构,解决并发的一些问题。api

遇到的问题

学习时也跟着课程进行编码,讲师讲的很是好,可是本课程的重点是后面的微服务架构,因此前面的功能有一些瑕疵,特此提出本身的实现,供你们学习交流。架构

功能描述

最初的功能很简单,由于后台是没有任何数据的,因此前台有请求,就直接去天气网要数据,而后再返回去。并发

clipboard.png

数据序列化问题

这是天气网api返回来的数据格式,乍一看没啥毛病。mvc

{
    "data": {
        "yesterday": {
            "date": "4日星期一", 
            "high": "高温 26℃", 
            "fx": "无持续风向", 
            "low": "低温 18℃", 
            "fl": "<![CDATA[<3级]]>", 
            "type": "多云"
        }, 
        "city": "深圳", 
        "forecast": [
            {
                "date": "5日星期二", 
                "high": "高温 25℃", 
                "fengli": "<![CDATA[<3级]]>", 
                "low": "低温 18℃", 
                "fengxiang": "无持续风向", 
                "type": "多云"
            }, 
            {
                "date": "6日星期三", 
                "high": "高温 26℃", 
                "fengli": "<![CDATA[<3级]]>", 
                "low": "低温 17℃", 
                "fengxiang": "无持续风向", 
                "type": "多云"
            }, 
            {
                "date": "7日星期四", 
                "high": "高温 27℃", 
                "fengli": "<![CDATA[<3级]]>", 
                "low": "低温 18℃", 
                "fengxiang": "无持续风向", 
                "type": "多云"
            }, 
            {
                "date": "8日星期五", 
                "high": "高温 26℃", 
                "fengli": "<![CDATA[<3级]]>", 
                "low": "低温 17℃", 
                "fengxiang": "无持续风向", 
                "type": "多云"
            }, 
            {
                "date": "9日星期六", 
                "high": "高温 24℃", 
                "fengli": "<![CDATA[<3级]]>", 
                "low": "低温 14℃", 
                "fengxiang": "无持续风向", 
                "type": "小雨"
            }
        ], 
        "ganmao": "相对今天出现了较大幅度降温,较易发生感冒,体质较弱的朋友请注意适当防御。", 
        "wendu": "23"
    }, 
    "status": 1000, 
    "desc": "OK"
}

缺点1:有拼音;ganmaowenduapp

缺点2:名称不一致;理论上来讲yesterdayforecast应该是同一个实体,都表示一天的天气状况,只是名称不一样。可是在yesterday中,风向和风力是fxfl,在forecast中,名称倒是fenglifengxiang

解决此问题,想到的思路就是使用jackson进行序列化与反序列化时进行配置的一些注解。

最初使用此种方法实现:

@JsonProperty("wendu")
private Float temperature;

一个对象中的名字,一个json数据中的名字。

能够实现,可是很差。

举个例子,天气api返回给我wendu,添加了@JsonProperty,而后wendu就绑定到了temperature上,可是若是我前台再返回该对象,序列化后生成的名称仍是wendu。很差!

目标是实现,反序列化时:从wendu能绑定到个人temperature,序列化时直接使用个人字段名。

get、set尝试

猜想是否是和getset方法有关。

就把@JsonProperty("wendu")添加到set方法上,发现并无用。

JsonAlias

后来通过查询,原来是注解用错了,此种状况应使用别名。

关于JsonPropertyJsonAlias的详细讲解,请参考Jackson @JsonProperty and @JsonAlias Example

@JsonAlias("wendu")
private Float temperature;

同时,能够应用多个别名:

@JsonAlias({"fengli", "fl"})
private String windForce;

发起请求

发起请求的示例代码,供之后参考。

@Autowired
private RestTemplate restTemplate;

@Override
public Weather getWeatherByCityName(String cityName) {
    return this.getWeatherByUrl(BASE_URL + "?" + CITY_NAME + "=" + cityName)
            .getData();
}

private Response getWeatherByUrl(String url) {
    // 发起Get请求
    ResponseEntity<String> response = restTemplate.getForEntity(url, String.class);
    // 若是状态码非200, 抛异常
    if (response.getStatusCodeValue() != 200) {
        throw new YunzhiNetworkException("数据请求失败");
    }
    // 实例化对象映射对象
    ObjectMapper mapper = new ObjectMapper();
    // 初始化响应数据
    Response data;
    // 从字符串转换为Response对象
    try {
        data = mapper.readValue(response.getBody(), Response.class);
    } catch (IOException e) {
        throw new YunzhiIOException("json数据转换失败");
    }
    // 返回
    return data;
}

RestTemplate配置

这里与正常的RestTemplate构建有些不一样,一般的RestTemplate是使用Spring工具类构造的,此处使用ApacheHttp组件构造,以支持更多的数据格式。

implementation 'org.apache.httpcomponents:httpclient'

同时去除了默认的对StringHttp消息转换器,默认的转换器使用的不是UTF-8编码。

讲师原文章:Spring RestTemplate 调用天气预报接口乱码的解决

@Configuration
public class BeanConfiguration {

    @Bean
    public RestTemplate restTemplate() {
        // 使用Apache HttpClient构建RestTemplate, 支持的比Spring自带的更多
        RestTemplate restTemplate = new RestTemplate(new HttpComponentsClientHttpRequestFactory());
        // 去除默认的String转换器
        restTemplate.getMessageConverters().removeIf(converter -> converter instanceof StringHttpMessageConverter);
        // 添加自定义的String转换器, 支持UTF-8
        restTemplate.getMessageConverters()
                .add(new StringHttpMessageConverter(StandardCharsets.UTF_8));
        return restTemplate;
    }
}

更完善的单元测试

同时,在编写单元测试的时候,看了一篇关于AssertJ的文章。Testing with AssertJ assertions - Tutorial

以前学Junit5的时候,以为这个东西挺好使的啊?为何被开源社区抛弃而使用AssertJ呢?

原来以前用的断言都太简单,其实AssertJ远比咱们使用的更强大。

@Test
public void getWeatherByCityName() throws Exception {
    final String cityName = "深圳";
    MvcResult mvcResult = this.mockMvc
            .perform(MockMvcRequestBuilders.get(BASE_URL + "/cityName/" + cityName))
            .andExpect(MockMvcResultMatchers.status().isOk())
            .andReturn();
    String json = mvcResult.getResponse().getContentAsString();
    Assertions.assertThat(json)
            .contains("cityName", "cold", "temperature", "windDirection", "windForce")
            .doesNotContain("ganmao", "wendu", "fx", "fl", "fengxiang", "fengli");
}

总结

多看英文文章, Tutorial写得都特别好。
相关文章
相关标签/搜索