基于SpringBoot的疫情信息统计平台【完整项目源码】

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 基于SpringBoot的疫情信息统计平台【完整项目源码】

Part 1


【学习目标】


1) SpringBoot框架 2) 业务功能的开发思路 3) 爬虫的底层原理 4) E-chart图标展示


【项目的诞生】


项目成员:项目经理 PM 产品经理 PD、 UI设计师 UED、 前端工程师 FE、后端工程师 RD、 测试工程师 QA、运维工程师 OP


【功能的诞生】


产品经理 -> 需求评审会 -> UI设计师(交互) -> UI评审 -> 技术 -> 技术方案设计 -> 开发(1/3) -> 测试和修改




2、postman(模拟http请求的工具)


验证分析出来的请求地址,在排除上下文环境后,是否依然能够拿到数据。


比如有的请求,依赖cookie、依赖动态的参数等等


Part 2


【解析数据】


(一) 认识JSON


JSON = JavaScript Object Notation (JavaScript对象表示法)


本质上,是存储和交换文本信息的语法。是一种轻量级的文本数据格式。


java领域内解析json的工具:gson 、 fastjson 、jackson


JSON 和 Java实体类image.png


image.pngimage.pngimage.png

(二) Gson


是google推出的,用来在json数据和java对象之间进行转换的类库。


Gson  gson = new Gson();

Gson  gson1 = new GsonBuilder().create();


// 将对象obj转化为json字符串

String jsonStr = gson.toJson(Obj);

// 将json字符串转化为java对象

T obj = gson.fromJson(jsonStr,class)

使用方式


1)引入gson依赖


       <dependency>

           <groupId>com.google.code.gson</groupId>

           <artifactId>gson</artifactId>

           <version>2.8.6</version>

       </dependency>

     

2)确认要转化的数据格式


import lombok.AllArgsConstructor;

import lombok.Data;


@Data @AllArgsConstructor

public class DataBean {


   private String area;

   private int nowConfirm;

   private int confirm;

   private int heal;

   private int dead;

}

3)解析文本数据


public class DataHandler {


   public static void main(String[] args) throws Exception{

       getData();

   }



   public static List<DataBean> getData() throws Exception {

//        Gson gson = new Gson();

       Gson gson1 = new GsonBuilder().create();

//        Map map = gson.fromJson(testStr,Map.class);

//        System.out.println(map);



       // 读取文件中的文本内容   然后再转化为java对象

//        File file = new File("tmp.txt");


       FileReader fr = new FileReader("tmp.txt");

       char[] cBuf = new char[1024];

       int cRead = 0;

       StringBuilder builder = new StringBuilder();

       while ((cRead = fr.read(cBuf)) > 0) {

           builder.append(new String(cBuf, 0, cRead));

       }


       fr.close();


//        System.out.println(builder.toString());

       Gson gson = new Gson();

       Map map = gson.fromJson(builder.toString(), Map.class);

       System.out.println(map);


       ArrayList areaList = (ArrayList) map.get("areaTree");

       Map dataMap = (Map) areaList.get(0);

       ArrayList childrenList = (ArrayList) dataMap.get("children");


       // 遍历然后转化

       List<DataBean> result = new ArrayList<>();


       for (int i = 0; i < childrenList.size(); i++) {

           Map tmp = (Map) childrenList.get(i);

           String name = (String)tmp.get("name");


           Map totalMap = (Map) tmp.get("total");

           double nowConfirm = (Double)totalMap.get("nowConfirm");

           double confirm = (Double)totalMap.get("confirm");

           double heal = (Double)totalMap.get("heal");

           double dead = (Double)totalMap.get("dead");


           DataBean dataBean = new DataBean(name,(int)nowConfirm,(int)confirm,

                   (int)heal,(int)dead);


           result.add(dataBean);

       }


       System.out.println(result);


       return result;

   }

}

Part 3


(三) 将数据展示在页面中


image.png


1、编写service和controller


public interface DataService extends IService<DataBean> {


   List<DataBean> list();


}

@Service

public class DataServiceImpl implements DataService {


   @Override

   public List<DataBean> list() {

       List<DataBean> result = null;


       try {

           result = DataHandler.getData();

       } catch (Exception e) {

           e.printStackTrace();

       }

       return result;

   }

}

@Controller

public class DataController {


   @Autowired

   private DataService dataService;


   @GetMapping("/")

