从天气项目看 Spring Cloud 微服务治理| |天气预报系统的微服务架构设计与实现

现有天气数据大而全,混杂了太多的功能,随着业务扩展,代码难以理解和维护,而且难以扩展,所以我们需要将天气预报单体项目进行拆分
对拆分也有一些需求:

  • 微服务的拆分足够小,每个微服务的业务时非常单一的
  • 微服务应能支持水平扩展(可以起多个未付实例)
  • 如果有需要,应能实现微服务间的相互调用

下面我们来介绍一下这篇主要要写的东西:

天气预报系统的架构设计

设计分为以下几个部分:

  1. 天气数据采集微服务的实现
  2. 城市数据API微服务的实现
  3. 天气数据API微服务的实现
  4. 天气预报微服务的实现

在这里插入图片描述

微服务代码的拆分

在这里插入图片描述

系统数据流向

在这里插入图片描述
流向从右至左:
1、城市数据API微服务通过本地XML文件可以获取城市数据
2、天气数据采集微服务通过调用第三方接口获取天气数据,获取天气数据时根据城市ID来查询相应的天气数据,所以天气数据采集微服务也依赖于城市API数据微服务
3、采集到天气数据后存入缓存中
4、天气数据API微服务从缓存中获取到天气数据,一部分可以直接给用户使用,另外一部分传递至天气预报微服务进行展示
5、天气预报微服务展示数据时,需要根据选定的城市来向客户端展示相应的天气情况

系统的通信设计

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

系统存储设计

在这里插入图片描述

代码实现:

msa-weather-collection-server

在这里插入图片描述
这个微服务通过Quartz调度器和Redis同步天气数据,也就是保留原有的Quartz,对于所依赖的城市数据,通过调用城市数据微服务来获取。

msa-weather-data-server

天气查询,把查询出数据的接口暴露给用户
在这里插入图片描述
只对天气数据查询,不同步

public interface WeatherDataService {
	/**
	 * 根据城市ID查询天气数据
	 * 
	 * @param cityId
	 * @return
	 */
	WeatherResponse getDataByCityId(String cityId);
	/**
	 * 根据城市名称查询天气数据
	 * 
	 * @param cityName
	 * @return
	 */
	WeatherResponse getDataByCityName(String cityName);
}

msa-weather-report-server

天气预报微服务:用户通过浏览器访问天气数据,会返回相应的UI界面(设计到前端的技术)
在这里插入图片描述
调用天气数据微服务来提供天气数据

@Service
public class WeatherReportServiceImpl implements WeatherReportService 
	@Override
	public Weather getDataByCityId(String cityId) {
		// TODO 改为由天气数据API微服务来提供
		Weather data = new Weather();
		data.setApi("81");
		data.setCity("深圳");
		data.setGanmao("容易感冒!多穿衣");
		data.setWendu("22");
		
		List<ForeCast> forecastList = new ArrayList<>();

        ForeCast forecast = new ForeCast();
		forecast.setDate("25日星期天");
		forecast.setType("晴");
		forecast.setFengxiang("无风");
		forecast.setHigh("高温 11度");
		forecastList.add(forecast);
		
		forecast = new ForeCast();
		forecast.setDate("26日星期天");
		forecast.setType("晴");
		forecast.setFengxiang("无风");
		forecast.setHigh("高温 11度");
		forecastList.add(forecast);
		
		forecast = new ForeCast();
		forecast.setDate("27日星期天");
		forecast.setType("晴");
		forecast.setFengxiang("无风");
		forecast.setHigh("高温 11度");
		forecastList.add(forecast);

		data.setForeCasts(forecastList);
		return data;
	}

}

通过调用城市数据API微服务查询出的天气数据来查询相应的天气数据

@RestController
@RequestMapping("/report")
public class WeatherReportController {
	private final static Logger logger = LoggerFactory.getLogger(WeatherReportController.class);  

	@Autowired
	private WeatherReportService weatherReportService;
	
	@GetMapping("/cityId/{cityId}")
	public ModelAndView getReportByCityId(@PathVariable("cityId") String cityId, Model model) throws Exception {
		// 获取城市ID列表
		// TODO 改为由城市数据API微服务来提供数据
		List<City> cityList = null;
		
		try {
			
			// TODO 改为由城市数据API微服务提供数据
			cityList = new ArrayList<>();
			City city = new City();
			city.setCityId("101280601");
			city.setCityName("深圳");
			cityList.add(city);
			
		} catch (Exception e) {
			logger.error("Exception!", e);
		}
		
		model.addAttribute("title", "老卫的天气预报");
		model.addAttribute("cityId", cityId);
		model.addAttribute("cityList", cityList);
		model.addAttribute("report", weatherReportService.getDataByCityId(cityId));
		return new ModelAndView("weather/report", "reportModel", model);
	}

}

msa-weather-city-server

在这里插入图片描述
只查询城市数据

@Service
public class CityDataServiceImpl implements CityDataService {
	@Override
	public List<City> listCity() throws Exception {
		// 读取XML文件
		Resource resource = new ClassPathResource("citylist.xml");
		BufferedReader br = new BufferedReader(new InputStreamReader(resource.getInputStream(), "utf-8"));
		StringBuffer buffer = new StringBuffer();
		String line = "";
		
		while ((line = br.readLine()) !=null) {
			buffer.append(line);
		}
		
		br.close();
		
		// XML转为Java对象
		CityList cityList = (CityList) XmlBuilder.xmlStrToOject(CityList.class, buffer.toString());
		return cityList.getCityList();
	}

}

posted @ 2023-05-25 12:45  _SpringCloud  阅读(92)  评论(0编辑  收藏  举报  来源