   public String list(Model model){

       List<DataBean> list = dataService.list();

       model.addAttribute("dataList",list);

       return "list";

   }

 

}

2、编写页面(注意已引入thymeleaf的maven依赖)


<!DOCTYPE html>

<html lang="en" xmlns:th="http://www.thymeleaf.org">

<head>

   <meta charset="UTF-8">

   <title>Title</title>

</head>

<body>



<h2>国内疫情情况如下</h2>

<br>


<table>

   <thead>

   <tr>

       <th>地区</th>

       <th>现有确诊</th>

       <th>累计确诊</th>

       <th>治愈</th>

       <th>死亡</th>

   </tr>

   </thead>

   <tbody>

   <tr th:each="data:${dataList}">

       <td th:text="${data.area}">name</td>

       <td th:text="${data.nowConfirm}">nowConfirm</td>

       <td th:text="${data.confirm}">confirm</td>

       <td th:text="${data.heal}">heal</td>

       <td th:text="${data.dead}">dead</td>

   </tr>

   </tbody>

</table>


</body>

</html>


(四) 转为实时数据


涉及知识点:用java代码模拟http请求


1、复习get和post请求


分别在使用场景、参数传递方式、数据大小限制、安全性等方面的异同


2、HttpURLConnection


连接时间和读取时间

连接时间:发送请求端 连接到  url目标地址端的时间

          受到距离长短和网络速度的影响

读取时间:指连接成功后  获取数据的时间

          受到数据量和服务器处理速度的影响

1) 通过创建url打开远程链接 (HttpURLConnection) 2) 设置相关参数(超时时间和请求头等等) 3) 发送请求 4) 接收结果(使用InputStream和BufferedReader)


public static String doGet(String urlStr) {

       HttpURLConnection conn = null;

       InputStream is = null;

       BufferedReader br = null;

       StringBuilder result = new StringBuilder();


       try {

           URL url = new URL(urlStr);

           // 通过url打开一个远程连接  强转类型

           conn = (HttpURLConnection) url.openConnection();

           conn.setRequestMethod("GET");


           // 连接时间和读取时间

           //  连接时间:发送请求端 连接到  url目标地址端的时间

           //            受到距离长短和网络速度的影响

           //  读取时间:指连接成功后  获取数据的时间

           //            受到数据量和服务器处理速度的影响

           conn.setConnectTimeout(15000);

           conn.setReadTimeout(60000);


           // 设定请求头参数的方式:如指定接收json数据   服务端的key值为content-type

//            conn.setRequestProperty("Accept", "application/json");


           // 发送请求

           conn.connect();


           if (conn.getResponseCode() != 200) {

               // TODO 此处应该增加异常处理

               return "error code";

           }


           is = conn.getInputStream();

           br = new BufferedReader(new InputStreamReader(is, "UTF-8"));

           String line;

           // 逐行读取  不为空就继续

           while ((line = br.readLine()) != null) {

               result.append(line);

               System.out.print(line);

           }



       } catch (Exception e) {

           e.printStackTrace();

       } finally {

           try {

               if (br != null) br.close();

               if (is != null) is.close();


           } catch (Exception e) {

               e.printStackTrace();

           }

       }


       return result.toString();

   }


Part 4


(五) 使用Jsoup解析html格式数据


1、Jsoup 是html的解析器,可以解析html文本和直接解析URL地址。本质上是通过DOM、CSS以及类似jQuery的方法来取出和操作数据。


       Document document = Jsoup.parse(htmlStr);

       System.out.println(document);


       //通过标签名找到元素

       Elements elements = document.getElementsByTag("p");

       System.out.println(elements);


       document.getElementsById    //通过id找到元素

       Element element = document.select("a[href]");  

       //还支持使用正则表达式查找元素

2、提供不同数据源的切换查询


1)增加了controller方法


   @GetMapping("/list/{id}")

   public String listById(Model model, @PathVariable String id) {

       List<DataBean> list = dataService.listById(Integer.parseInt(id));

       model.addAttribute("dataList", list);

       return "list";

   }

@PathVariavle 将接收到的地址数据,映射到方法的参数中


2)完善service


   @Override

   public List<DataBean> listById(int id) {

       if (id == 2) {

           return JsoupHandler.getData();

       }

       return list();

   }

处理数据的方法


public static String urlStr = "https://ncov.dxy.cn/ncovh5/view/pneumonia?" +

           "scene=2&from=singlemessage&isappinstalled=0";


   public static ArrayList<DataBean> getData() {


       ArrayList<DataBean> result = new ArrayList<>();

       try {

           Document doc = Jsoup.connect(urlStr).get();

//            Elements scripts = doc.select("script");

           // 找到指定的标签数据

           Element oneScript = doc.getElementById("getAreaStat");


           String data = oneScript.data();

           // 字符串截取出json格式的数据

           String subData = data.substring(data.indexOf("["),

                   data.lastIndexOf("]") + 1);


//            System.out.println(subData);


           Gson gson = new Gson();

           ArrayList list = gson.fromJson(subData, ArrayList.class);


           for (int i = 0; i < list.size(); i++) {

               Map map = (Map) list.get(i);

               String name = (String) map.get("provinceName");

               double nowConfirm = (Double) map.get("currentConfirmedCount");

               double confirm = (Double) map.get("confirmedCount");

               double heal = (Double) map.get("curedCount");

               double dead = (Double) map.get("deadCount");


               DataBean dataBean = new DataBean(name, (int) nowConfirm, (int) confirm

                       , (int) heal, (int) dead);

               result.add(dataBean);

           }


       } catch (Exception e) {

           e.printStackTrace();

       }


       return result;

   }

(六) 增加数据存储逻辑


1、引入相关的依赖


       <dependency>

           <groupId>org.mybatis.spring.boot</groupId>

           <artifactId>mybatis-spring-boot-starter</artifactId>

           <version>2.1.0</version>

       </dependency>


       <dependency>

           <groupId>mysql</groupId>

           <artifactId>mysql-connector-java</artifactId>

           <scope>runtime</scope>

       </dependency>


       <dependency>

           <groupId>com.baomidou</groupId>

           <artifactId>mybatis-plus-boot-starter</artifactId>

           <version>3.2.0</version>

       </dependency>

2、配置数据库


spring.datasource.url=jdbc:mysql://localhost:3366/epidemic?serverTimezone=UTC&useUnicode=true&characterEncoding=utf8 //3366是MySQL数据库端口号

spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver

spring.datasource.username=root //MySQL数据库管理用户名

spring.datasource.password=root //MySQL数据库管理用户密码

3、使用mybatis-plus进行增删改查的操作


1) 创建mapper


public interface DataMapper extends BaseMapper<DataBean> {


}

扫描mapper的注解 在主程序入口类中添加


@MapperScan("com.duing.mapper")

创建service及其实现类


注意泛型是要处理的实体类


public interface DataService extends IService<DataBean> {


}

// 泛型 分别是mapper 以及 实体类



@Service

public class DataServiceImpl extends ServiceImpl<DataMapper,DataBean>

       implements DataService {

     

}

4)改造实体类databean


此时要满足,存在无参构造器以及可被序列化 同时指定具体映射的表名 通过@TableName


@Data @AllArgsConstructor

@NoArgsConstructor

@TableName("illness")

public class DataBean implements Serializable {


   private String area;

   private int nowConfirm;

   private int confirm;

   private int heal;

   private int dead;

}

4、初始化数据存储的逻辑


@PostConstruct 修饰的方法,在服务器加载Servlet时运行,而且只执行一次


改造逻辑:首先将DataHandler声明为组件 @Component


   @Autowired

   private DataService dataService;

 

   @PostConstruct

   public void saveData() {

       try {

           List<DataBean> dataBeans = getData();

           // 先将数据清空  然后存储数据

           dataService.remove(null);

           dataService.saveBatch(dataBeans);


       } catch (Exception e) {

           e.printStackTrace();

       }


   }

@Scheduled


使用前需要在主程序入口类上打开开关


@EnableScheduling

在方法上使用注解的参数


1) @Scheduled(fixedRate = 10000) 指定频率的执行任务 从方法执行开始就计时 假设方法执行5s 那么第一次执行开始过了10s后,开始第二次执行


2) @Scheduled(fixedDelay = 10000) 指定间隔的执行任务 从方法执行完成开始计时 假设方法执行5s 那么第一次执行完成过了10s后,开始第二次执行


3) cron表达式 —— 计划执行的表达式 把六个位置用空格分隔,指代不同单位的时间,执行的规律 秒、分钟、小时、日期、月份、星期、(年,可选)


   // 配置定时执行的注解  支持cron表达式  

   // 每分钟执行一次  更改可参考cron表达式生成网站

   @Scheduled(cron = "0 0/1 * * * ?")

   public void updateData() {

       System.out.println("更新数据");

       saveData();

   }

Part 5


【展示数据Echarts】


是由百度前端技术部开发,基于js的数据可视化图表库。


https://echarts.apache.org/examples/zh/index.html#chart-type-line


image.png


分析图形展示的数据来源,然后请求数据后转化成我们需要的格式,传递给页面,通过Echarts渲染出来。


(一)折线图


image.png


1)分析的请求地址


https://view.inews.qq.com/g2/getOnsInfo?name=disease_other


可以获得json格式的数据,数据的key是chinaDayList


image.png


2)模拟请求


HttpClient使用,应用最广泛的处理http请求的工具。


<dependency>

   <groupId>org.apache.httpcomponents</groupId>

   <artifactId>httpclient</artifactId>

   <version>4.5.12</version>

</dependency>

public static String doGet(String urlStr) {


       // 提供了 闭合的httpclient对象

       CloseableHttpClient httpClient = null;

       // 也提供了  闭合的响应对象

       CloseableHttpResponse response = null;


       String result = null;


       try {

           // 使用默认创建方式

           httpClient = HttpClients.createDefault();

           // 创建一个get请求  传入url

           HttpGet httpGet = new HttpGet(urlStr);

           // 设置请求头的方式

           httpGet.addHeader("Accept", "application/json");


           // 设置请求参数  连接时间、数据读取时间(socketTimeOut)等  单位是ms

           //   ConnectionRequestTimeout  指从共享连接池中取出连接的超时时间

           RequestConfig requestConfig = RequestConfig.custom()

                   .setConnectTimeout(35000)

                   .setConnectionRequestTimeout(35000)

                   .setSocketTimeout(60000)

                   .build();


           // 设置配置参数

           httpGet.setConfig(requestConfig);

           // 执行请求

           response = httpClient.execute(httpGet);

           // 从返回对象中获取返回数据

           HttpEntity entity = response.getEntity();


           result = EntityUtils.toString(entity);


       }catch (Exception e){

           e.printStackTrace();

       }


       return result;

   }

解析出数据


public class GraphHandler {


   public static String urlStr = "https://view.inews.qq.com/g2/getOnsInfo?name=disease_other";


   public static List<GraphBean> getGraphData() {

       List<GraphBean> result = new ArrayList<>();


       String str = HttpClientUtil.doGet(urlStr);


       Gson gson = new Gson();

       Map map = gson.fromJson(str, Map.class);


       String subStr = (String) map.get("data");

       Map subMap = gson.fromJson(subStr, Map.class);


       ArrayList list = (ArrayList) subMap.get("chinaDayList");


       for (int i = 0; i < list.size(); i++) {

           Map tmp = (Map)list.get(i);


           String date = (String)tmp.get("date");

           double nowConfirm = (Double)tmp.get("nowConfirm");

           GraphBean graphBean = new GraphBean(date,(int)nowConfirm);

           result.add(graphBean);

       }


       return result;

   }


}

数据结构


@Data

@AllArgsConstructor

public class GraphBean {


   private String date;

   private int nowConfirm;

}

3)返回给页面渲染


   @GetMapping("/graph")

   public String graph(Model model) {

       List<GraphBean> list = GraphHandler.getGraphData();

       //  进一步改造数据格式

       //  因为前端需要的数据是  x轴所有数据的数组和y轴所有数据的数组


       ArrayList<String> dateList = new ArrayList<>();

       ArrayList<Integer> nowConfirmList = new ArrayList<>();


       for (int i = 0; i < list.size(); i++) {

           GraphBean graphBean = list.get(i);

           dateList.add(graphBean.getDate());

           nowConfirmList.add(graphBean.getNowConfirm());

       }


       model.addAttribute("dateList", new Gson().toJson(dateList));

       model.addAttribute("nowConfirmList", new Gson().toJson(nowConfirmList));

       return "graph";

   }

<!DOCTYPE html>

<html lang="en" xmlns:th="http://www.thymeleaf.org">

<head>

   <meta charset="UTF-8">

   <title>Title</title>

   <script type="text/javascript" src="echarts/echarts.min.js"></script>

</head>

<body>



<!-- 为 ECharts 准备一个具备大小(宽高)的 DOM -->

<div id="main" style="width: 600px;height:400px;"></div>


<!--在js中接收服务端返回数据-->

<script th:inline="javascript">

   // 基于准备好的dom,初始化echarts实例

   var myChart = echarts.init(document.getElementById('main'));


   var dateStr = [[${dateList}]];

   var nowConfirmStr = [[${nowConfirmList}]];


   // 指定图表的配置项和数据

   var option = {

       title: {  // 标题组件

           text: '全国现有确诊趋势'

       },

       tooltip: {  // 提示框组件

           trigger: 'axis'

       },

       legend: {  // 曲线含义说明

           data: ['现有确诊']

       },

       xAxis: {

           // 转化为json对象

           data: JSON.parse(dateStr)

       },

       yAxis: {

           type: 'value'

       },

       series: [{

           name: '现有确诊',

           data: JSON.parse(nowConfirmStr),

           type: 'line'

       }]

   };


   // 使用刚指定的配置项和数据显示图表。

   myChart.setOption(option);

</script>


</body>

</html>

Echars教程地址:[教程地址](https://echarts.apache.org/zh/tutorial.html#5 分钟上手 ECharts)


准备dom -> 通过js渲染数据 -> 使用[[${ sth }]] 接收服务端数据

-> 使用JSON.parse()解析json字符串 -> 获得渲染结果


Part 6


(二)折线图2


image.png


相关逻辑在 GraphAddBean对应的代码中


处理数据 -> 转化格式 -> 返回数据给echarts渲染


GraphHandler DataController *.html


1)GraphHandler


public static List<GraphAddBean> getGraphAddData(String str) {


       List<GraphAddBean> result = new ArrayList<>();


       Gson gson = new Gson();

       Map map = gson.fromJson(str, Map.class);


       String subStr = (String) map.get("data");

       Map subMap = gson.fromJson(subStr, Map.class);


       ArrayList list = (ArrayList) subMap.get("chinaDayAddList");


       for (int i = 0; i < list.size(); i++) {

           Map tmp = (Map) list.get(i);

           String date = (String) tmp.get("date");

           double addConfirm = (Double) tmp.get("confirm");

           double addSuspect = (Double) tmp.get("suspect");


           GraphAddBean graphAddBean = new GraphAddBean(date,

                   (int) addConfirm, (int) addSuspect);

           result.add(graphAddBean);

       }


       return result;

   }

2)DataController


@GetMapping("/graphAdd")

   public String graphAdd(Model model) {

       List<GraphAddBean> list = GraphHandler.getGraphAddData();


       ArrayList<String> dateList = new ArrayList<>();

       ArrayList<Integer> addConfirmList = new ArrayList<>();

       ArrayList<Integer> addSuspectList = new ArrayList<>();


       for (int i = 0; i < list.size(); i++) {

           GraphAddBean graphAddBean = list.get(i);

           dateList.add(graphAddBean.getDate());

           addConfirmList.add(graphAddBean.getAddConfirm());

           addSuspectList.add(graphAddBean.getAddSuspect());

       }


       model.addAttribute("dateList", new Gson().toJson(dateList));

       model.addAttribute("addConfirmList", new Gson().toJson(addConfirmList));

       model.addAttribute("addSuspectList", new Gson().toJson(addSuspectList));

       return "graphAdd";

   }

3)HTML


增加折线时,主要在legend和series中增加对应元素


<!DOCTYPE html>

<html lang="en" xmlns:th="http://www.thymeleaf.org">

<head>

   <meta charset="UTF-8">

   <title>Title</title>

   <script type="text/javascript" src="echarts/echarts.min.js"></script>

</head>

<body>



<!-- 为 ECharts 准备一个具备大小(宽高)的 DOM -->

<div id="main" style="width: 600px;height:400px;"></div>


<!--在js中接收服务端返回数据-->

<script th:inline="javascript">

   // 基于准备好的dom,初始化echarts实例

   var myChart = echarts.init(document.getElementById('main'));


   var dateStr = [[${dateList}]];

   var addConfirmStr = [[${addConfirmList}]];

   var addSuspectStr = [[${addSuspectList}]];


   // 指定图表的配置项和数据

   var option = {

       title: {  // 标题组件

           text: '全国疫情新增趋势'

       },

       tooltip: {  // 提示框组件

           trigger: 'axis'

       },

       legend: {  // 曲线含义说明

           data: ['新增确诊', '新增疑似']

       },

       xAxis: {

           // 转化为json对象

           data: JSON.parse(dateStr)

       },

       yAxis: {

           type: 'value'

       },

       series: [{

           name: '新增确诊',

           data: JSON.parse(addConfirmStr),

           type: 'line'

       }, {

           name: '新增疑似',

           data: JSON.parse(addSuspectStr),

           type: 'line'

       }]

   };


   // 使用刚指定的配置项和数据显示图表。

   myChart.setOption(option);

</script>


</body>

</html>

(三)柱状图


image.png


先分析数据的来源 -> 经过对数据的处理和计算 -> 发送给前端组件进行渲染


对应在GraphColumnarBean相关的逻辑中


特别之处在于 拿到数据之后需要排序


@Data

@AllArgsConstructor

public class GraphColumnarBean implements Comparable<GraphColumnarBean> {


   private String area;

   private int fromAbroad;


   @Override

   public int compareTo(GraphColumnarBean o) {

       return o.getFromAbroad() - this.getFromAbroad();

   }

}

排序之后将前十的数据返回


   @GetMapping("/graphColumnar")

   public String graphColumnar(Model model) {

       List<GraphColumnarBean> list = GraphHandler.getGraphColumnarData();

       Collections.sort(list);


       ArrayList<String> nameList = new ArrayList<>();

       ArrayList<Integer> fromAbroadList = new ArrayList<>();


       for (int i = 0; i < 10; i++) {

           GraphColumnarBean bean = list.get(i);

           nameList.add(bean.getArea());

           fromAbroadList.add(bean.getFromAbroad());

       }


       model.addAttribute("nameList", new Gson().toJson(nameList));

       model.addAttribute("fromAbroadList", new Gson().toJson(fromAbroadList));

       return "graphColumnar";

   }

柱状图的数据示例


// 指定图表的配置项和数据

   var option = {

       title: {  // 标题组件

           text: '境外输入省市TOP10'

       },

       tooltip: {  // 提示框组件

           trigger: 'axis'

       },

       xAxis: {

           // 转化为json对象

           data: JSON.parse(nameStr)

       },

       yAxis: {

           type: 'value'

       },

       series: [

           {

               name: '境外输入',

               type: 'bar',

               barWidth: '60%',

               data: JSON.parse(fromAbroadStr)

           }

       ]

   };

(四)饼状图


image.png


对应在GraphPieBean相关的逻辑中


graphPie.html


   // 指定图表的配置项和数据

   var option = {

       title: {  // 标题组件

           text: '全国现有确诊构成'

       },

       tooltip: {  // 提示框组件

           trigger: 'axis'

       },

       series: [

           {

               type: 'pie',

               radius: '55%',

               center: ['50%', '60%'],

               data: JSON.parse(str)

           }

       ]

   };

Part 7


(五) 中国地图


image.png


引入新的js 【china.js】 数据来源已存储到表格中


map.html


<!DOCTYPE html>

<html lang="en" xmlns:th="http://www.thymeleaf.org">

<head>

   <meta charset="UTF-8">

   <title>Title</title>


   <script type="text/javascript" src="echarts/echarts.min.js"></script>

   <script type="text/javascript" src="echarts/china.js"></script>

</head>

<body>



<div id="main" style="width: 1000px;height:700px;"></div>

<script th:inline="javascript">

   var dataStr = [[${mapData}]];


   option = {

       title: {

           text: '疫情地图',

           subtext: '仅供参考',

           x: 'center'

       },

       tooltip: {

           trigger: 'item'

       },


       series: [

           {

               name: '现存确诊',

               type: 'map',

               mapType: 'china',

               roam: false,

               label: {

                   normal: {

                       // formatter: '{b}',

                       position: 'center',

                       show: true,

                       textStyle: {

                           color: 'rgba(0,0,0,0.4)'

                       }

                   },

                   emphasis: {

                       show: true

                   }

               },

               data: JSON.parse(dataStr)

           }

       ]

   };

   var chart = echarts.init(document.getElementById("main"))

   chart.setOption(option)


</script>


</body>

</html>

【国际化】


是对thymeleaf中消息表达式的一种应用, #{} , 提供的是对配置文件中信息的读取。


(一) 使用浏览器识别语种


1) 在resources目录下,创建mg文件夹,再继续创建 *.properties文件。


设置其中的key和value (注意:此时的编码格式需要在idea的设置中确认)


list.title = 疫情最新动态

list.h2 = 国内疫情情况如下

list.table.name1 = 地区

list.table.name2 = 现有确诊

list.table.name3 = 累计确诊

list.table.name4 = 治愈

list.table.name5 = 死亡

创建其他语种对应的配置文件,如 *_en_US.properties 和 *_zh_CN.properties


3)在html中更改对key值的引用方式,使用消息表达式


4)让spring找到国际化文件对应的位置,在application.properties中


spring.messages.basename = mg.list

验证方式 通过更改浏览器中 【设置】 - 【高级】 - 【语言】 - 【找到英语(美国)】,可以通过置顶,切换中英文显示。


本质原因是,因为请求的请求头中,会设置不同的Accept-Language的值。


(二)自定义切换语种


使用spring提供的国际化使用类 LocaleResolver


1) 页面中增加按钮


   <label>

       <a class="btn btn-sm" th:href="@{/(lan='zh_CN')}">中文</a>

       <a class="btn btn-sm" th:href="@{/(lan='en_US')}">英文</a>

   </label>

自定义LocaleResolver类的实现


public class MyLocaleResolver implements LocaleResolver {


   @Override

   public Locale resolveLocale(HttpServletRequest httpServletRequest) {

       String lan = httpServletRequest.getParameter("lan");

       Locale locale = Locale.getDefault();

       if (!StringUtils.isEmpty(lan)) {

           String[] split = lan.split("_");

           locale = new Locale(split[0], split[1]);

       }

       return locale;

   }


   @Override

   public void setLocale(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Locale locale) {


   }

}

2)注入spring


@Configuration

public class MyConfig {


   @Bean

   public LocaleResolver localeResolver() {

       return new MyLocaleResolver();

   }

}

验证 此时切换按钮时,可以看到不同语种的显示image.png


image.png

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
3天前
|
Java 数据库连接 Maven
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
自动装配是现在面试中常考的一道面试题。本文基于最新的 SpringBoot 3.3.3 版本的源码来分析自动装配的原理,并在文未说明了SpringBoot2和SpringBoot3的自动装配源码中区别,以及面试回答的拿分核心话术。
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
|
14天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
79 13
|
22天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
3月前
|
前端开发 JavaScript Java
基于Java+Springboot+Vue开发的服装商城管理系统
基于Java+Springboot+Vue开发的服装商城管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的服装商城管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。
192 2
基于Java+Springboot+Vue开发的服装商城管理系统
|
2月前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 实现动态路由和菜单功能,快速搭建前后端分离的应用框架
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 实现动态路由和菜单功能,快速搭建前后端分离的应用框架。首先,确保开发环境已安装必要的工具,然后创建并配置 Spring Boot 项目,包括添加依赖和配置 Spring Security。接着,创建后端 API 和前端项目,配置动态路由和菜单。最后,运行项目并分享实践心得,包括版本兼容性、安全性、性能调优等方面。
190 1
|
3月前
|
前端开发 JavaScript Java
SpringBoot项目部署打包好的React、Vue项目刷新报错404
本文讨论了在SpringBoot项目中部署React或Vue打包好的前端项目时,刷新页面导致404错误的问题,并提供了两种解决方案:一是在SpringBoot启动类中配置错误页面重定向到index.html,二是将前端路由改为hash模式以避免刷新问题。
330 1
|
1月前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个具有动态路由和菜单功能的前后端分离应用。
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个具有动态路由和菜单功能的前后端分离应用。首先,创建并配置 Spring Boot 项目,实现后端 API;然后,使用 Ant Design Pro Vue 创建前端项目,配置动态路由和菜单。通过具体案例,展示了如何快速搭建高效、易维护的项目框架。
119 62
|
1月前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能。首先,确保开发环境已安装必要的工具,然后创建并配置 Spring Boot 项目,包括添加依赖和配置 Spring Security。接着,创建后端 API 和前端项目,配置动态路由和菜单。最后,运行项目并分享实践心得,帮助开发者提高开发效率和应用的可维护性。
102 2
|
1月前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
3月前
|
前端开发 JavaScript Java
基于Java+Springboot+Vue开发的大学竞赛报名管理系统
基于Java+Springboot+Vue开发的大学竞赛报名管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的大学竞赛报名管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。
239 3
基于Java+Springboot+Vue开发的大学竞赛报名管理系统