# 课程安排 - 路线规划需求分析 - 熟悉路线规划项目工程 - 实现机构数据同步 - 实现路线管理 - 综合功能测试 # 1、背景说明 通过前面的Neo4j的学习,你已经基本掌握了Neo4j的使用,接下来就需要应用Neo4j来实现路线规划微服务了。 目前路线规划微服务中的代码框架基本上已经写好了,但是具体的实现并没有编写,所以就需要你来填充这些关键核心的代码。 ![lg.gif](https://cdn.nlark.com/yuque/0/2022/gif/27683667/1661393495230-5ed7dd64-d9ed-4911-af64-6ac745e62281.gif#averageHue=%23f2eddf&clientId=ufeadf2ce-7102-4&errorMessage=unknown%20error&from=paste&height=145&id=ufdf67f2b&name=lg.gif&originHeight=240&originWidth=240&originalType=binary&ratio=1&rotation=0&showTitle=false&size=25250&status=error&style=none&taskId=u6ffe84dc-97a5-4342-a53c-be75fb9f891&title=&width=145.45453704749963) # 2、需求分析 对于运输路线规划,总的需求有两个,一个是机构管理,一个是路线管理。 其中,机构的管理是需要与【权限管家】系统中的【组织管理】中的数据进行同步,该同步是需要通过MQ完成的。 ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1668684314048-72fd52ba-8472-4648-a83c-8d0f00070e14.png#averageHue=%23fefefe&clientId=u3603cbee-3ab6-4&from=paste&height=346&id=u7e6bf3a6&name=image.png&originHeight=519&originWidth=1387&originalType=binary&ratio=1&rotation=0&showTitle=false&size=34969&status=done&style=shadow&taskId=u09f81570-b4af-4a62-b35d-e80169b642e&title=&width=924.6666666666666) 路线管理,主要提供路线的维护,最核心的服务是提供路线规划查询服务。 ![image.png](https://cdn.nlark.com/yuque/0/2023/png/28217986/1673668095843-69f17b95-e142-4717-bd06-8d2bf15a4ee4.png#averageHue=%23fefcfb&clientId=u0af63341-747c-4&from=paste&height=498&id=u8edc9057&name=image.png&originHeight=498&originWidth=598&originalType=binary&ratio=1&rotation=0&showTitle=true&size=23821&status=done&style=shadow&taskId=u086090c5-4303-4cc2-86c9-499f7824445&title=%E6%96%B0%E5%A2%9E%E7%BA%BF%E8%B7%AF&width=598 "新增线路") ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661395537827-42d0c5fe-ba6e-4455-8714-cde074d13dc2.png#averageHue=%23e6c28e&clientId=u93f722b6-ba7f-4&errorMessage=unknown%20error&from=paste&height=476&id=u9ffd0723&name=image.png&originHeight=786&originWidth=1601&originalType=binary&ratio=1&rotation=0&showTitle=true&size=118643&status=error&style=shadow&taskId=u8bf1641b-2542-4dd9-ad60-7cbb3e67dd7&title=%E8%B7%AF%E7%BA%BF%E5%88%97%E8%A1%A8&width=970.3029742210288 "路线列表") :::danger 需要说明的是,一个完整的运输路线是由多个转运节点组合完成的,并且每一个转运路线都是双向往返的,也就是A与B节点直接的路线必须是成对创建的。 ::: # 3、项目工程 ## 3.1、拉取代码 拉取`sl-express-ms-transport`相关的代码: | 工程名 | git地址 | | --- | --- | | sl-express-ms-transport-api | [http://git.sl-express.com/sl/sl-express-ms-transport-api](http://git.sl-express.com/sl/sl-express-ms-transport-api) | | sl-express-ms-transport-domain | [http://git.sl-express.com/sl/sl-express-ms-transport-domain.git](http://git.sl-express.com/sl/sl-express-ms-transport-domain.git) | | sl-express-ms-transport-service | [http://git.sl-express.com/sl/sl-express-ms-transport-service.git](http://git.sl-express.com/sl/sl-express-ms-transport-service.git) | ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661409604471-a3d7b14b-3618-4b82-83c6-eec91a90cedf.png#averageHue=%23f1efed&clientId=uf78e1dbf-4cc4-4&errorMessage=unknown%20error&from=paste&height=81&id=ud22689cb&name=image.png&originHeight=133&originWidth=492&originalType=binary&ratio=1&rotation=0&showTitle=false&size=8740&status=error&style=shadow&taskId=u13498189-fbc1-4cfb-92f9-0628d400a91&title=&width=298.18180094737426) ## 3.2、配置文件 在配置文件中引入了如下共享配置: | 文件名 | 说明 | | --- | --- | | shared-spring-rabbitmq.yml | 关于rabbitmq的统一配置,其中有对于消息消费失败处理的配置项 | | shared-spring-eaglemap.yml | 自研对接地图服务商的中台服务EagleMap的配置 | | shared-spring-neo4j.yml | neo4j的相关配置 | ### 3.2.1、shared-spring-rabbitmq.yml ```yaml #rabbitmq的基础配置 spring: rabbitmq: #mq的配置 host: 192.168.150.101 port: 5672 username: sl password: sl321 virtual-host: /dispatch publisher-confirm-type: correlated #发送消息的异步回调,记录消息是否发送成功 publisher-returns: true #开启publish-return功能,消息到达交换机,但是没有到达对列表 template: mandatory: true #消息路由失败时的策略, true: 调用ReturnCallback, false:丢弃消息 listener: simple: acknowledge-mode: auto #,出现异常时返回nack,消息回滚到mq;没有异常,返回ack retry: enabled: true # 开启消费者失败重试 initial-interval: 1000 # 初识的失败等待时长为1秒 multiplier: 1 # 失败的等待时长倍数,下次等待时长 = multiplier * last-interval max-attempts: 3 # 最大重试次数 stateless: true # true无状态;false有状态。如果业务中包含事务,这里改为false ``` ### 3.2.2、shared-spring-eaglemap.yml :::info 关于EagleMap使用,在后面讲解。 ::: ```yaml eagle: host: 192.168.150.101 #EagleMap服务地址 port: 8484 #EagleMap服务端口 timeout: 20000 #http请求的超时时间 ``` ### 3.2.3、shared-spring-neo4j.yml ```yaml spring: data: neo4j: database: ${neo4j.database} neo4j: authentication: username: ${neo4j.username} password: ${neo4j.password} uri: ${neo4j.uri} ``` 具体的参数值在`sl-express-transport.properties`文件中: ```properties neo4j.uri=neo4j://192.168.150.101:7687 neo4j.username=neo4j neo4j.password=neo4j123 neo4j.database=neo4j jdbc.url = jdbc:mysql://192.168.150.101:3306/sl_transport?useUnicode=true&characterEncoding=utf8&autoReconnect=true&allowMultiQueries=true&useSSL=false jdbc.username = root jdbc.password = 123 #权限系统对接的交换机 rabbitmq.exchange = itcast-auth ``` ## 3.3、代码结构 下面是路线规划微服务代码结构,主要是实现下面选中的部分: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661410871116-6f4ce17e-cef8-48fc-8b70-272beda91f4f.png#averageHue=%23faf9f8&clientId=uf78e1dbf-4cc4-4&errorMessage=unknown%20error&from=paste&height=503&id=u1b9face6&name=image.png&originHeight=830&originWidth=622&originalType=binary&ratio=1&rotation=0&showTitle=false&size=44051&status=error&style=shadow&taskId=u51a75f2e-f39c-4a91-9618-d07cba5335d&title=&width=376.9696751814365) :::danger 关于Entity,与sl-express-sdn工程的类似,只是属性多了一些,按照项目的业务需求制定的。 ::: Feign接口定义: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661602795290-b212b85e-df2d-4537-8817-b8de86115e3f.png#averageHue=%23fbfaf9&clientId=u4d76cb3d-b589-4&errorMessage=unknown%20error&from=paste&height=170&id=u500f73f6&name=image.png&originHeight=280&originWidth=635&originalType=binary&ratio=1&rotation=0&showTitle=false&size=12840&status=error&style=shadow&taskId=ueb447e7e-5932-4010-9823-00355675109&title=&width=384.8484626048428) domain定义: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661602815385-111e2014-acf5-4d26-ab07-5db679b86488.png#averageHue=%23faf9f7&clientId=u4d76cb3d-b589-4&errorMessage=unknown%20error&from=paste&height=201&id=u7ebf2c74&name=image.png&originHeight=332&originWidth=591&originalType=binary&ratio=1&rotation=0&showTitle=false&size=17882&status=error&style=shadow&taskId=u44f63c0d-29b9-4c73-bec2-6ae6650c336&title=&width=358.18179747946783) ## 3.4、sl-express-mq 在项目中,为了统一使用RabbitMQ,所以将MQ的使用进行了封装,使用方法参考文档《[sl-express-mq使用手册](https://sl-express.itheima.net/#/zh-cn/modules/sl-express-mq)》 ### 3.4.1、发送消息 对于发送消息的场景,正常情况没有问题,直接发送即可: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677747565104-1479809b-38a7-4ee7-ab85-70e36e717016.png#averageHue=%23fcfcfb&clientId=u1be9e8bc-373d-4&from=paste&height=159&id=ud32b9dee&name=image.png&originHeight=239&originWidth=1197&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=22847&status=done&style=shadow&taskId=u95a12cd2-50a1-4f99-88ef-a1c2545fca1&title=&width=798) 如果是非正常情况就需要特殊处理了,一般会有三种非正常情况需要处理: - 第一种情况,消息发送到交换机(exchange),但是没有队列与交换机绑定,消息会丢失。 ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677747542317-e461628f-3269-44b5-b5e2-5ab7bd7e8a68.png#averageHue=%23fcfbfa&clientId=u1be9e8bc-373d-4&from=paste&height=171&id=u1d79372f&name=image.png&originHeight=257&originWidth=1222&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=26503&status=done&style=shadow&taskId=u304e0b28-f99a-459d-93c6-2040090801a&title=&width=814.6666666666666) - 第二种情况,在消息的发送后进行确认,如果发送失败需要将消息持久化,例如:发送的交换机不存在的情况。 ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677747891930-b527fd38-5616-4a6a-9ee3-bf4bb4dacee4.png#averageHue=%23fcf8f6&clientId=u1be9e8bc-373d-4&from=paste&height=149&id=u453dea89&name=image.png&originHeight=223&originWidth=1013&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=23533&status=done&style=shadow&taskId=ubccb3dc5-29fb-4427-b5b9-a443c58fc68&title=&width=675.3333333333334) - 第三种情况,由于网络、MQ服务宕机等原因导致消息没有发送到MQ服务器。 ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677748019068-47a2b11d-ca88-48eb-99aa-f956d901cee5.png#averageHue=%23fdf5f1&clientId=u1be9e8bc-373d-4&from=paste&height=127&id=u293aeaac&name=image.png&originHeight=191&originWidth=670&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=13055&status=done&style=shadow&taskId=u08cda024-cb1b-4632-b7af-077970ead2a&title=&width=446.6666666666667) **第一种情况:** 对于消息只是到了交换机,并没有到达队列,这种情况记录日志即可,因为我们也不确定哪个队列需要这个消息。 配置如下(nacos中的`shared-spring-rabbitmq.yml`文件): ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1676427302100-873277d3-435c-4f1d-8911-f4306b027028.png#averageHue=%23222020&clientId=u6f0f8ddb-318f-4&from=paste&height=379&id=ufef3248a&name=image.png&originHeight=568&originWidth=1375&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=150968&status=done&style=shadow&taskId=u6995b5dc-479a-44dd-86dc-aa0c6a7906b&title=&width=916.6666666666666) ```java package com.sl.mq.config; import cn.hutool.core.util.StrUtil; import com.sl.transport.common.constant.Constants; import lombok.extern.slf4j.Slf4j; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.beans.BeansException; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; import org.springframework.context.annotation.Configuration; @Slf4j @Configuration public class MessageConfig implements ApplicationContextAware { /** * 发送者回执 没有路由到队列的情况 * * @param applicationContext 应用上下文 * @throws BeansException 异常 */ @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { // 获取RabbitTemplate RabbitTemplate rabbitTemplate = applicationContext.getBean(RabbitTemplate.class); // 设置ReturnCallback rabbitTemplate.setReturnsCallback(message -> { if (StrUtil.contains(message.getExchange(), Constants.MQ.DELAYED_KEYWORD)) { //延迟消息没有发到队列是正常情况,无需记录日志 return; } // 投递失败,记录日志 log.error("消息没有投递到队列,应答码:{},原因:{},交换机:{},路由键:{},消息:{}", message.getReplyCode(), message.getReplyText(), message.getExchange(), message.getRoutingKey(), message.getMessage()); }); } } ``` **第二种情况:** 在配文件中开启配置`publisher-confirm-type`,即可在发送消息时添加回调方法: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1676427331425-de958f75-5448-4385-ad3a-35868b1c1eb0.png#averageHue=%23222020&clientId=u6f0f8ddb-318f-4&from=paste&height=381&id=u498a6541&name=image.png&originHeight=571&originWidth=1388&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=151290&status=done&style=shadow&taskId=uc0f44209-29b6-4b2c-ba8c-bdbfd798528&title=&width=925.3333333333334) 在代码中进行处理,将消息数据持久化到数据库中,后续通过xxl-job进行处理,将消息进行重新发送。 ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1672304142276-ef564ea6-dbfd-4f7d-ad23-77f0e0e85983.png#averageHue=%23fdfcfb&clientId=ub939c037-ffdd-4&from=paste&height=445&id=u23ee5b9f&name=image.png&originHeight=668&originWidth=1585&originalType=binary&ratio=1&rotation=0&showTitle=false&size=66901&status=done&style=shadow&taskId=udd2ab883-3c74-4f4d-90af-367b6bf3abe&title=&width=1056.6666666666667) 同样,如果出现异常情况也是将消息持久化: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677727282231-d0f433b6-5466-4e06-b894-576f3d9bb584.png#averageHue=%23fdfcfb&clientId=uabbce504-f769-4&from=paste&height=294&id=u1e4534ca&name=image.png&originHeight=441&originWidth=1480&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=53023&status=done&style=shadow&taskId=udd86a478-f8bb-4fb1-b0e2-79013451a88&title=&width=986.6666666666666) **第三种情况:** 将发送消息的代码进行try{}catch{}处理,如果出现异常会通过Spring-retry机制进重试,最多重试3次,如果依然失败就将消息数据进行持久化: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677727332009-841bf175-d82e-491a-ad87-2c2baa676ec9.png#averageHue=%23fdfcfc&clientId=uabbce504-f769-4&from=paste&height=504&id=ude0fbf75&name=image.png&originHeight=756&originWidth=1571&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=71101&status=done&style=shadow&taskId=uaaa1eb79-3850-4e5f-be6e-0bc32bd5bde&title=&width=1047.3333333333333) 设置重试: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677727362761-4ab68fae-f89d-43c5-bc3c-74325dc85672.png#averageHue=%23fcfbfa&clientId=uabbce504-f769-4&from=paste&height=189&id=uced0fd8f&name=image.png&originHeight=283&originWidth=1671&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=40401&status=done&style=shadow&taskId=u366409e1-410c-421f-be6a-6f9bda6b50d&title=&width=1114) 最终的落库操作: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1672304978965-69e906b5-fd6c-4b8b-9cbf-022a2382f2a7.png#averageHue=%23fdfcfc&clientId=ub939c037-ffdd-4&from=paste&height=232&id=ubdaa8996&name=image.png&originHeight=348&originWidth=1231&originalType=binary&ratio=1&rotation=0&showTitle=false&size=36698&status=done&style=shadow&taskId=u2e9d5254-ca1f-44aa-bc4c-0537831b8f2&title=&width=820.6666666666666) xxl-job任务,主要负责从数据库中查询出错误消息数据然后进行重试: ```java package com.sl.mq.job; import cn.hutool.core.collection.CollUtil; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.sl.mq.entity.FailMsgEntity; import com.sl.mq.service.FailMsgService; import com.sl.mq.service.MQService; import com.xxl.job.core.handler.annotation.XxlJob; import lombok.extern.slf4j.Slf4j; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.stereotype.Component; import javax.annotation.Resource; import java.util.List; /** * 失败消息的处理任务 */ @Slf4j @Component @ConditionalOnBean({MQService.class, FailMsgService.class}) public class FailMsgJob { @Resource private FailMsgService failMsgService; @Resource private MQService mqService; @XxlJob("failMsgJob") public void execute() { //查询失败的数据,每次最多处理100条错误消息 LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper() .orderByAsc(FailMsgEntity::getCreated) .last("limit 100"); List failMsgEntityList = this.failMsgService.list(queryWrapper); if (CollUtil.isEmpty(failMsgEntityList)) { return; } for (FailMsgEntity failMsgEntity : failMsgEntityList) { try { //发送消息 this.mqService.sendMsg(failMsgEntity.getExchange(), failMsgEntity.getRoutingKey(), failMsgEntity.getMsg()); //删除数据 this.failMsgService.removeById(failMsgEntity.getId()); } catch (Exception e) { log.error("处理错误消息失败, failMsgEntity = {}", failMsgEntity); } } } } ``` xxl-job中的任务调度: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1672305271514-512e097a-fb7a-4d84-981a-f22b2fbecae5.png#averageHue=%23e0c498&clientId=ub939c037-ffdd-4&from=paste&height=304&id=u4a2bc32e&name=image.png&originHeight=456&originWidth=1564&originalType=binary&ratio=1&rotation=0&showTitle=false&size=51664&status=done&style=shadow&taskId=ucf0c3047-a0b1-46be-a45b-2ca52c626fd&title=&width=1042.6666666666667) ### 3.4.2、消费消息 对于消息的消费,首先采用的自动确认策略: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1672306644098-25b4bd13-3309-4ff0-98b6-ac4450ff3d4c.png#averageHue=%23212020&clientId=ub939c037-ffdd-4&from=paste&height=77&id=uc082f893&name=image.png&originHeight=116&originWidth=1037&originalType=binary&ratio=1&rotation=0&showTitle=false&size=20829&status=done&style=shadow&taskId=u3cbb9f15-2b76-4920-a7d4-81627510d14&title=&width=691.3333333333334) 如果出现消费错误,会进行重试,最多重试3次: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1672306709957-86675680-af17-493f-835e-728e4fc6e05c.png#averageHue=%23242221&clientId=ub939c037-ffdd-4&from=paste&height=196&id=u798babde&name=image.png&originHeight=294&originWidth=1012&originalType=binary&ratio=1&rotation=0&showTitle=false&size=75808&status=done&style=shadow&taskId=u4c9a1035-4a51-4b2a-ad6b-f9feb4a6678&title=&width=674.6666666666666) 如果3次后依然失败,需要将消息发送到指定的队列,为了区分不同的微服务,所以会针对不同微服务创建不同的队列,但是交换机是同一个: ```java package com.sl.mq.config; import com.sl.transport.common.constant.Constants; import org.springframework.amqp.core.Binding; import org.springframework.amqp.core.BindingBuilder; import org.springframework.amqp.core.Queue; import org.springframework.amqp.core.TopicExchange; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.amqp.rabbit.retry.MessageRecoverer; import org.springframework.amqp.rabbit.retry.RepublishMessageRecoverer; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @Configuration public class ErrorMessageConfig { @Value("${spring.application.name}") //获取微服务的名称 private String appName; @Bean public TopicExchange errorMessageExchange() { //定义错误消息的交换机,类型为:topic return new TopicExchange(Constants.MQ.Exchanges.ERROR, true, false); } @Bean public Queue errorQueue() { //【前缀+微服务】名作为错误消息存放的队列名称,并且开启了持久化 return new Queue(Constants.MQ.Queues.ERROR_PREFIX + appName, true); } @Bean public Binding errorBinding(Queue errorQueue, TopicExchange errorMessageExchange) { //完成绑定关系 return BindingBuilder.bind(errorQueue).to(errorMessageExchange).with(appName); } @Bean public MessageRecoverer republishMessageRecoverer(RabbitTemplate rabbitTemplate) { //设置全部重试失败后进行重新发送消息,指定了交换机以及路由key //需要注意的是,路由key是应用名称,与上述的绑定关系中的路由key一致 return new RepublishMessageRecoverer(rabbitTemplate, Constants.MQ.Exchanges.ERROR, appName); } } ``` 最终会以微服务名称创建队列: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1672307521272-1f1aeb65-8c4d-48c1-affc-3981359143fe.png#averageHue=%23ebe9e8&clientId=ub939c037-ffdd-4&from=paste&height=184&id=u44d199a4&name=image.png&originHeight=276&originWidth=1606&originalType=binary&ratio=1&rotation=0&showTitle=false&size=83220&status=done&style=shadow&taskId=u4e897646-8938-4b64-9830-c745c6d5a91&title=&width=1070.6666666666667) 其绑定关系如下: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1672307540927-59ab439e-ed6a-403e-ae73-12f1f7221fe7.png#averageHue=%23f9f9f9&clientId=ub939c037-ffdd-4&from=paste&height=264&id=u66db07dc&name=image.png&originHeight=396&originWidth=991&originalType=binary&ratio=1&rotation=0&showTitle=false&size=28463&status=done&style=shadow&taskId=u2551c9b4-9546-4289-bd2f-180e5a7a8ae&title=&width=660.6666666666666) ### 3.4.3、统一封装 为了在各个微服务中方便发送消息,所以在`sl-express-ms-base`微服务中进行了封装,使用时`com.sl.ms.base.api.common.MQFeign`调用即可。 在base微服务中添加了配置以及启用Spring-retry机制: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677728445152-1e022e5f-806f-4842-9284-c51a12ed4ab5.png#averageHue=%23fbfaf5&clientId=u5dece25d-787d-4&from=paste&height=297&id=u866103d4&name=image.png&originHeight=446&originWidth=1325&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=56111&status=done&style=shadow&taskId=uef8112ea-5cba-4b6d-9d2b-9b741e9150b&title=&width=883.3333333333334) ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677728466984-0f8ea414-9cb4-4e6a-89e5-9e0f5c7acda8.png#averageHue=%23fcfcfa&clientId=u5dece25d-787d-4&from=paste&height=290&id=ub7992c2a&name=image.png&originHeight=435&originWidth=1081&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=30078&status=done&style=shadow&taskId=u5de5a23f-641d-42c3-a758-838b05a2734&title=&width=720.6666666666666) 使用示例如下: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1677728596414-94681e10-89d4-4307-9a7f-c45f07ed2912.png#averageHue=%23fbfafa&clientId=u5dece25d-787d-4&from=paste&height=317&id=u916cea4d&name=image.png&originHeight=476&originWidth=1618&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=54444&status=done&style=shadow&taskId=ufaeda5aa-35e8-4bf3-807a-075a33de4b3&title=&width=1078.6666666666667) > **发送时指定交换机、路由key、消息内容、延时时间(毫秒)即可。** # 4、机构同步 机构的新增、更新、删除是在权限管家中完成的,需要是操作后同步到路线规划微服务中,这里采用的是MQ消息通知的方式。 ## 4.1、业务流程 ![](https://cdn.nlark.com/yuque/__puml/e87b243b2b36f12c01374c888fb0d622.svg#lake_card_v2=eyJ0eXBlIjoicHVtbCIsImNvZGUiOiJAc3RhcnR1bWxcblxuYXV0b251bWJlclxuXG5hY3RvciBcIueUqOaIt1wiIGFzIHVzZXJcbnBhcnRpY2lwYW50IFwi5p2D6ZmQ566h5a62XCIgYXMgYXV0aFxucGFydGljaXBhbnQgXCLot6_nur_op4TliJLlvq7mnI3liqFcIiBhcyB0cmFuc3BvcnRcbnBhcnRpY2lwYW50IFwiTmVvNGpcIiBhcyBuZW80alxuXG51c2VyIC0-IGF1dGggLS0rKzog5paw5aKe5py65p6EXG5hdXRoIC0-IGF1dGg6IOS_neWtmOaVsOaNrlxuYXV0aCAtLT4gdXNlcjog6L-U5ZueXG5hdXRoIC0tPiB0cmFuc3BvcnQgLS0rKzog5Y-R6YCB5raI5oGvXG50cmFuc3BvcnQgLT4gdHJhbnNwb3J0IDog5Lia5Yqh5aSE55CGXG50cmFuc3BvcnQgLT4gbmVvNGogKys6IOS_neWtmOaVsOaNrlxubmVvNGogLT4gbmVvNGogOiDkv53lrZjmlbDmja5cbm5lbzRqIC0-IHRyYW5zcG9ydCAtLTog6L-U5ZueXG5AZW5kdW1sIiwidXJsIjoiaHR0cHM6Ly9jZG4ubmxhcmsuY29tL3l1cXVlL19fcHVtbC9lODdiMjQzYjJiMzZmMTJjMDEzNzRjODg4ZmIwZDYyMi5zdmciLCJpZCI6ImRGRmNqIiwibWFyZ2luIjp7InRvcCI6dHJ1ZSwiYm90dG9tIjp0cnVlfSwiY2FyZCI6ImRpYWdyYW0ifQ==)## 4.2、权限管家配置 权限管家的MQ配置是在 `/itcast/itcast-auth-server/application-test.properties`文件中,如下: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661414367036-b19987ad-7794-47c8-92ad-a2956210d3c2.png#averageHue=%230d141e&clientId=uf78e1dbf-4cc4-4&errorMessage=unknown%20error&from=paste&height=216&id=uc987dcca&name=image.png&originHeight=357&originWidth=1230&originalType=binary&ratio=1&rotation=0&showTitle=false&size=37168&status=error&style=shadow&taskId=u3badaf30-4a59-41fa-924f-4b6ed4ec63c&title=&width=745.4545023684356) 可以看出,消息发往的交换机为:itcast-auth,交换机的类型为:topic 发送消息的规则如下: :::info - 消息为json字符串 - 如:{"type":"ORG","content":[{"managerId":"1","parentId":"0","name":"测试组织","id":"973902113476182273","status":true}],"operation":"UPDATE"} - type表示变更的对象,比如组织:ORG - content为更改对象列表 - operation类型列表 - 新增-ADD - 修改-UPDATE - 删除-DEL ::: 所以,对应的在`sl-express-transport.properties`中配置相同的交换机。 ## 4.3、业务规范 ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661418858904-20af71d3-24c9-4b12-8902-0ea95643a9e8.png#averageHue=%23fefefe&clientId=uf78e1dbf-4cc4-4&errorMessage=unknown%20error&from=paste&height=435&id=u0ecaf4a0&name=image.png&originHeight=717&originWidth=859&originalType=binary&ratio=1&rotation=0&showTitle=false&size=26996&status=error&style=shadow&taskId=uc031847f-b190-4ab9-b2a9-633c44ab01a&title=&width=520.6060305158425) 上图是在权限管家中新增组织的界面,可以从界面中看出,添加的组织并没有标识是【网点】还是【转运中心】,所以,在这里我们做一下约定,按照机构名称的后缀进行区分,具体规则如下: - xxx转运中心 → 一级转运中心(OLT) - xxx分拣中心 → 二级转运中心 (TLT) - xxx营业部 → 网点(AGENCY) ## 4.4、具体实现 ```java package com.sl.transport.mq; import cn.hutool.core.util.StrUtil; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; import com.sl.transport.common.constant.Constants; import com.sl.transport.entity.node.AgencyEntity; import com.sl.transport.entity.node.BaseEntity; import com.sl.transport.entity.node.OLTEntity; import com.sl.transport.entity.node.TLTEntity; import com.sl.transport.enums.OrganTypeEnum; import com.sl.transport.service.IService; import com.sl.transport.utils.OrganServiceFactory; import lombok.extern.slf4j.Slf4j; import org.springframework.amqp.core.ExchangeTypes; import org.springframework.amqp.core.TopicExchange; import org.springframework.amqp.rabbit.annotation.Exchange; import org.springframework.amqp.rabbit.annotation.Queue; import org.springframework.amqp.rabbit.annotation.QueueBinding; import org.springframework.amqp.rabbit.annotation.RabbitListener; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.stereotype.Component; /** * 对于权限管家系统消息的处理 */ @Slf4j @Component public class AuthMQListener { @RabbitListener(bindings = @QueueBinding( value = @Queue(name = Constants.MQ.Queues.AUTH_TRANSPORT), exchange = @Exchange(name = "${rabbitmq.exchange}", type = ExchangeTypes.TOPIC), key = "#" )) public void listenAgencyMsg(String msg) { //{"type":"ORG","operation":"ADD","content":[{"id":"977263044792942657","name":"55","parentId":"0","managerId":null,"status":true}]} log.info("接收到消息 -> {}", msg); JSONObject jsonObject = JSONUtil.parseObj(msg); String type = jsonObject.getStr("type"); if (!StrUtil.equalsIgnoreCase(type, "ORG")) { //非机构消息 return; } String operation = jsonObject.getStr("operation"); JSONObject content = (JSONObject) jsonObject.getJSONArray("content").getObj(0); String name = content.getStr("name"); Long parentId = content.getLong("parentId"); IService iService; BaseEntity entity; if (StrUtil.endWith(name, "转运中心")) { //一级转运中心 iService = OrganServiceFactory.getBean(OrganTypeEnum.OLT.getCode()); entity = new OLTEntity(); entity.setParentId(0L); } else if (StrUtil.endWith(name, "分拣中心")) { //二级转运中心 iService = OrganServiceFactory.getBean(OrganTypeEnum.TLT.getCode()); entity = new TLTEntity(); entity.setParentId(parentId); } else if (StrUtil.endWith(name, "营业部")) { //网点 iService = OrganServiceFactory.getBean(OrganTypeEnum.AGENCY.getCode()); entity = new AgencyEntity(); entity.setParentId(parentId); } else { return; } //设置参数 entity.setBid(content.getLong("id")); entity.setName(name); entity.setStatus(content.getBool("status")); switch (operation) { case "ADD": { iService.create(entity); break; } case "UPDATE": { iService.update(entity); break; } case "DEL": { iService.deleteByBid(entity.getBid()); break; } } } } ``` :::danger 由于Service还没有具体实现,暂时不对代码测试,后面实现后进行测试。 ::: # 5、IService 在Service中一些方法是通用的,比如新增、更新、删除等,这个通用的方法可以写到一个Service中,其他的Service继承该Service即可。 ## 5.1、IService 接口定义: ```java package com.sl.transport.service; import com.sl.transport.entity.node.BaseEntity; /** * 基础服务实现 */ public interface IService { /** * 根据业务id查询数据 * * @param bid 业务id * @return 节点数据 */ T queryByBid(Long bid); /** * 新增节点 * * @param t 节点数据 * @return 新增的节点数据 */ T create(T t); /** * 更新节点 * * @param t 节点数据 * @return 更新的节点数据 */ T update(T t); /** * 根据业务id删除数据 * * @param bid 业务id * @return 是否删除成功 */ Boolean deleteByBid(Long bid); } ``` ## 5.2、ServiceImpl 下面编写具体的实现类: ```java package com.sl.transport.service.impl; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.bean.copier.CopyOptions; import com.sl.transport.common.util.ObjectUtil; import com.sl.transport.entity.node.BaseEntity; import com.sl.transport.repository.BaseRepository; import com.sl.transport.service.IService; import org.springframework.beans.factory.annotation.Autowired; /** * 基础服务的实现 */ public class ServiceImpl implements IService { @Autowired private R repository; @Override public T queryByBid(Long bid) { return (T) this.repository.findByBid(bid).orElse(null); } @Override public T create(T t) { t.setId(null);//id由neo4j自动生成 return (T) this.repository.save(t); } @Override public T update(T t) { //先查询,再更新 T tData = this.queryByBid(t.getBid()); if (ObjectUtil.isEmpty(tData)) { return null; } BeanUtil.copyProperties(t, tData, CopyOptions.create().ignoreNullValue().setIgnoreProperties("id", "bid")); return (T) this.repository.save(tData); } @Override public Boolean deleteByBid(Long bid) { return this.repository.deleteByBid(bid) > 0; } } ``` ## 5.3、AgencyServiceImpl 网点服务实现类: ```java package com.sl.transport.service.impl; import com.sl.transport.entity.node.AgencyEntity; import com.sl.transport.repository.AgencyRepository; import com.sl.transport.service.AgencyService; import org.springframework.stereotype.Service; @Service public class AgencyServiceImpl extends ServiceImpl implements AgencyService { } ``` ## 5.4、OLTServiceImpl 一级转运中心服务实现类: ```java package com.sl.transport.service.impl; import com.sl.transport.entity.node.OLTEntity; import com.sl.transport.repository.OLTRepository; import com.sl.transport.service.OLTService; import org.springframework.stereotype.Service; @Service public class OLTServiceImpl extends ServiceImpl implements OLTService { } ``` ## 5.5、TLTServiceImpl 二级转运中心服务实现类: ```java package com.sl.transport.service.impl; import com.sl.transport.entity.node.TLTEntity; import com.sl.transport.repository.TLTRepository; import com.sl.transport.service.TLTService; import org.springframework.stereotype.Service; @Service public class TLTServiceImpl extends ServiceImpl implements TLTService { } ``` ## 5.6、单元测试 编写测试用例: ```java package com.sl.transport.service; import com.sl.transport.entity.node.AgencyEntity; import org.junit.jupiter.api.Test; import org.springframework.boot.test.context.SpringBootTest; import javax.annotation.Resource; @SpringBootTest class AgencyServiceTest { @Resource private AgencyService agencyService; @Test public void testQueryByBid(){ AgencyEntity agencyEntity = this.agencyService.queryByBid(25073L); System.out.println(agencyEntity); //AgencyEntity(super=BaseEntity(id=18, parentId=null, bid=25073, name=江苏省南京市玄武区紫金墨香苑, managerName=null, phone=025-58765331,025-83241955,025-83241881, address=栖霞区燕尧路100号, location=Point [x=32.117016, y=118.863193], status=null, extra=null)) } } ``` > 🚨注意:需要将OrganController、TransportLineController中的@RestController注释掉才能测试,否则会抛出异常。 ## 5.7、测试机构同步 将Neo4j中的数据全部删除:`MATCH (n) DETACH DELETE n` 创建机构: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661426006150-2a008fe5-4744-4217-a6f5-934794d693b5.png#averageHue=%23fefefe&clientId=u876d64eb-b77d-4&errorMessage=unknown%20error&from=paste&height=432&id=ub93109cb&name=image.png&originHeight=713&originWidth=856&originalType=binary&ratio=1&rotation=0&showTitle=false&size=27078&status=error&style=shadow&taskId=u62c1100d-f4d3-416b-9644-3781ccd9d8d&title=&width=518.7878488027486) 可以看到对应的Neo4j中已经有数据: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661425946094-dbda9e63-7fa9-47d6-a3e6-ba4f0921bfc7.png#averageHue=%23d4e3d4&clientId=u876d64eb-b77d-4&errorMessage=unknown%20error&from=paste&height=254&id=u140fb50b&name=image.png&originHeight=419&originWidth=809&originalType=binary&ratio=1&rotation=0&showTitle=false&size=39874&status=error&style=shadow&taskId=u09171e1c-92c4-42bf-aa4d-db87485f58a&title=&width=490.30300196428) 同理可以测试更新、删除操作。 # 6、机构管理 按照业务系统的需求,会通过bid查询机构,无需指定type,也就是说,我们需要将网点和转运中心都看作是机构,需要实现两个查询方法: - 根据bid查询 - 查询机构列表 ## 6.1、接口定义 ```java package com.sl.transport.service; import com.sl.transport.domain.OrganDTO; import java.util.List; /** * @author zzj * @version 1.0 */ public interface OrganService { /** * 无需指定type,根据id查询 * * @param bid 机构id * @return 机构信息 */ OrganDTO findByBid(Long bid); /** * 无需指定type,根据ids查询 * * @param bids 机构ids * @return 机构信息 */ List findByBids(List bids); /** * 查询所有的机构,如果name不为空的按照name模糊查询 * * @param name 机构名称 * @return 机构列表 */ List findAll(String name); /** * 查询机构树 * @return 机构树 */ String findAllTree(); } ``` ## 6.2、具体实现 ```java package com.sl.transport.service.impl; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.collection.CollUtil; import cn.hutool.core.lang.tree.Tree; import cn.hutool.core.lang.tree.TreeUtil; import cn.hutool.core.util.ObjectUtil; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.ObjectMapper; import com.sl.transport.common.exception.SLException; import com.sl.transport.domain.OrganDTO; import com.sl.transport.enums.ExceptionEnum; import com.sl.transport.repository.OrganRepository; import com.sl.transport.service.OrganService; import org.springframework.stereotype.Service; import javax.annotation.Resource; import java.util.List; @Service public class OrganServiceImpl implements OrganService { @Resource private OrganRepository organRepository; @Resource private ObjectMapper objectMapper; @Override public OrganDTO findByBid(Long bid) { OrganDTO organDTO = this.organRepository.findByBid(bid); if (ObjectUtil.isNotEmpty(organDTO)) { return organDTO; } throw new SLException(ExceptionEnum.ORGAN_NOT_FOUND); } @Override public List findByBids(List bids) { List organDTOS = this.organRepository.findByBids(bids); if (ObjectUtil.isNotEmpty(organDTOS)) { return organDTOS; } throw new SLException(ExceptionEnum.ORGAN_NOT_FOUND); } @Override public List findAll(String name) { return this.organRepository.findAll(name); } @Override public String findAllTree() { return ""; } } ``` ## 6.3、OrganRepositoryImpl 下面对于OrganRepository接口进行实现: ```java package com.sl.transport.repository.impl; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.collection.CollUtil; import cn.hutool.core.collection.ListUtil; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.StrUtil; import com.sl.transport.domain.OrganDTO; import com.sl.transport.enums.OrganTypeEnum; import com.sl.transport.repository.OrganRepository; import org.neo4j.driver.internal.InternalPoint2D; import org.springframework.data.neo4j.core.Neo4jClient; import org.springframework.stereotype.Component; import javax.annotation.Resource; import java.util.List; import java.util.Map; @Component public class OrganRepositoryImpl implements OrganRepository { @Resource private Neo4jClient neo4jClient; @Override public OrganDTO findByBid(Long bid) { String cypherQuery = StrUtil.format("MATCH (n)\n" + "WHERE n.bid = {}\n" + "RETURN n", bid); return CollUtil.getFirst(executeQuery(cypherQuery)); } @Override public List findByBids(List bids) { String cypherQuery = StrUtil.format("MATCH (n)\n" + "WHERE n.bid in {}\n" + "RETURN n", bids); return executeQuery(cypherQuery); } @Override public List findAll(String name) { name = StrUtil.removeAll(name, '\'', '"'); String cypherQuery = StrUtil.isEmpty(name) ? "MATCH (n) RETURN n" : StrUtil.format("MATCH (n) WHERE n.name CONTAINS '{}' RETURN n", name); return executeQuery(cypherQuery); } private List executeQuery(String cypherQuery) { return ListUtil.toList(this.neo4jClient.query(cypherQuery) .fetchAs(OrganDTO.class) //设置响应的类型 .mappedBy((typeSystem, record) -> { //对结果进行封装处理 Map map = record.get("n").asMap(); OrganDTO organDTO = BeanUtil.toBean(map, OrganDTO.class); InternalPoint2D location = (InternalPoint2D) map.get("location"); if (ObjectUtil.isNotEmpty(location)) { organDTO.setLongitude(location.x()); organDTO.setLatitude(location.y()); } //获取类型 String type = CollUtil.getFirst(record.get("n").asNode().labels()); organDTO.setType(OrganTypeEnum.valueOf(type).getCode()); return organDTO; }).all()); } } ``` ## 6.4、测试用例 ```java package com.sl.transport.service; import com.sl.transport.domain.OrganDTO; import org.junit.jupiter.api.Test; import org.springframework.boot.test.context.SpringBootTest; import javax.annotation.Resource; import java.util.List; import static org.junit.jupiter.api.Assertions.*; @SpringBootTest class OrganServiceTest { @Resource private OrganService organService; @Test void findByBid() { //bid值要改成自己neo4j中的值 OrganDTO organDTO = this.organService.findByBid(1012479939628238305L); System.out.println(organDTO); } @Test void findAll() { //查询包含“上海”关键字的机构 List list = this.organService.findAll("上海"); list.forEach(System.out::println); } } ``` ## 6.5、整合测试 基于swagger接口进行测试:[http://127.0.0.1:18083/doc.html](http://127.0.0.1:18083/doc.html) :::danger 测试时,由于部分Service还没有实现,会导致启动报错,所以需要将报错Controller中的@RestController注释掉。 ::: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661436832675-b01d0c60-6453-44e4-9cd0-244d58f29adb.png#averageHue=%23fcfcfc&clientId=u8e19e744-02a1-4&errorMessage=unknown%20error&from=paste&height=487&id=ud1ee79f1&name=image.png&originHeight=803&originWidth=1429&originalType=binary&ratio=1&rotation=0&showTitle=false&size=109499&status=error&style=shadow&taskId=u74150fe8-6f63-4956-86a8-815dc97f1a7&title=&width=866.060556003654) ## 6.6、树形结构 在后台系统中,对于机构数据的展现需要通过树形结构展现,如下: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1673419872381-74654f1d-9521-4606-a422-1d462e1aebcd.png#averageHue=%23fdfbfa&clientId=u5a15bc20-4154-4&from=paste&height=437&id=u007ea3c7&name=image.png&originHeight=655&originWidth=468&originalType=binary&ratio=1&rotation=0&showTitle=false&size=41244&status=done&style=shadow&taskId=u3e4ed56e-f86e-422f-8282-1b6f4f63fcb&title=&width=312) 所以在`com.sl.transport.service.OrganService`中`findAllTree()`方法中封装了树形结构。 具体的封装逻辑采用hutool工具包中的TreeUtil,参考文档:[点击查看](https://www.hutool.cn/docs/#/core/%E8%AF%AD%E8%A8%80%E7%89%B9%E6%80%A7/%E6%A0%91%E7%BB%93%E6%9E%84/%E6%A0%91%E7%BB%93%E6%9E%84%E5%B7%A5%E5%85%B7-TreeUtil) 代码实现如下: ```java @Override public String findAllTree() { List organList = this.findAll(null); if (CollUtil.isEmpty(organList)) { return ""; } //构造树结构 List> treeNodes = TreeUtil.build(organList, 0L, (organDTO, tree) -> { tree.setId(organDTO.getId()); tree.setParentId(organDTO.getParentId()); tree.putAll(BeanUtil.beanToMap(organDTO)); tree.remove("bid"); }); try { return this.objectMapper.writeValueAsString(treeNodes); } catch (JsonProcessingException e) { throw new SLException("序列化json出错!", e); } } ``` 数据类似这样: ```json [ { "id": "1012438698496623009", "parentId": "0", "name": "上海市转运中心", "type": 1, "phone": null, "address": null, "latitude": null, "longitude": null, "managerName": null, "extra": null, "status": true, "children": [ { "id": "1012479939628238305", "parentId": "1012438698496623009", "name": "浦东区分拣中心", "type": 2, "phone": null, "address": null, "latitude": null, "longitude": null, "managerName": null, "extra": null, "status": true } ] }, { "id": "1012479716659037537", "parentId": "0", "name": "北京市转运中心", "type": 1, "phone": null, "address": null, "latitude": null, "longitude": null, "managerName": null, "extra": null, "status": true } ] ``` 功能测试: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661436796038-a4700a64-d9f6-49f7-935c-ead08dce6c84.png#averageHue=%23f8f8f8&clientId=u8e19e744-02a1-4&errorMessage=unknown%20error&from=paste&height=326&id=Gat60&name=image.png&originHeight=538&originWidth=1412&originalType=binary&ratio=1&rotation=0&showTitle=false&size=80788&status=error&style=shadow&taskId=u55b1777d-917b-444d-927c-0ac0e8e8b71&title=&width=855.7575262961228) > 🔔如果测试没有数据,需要自行在权限管家中创建相对应的网点、转运中心等数据,进行测试。 ## 6.7、编辑机构 在后台系统中可以对机构数据进行编辑,主要是填充一些属性数据,例如:经纬度、详细地址。如下: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661565905827-d35c4b85-4929-4ab0-aaeb-87a8392493f3.png#averageHue=%23fdfcfb&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=333&id=u1e871295&name=image.png&originHeight=550&originWidth=1612&originalType=binary&ratio=1&rotation=0&showTitle=false&size=49753&status=error&style=shadow&taskId=u8cd705ac-79c6-48c8-802d-cf8f2054af3&title=&width=976.9696405023725) 经纬度是如何计算出来的呢?这里使用的是高德地图的API进行查询的,将中文字段转化为经纬度值,我们是直接调用的高德地图的API吗,不是的,我们是通过EagleMap调用的。 ### 6.7.1、EagleMap介绍 EagleMap是黑马程序员研究院自研的地图中台服务,它可以对接多个地图服务商,目前已经完成百度地图和高德地图的对接。 目前EagleMap已经部署安装在101机器中,配置文件所在的位置:`/itcast/eaglemap/app/application.yml` :::danger 在这里强烈建议将高德地图的ak改成自己的,不要使用默认的,因为使用人多了可能会被封号,将不能正常使用。更改了ak后,要记得重启EagleMap服务`docker restart eagle-map-server` ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661566455518-f9b4b85b-f55b-489e-9ea6-7418c111c0bd.png#averageHue=%230e1623&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=281&id=u71e84cf4&name=image.png&originHeight=463&originWidth=741&originalType=binary&ratio=1&rotation=0&showTitle=false&size=51484&status=error&style=shadow&taskId=u944a2167-778b-403c-94bb-a1a09999339&title=&width=449.0908831341551) **web API的申请:**[https://lbs.amap.com/dev/key](https://lbs.amap.com/dev/key) **申请时注意【服务平台】选项,需要申请【Web端(JS API)】和【Web服务】的key。** ::: 具体的使用,参考sdk使用手册: # 7、路线管理 路线管理是在路线规划中核心的功能,用户在下单时、订单转运单时会进行调用路线规划,后台系统对路线进行维护管理。路线类型如下: - **干线** - 一级转运中心到一级转运中心 - **支线** - 一级转运中心与二级转运中心之间线路 - **接驳路线** - 二级转运中心到网点 - **专线(暂时不支持)** - 任务城市到任意城市 - **临时线路(暂时不支持)** - 任意转运中心到任意转运中心 :::danger 新增路线业务规则:干线:起点终点无顺序,支线:起点必须是二级转运中心,接驳路线:起点必须是网点 ::: ## 7.1、业务流程 ![](https://cdn.nlark.com/yuque/__puml/f54961d9d4e7a6b0caa0c07382e54e1c.svg#lake_card_v2=eyJ0eXBlIjoicHVtbCIsImNvZGUiOiJAc3RhcnR1bWxcblxuYXV0b251bWJlclxuXG5hY3RvciBcIueUqOaIt1wiIGFzIHVzZXJcbnBhcnRpY2lwYW50IFwi6K6i5Y2V5b6u5pyN5YqhXCIgYXMgb21zXG5wYXJ0aWNpcGFudCBcIuWQjuWPsOeuoeeQhuezu-e7n1wiIGFzIGFkbWluXG5wYXJ0aWNpcGFudCBcIndvcmvlvq7mnI3liqFcIiBhcyB3b3JrXG5wYXJ0aWNpcGFudCBcIui3r-e6v-inhOWIkuW-ruacjeWKoVwiIGFzIHRyYW5zcG9ydFxucGFydGljaXBhbnQgXCJOZW80alwiIGFzIG5lbzRqXG5cbnVzZXIgLT4gb21zICsrOiDkuIvljZVcbm9tcyAtPiB0cmFuc3BvcnQgKys6IOi3r-e6v-inhOWIklxudHJhbnNwb3J0IC0-IG5lbzRqICsrOiDmn6Xor6Lot6_nur9cbnJldHVybiDov5Tlm55cbnRyYW5zcG9ydCAtPiBvbXMgLS06IOi_lOWbnui3r-e6v1xub21zIC0-IHVzZXIgOiDkuIvljZXmiJDlip9cbm9tcyAtPiB3b3JrICsrOuiuouWNlei9rOi_kOWNlVxud29yayAtPiB0cmFuc3BvcnQgKys6IOi3r-e6v-inhOWIklxudHJhbnNwb3J0IC0-IHdvcmsgLS06IOi_lOWbnui3r-e6v1xud29yayAtPiB3b3JrIDog5a2Y5YKo5pWw5o2uXG5cbmdyb3VwIOi3r-e6v-S4jeWtmOWcqFxudHJhbnNwb3J0IC0-IG9tcyAtLTog6Lev57q_5LiN5a2Y5ZyoXG5kZWFjdGl2YXRlIHdvcmtcbm9tcyAtPiB1c2VyIC0tOiDkuIvljZXlpLHotKVcbmVuZFxuXG5ncm91cCDot6_nur_nrqHnkIZcbmFkbWluIC0-IHRyYW5zcG9ydCArKzog6Lev57q_566h55CGXG50cmFuc3BvcnQgLT4gbmVvNGogKys6IENSVURcbnJldHVybiDov5Tlm55cbnJldHVybiDov5Tlm55cbmVuZFxuQGVuZHVtbCIsInVybCI6Imh0dHBzOi8vY2RuLm5sYXJrLmNvbS95dXF1ZS9fX3B1bWwvZjU0OTYxZDlkNGU3YTZiMGNhYTBjMDczODJlNTRlMWMuc3ZnIiwiaWQiOiJabWtOOSIsIm1hcmdpbiI6eyJ0b3AiOnRydWUsImJvdHRvbSI6dHJ1ZX0sImNhcmQiOiJkaWFncmFtIn0=)## 7.2、Repository ### 7.2.1、接口定义 ```java package com.sl.transport.repository; import com.sl.transport.common.util.PageResponse; import com.sl.transport.domain.TransportLineNodeDTO; import com.sl.transport.domain.TransportLineSearchDTO; import com.sl.transport.entity.line.TransportLine; import com.sl.transport.entity.node.AgencyEntity; import com.sl.transport.entity.node.BaseEntity; import java.util.List; /** * 运输路线查询 */ public interface TransportLineRepository { /** * 查询两个网点之间最短的路线,查询深度为:10 * * @param start 开始网点 * @param end 结束网点 * @return 路线 */ TransportLineNodeDTO findShortestPath(AgencyEntity start, AgencyEntity end); /** * 查询两个网点之间最短的路线,最大查询深度为:10 * * @param start 开始网点 * @param end 结束网点 * @param depth 查询深度,最大为:10 * @return 路线 */ TransportLineNodeDTO findShortestPath(AgencyEntity start, AgencyEntity end, int depth); /** * 查询两个网点之间的路线列表,成本优先 > 转运节点优先 * * @param start 开始网点 * @param end 结束网点 * @param depth 查询深度 * @param limit 返回路线的数量 * @return 路线 */ List findPathList(AgencyEntity start, AgencyEntity end, int depth, int limit); /** * 查询数据节点之间的关系数量 * * @param firstNode 第一个节点 * @param secondNode 第二个节点 * @return 数量 */ Long queryCount(BaseEntity firstNode, BaseEntity secondNode); /** * 新增路线 * * @param firstNode 第一个节点 * @param secondNode 第二个节点 * @param transportLine 路线数据 * @return 新增关系的数量 */ Long create(BaseEntity firstNode, BaseEntity secondNode, TransportLine transportLine); /** * 更新路线 * * @param transportLine 路线数据 * @return 更新的数量 */ Long update(TransportLine transportLine); /** * 删除路线 * * @param lineId 关系id * @return 删除关系的数量 */ Long remove(Long lineId); /** * 分页查询路线 * * @param transportLineSearchDTO 搜索参数 * @return 路线列表 */ PageResponse queryPageList(TransportLineSearchDTO transportLineSearchDTO); /** * 根据ids批量查询路线 * * @param ids id列表 * @return 路线列表 */ List queryByIds(Long... ids); /** * 根据id查询路线 * * @param id 路线id * @return 路线数据 */ TransportLine queryById(Long id); } ``` ### 7.2.2、接口实现 ```java package com.sl.transport.repository.impl; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.collection.CollUtil; import cn.hutool.core.collection.ListUtil; import cn.hutool.core.convert.Convert; import cn.hutool.core.map.MapUtil; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.PageUtil; import cn.hutool.core.util.StrUtil; import com.sl.transport.common.util.PageResponse; import com.sl.transport.domain.TransportLineNodeDTO; import com.sl.transport.domain.TransportLineSearchDTO; import com.sl.transport.entity.line.TransportLine; import com.sl.transport.entity.node.AgencyEntity; import com.sl.transport.entity.node.BaseEntity; import com.sl.transport.repository.TransportLineRepository; import com.sl.transport.utils.TransportLineUtils; import org.neo4j.driver.Record; import org.neo4j.driver.internal.value.PathValue; import org.neo4j.driver.types.Relationship; import org.springframework.data.neo4j.core.Neo4jClient; import org.springframework.data.neo4j.core.schema.Node; import org.springframework.stereotype.Component; import javax.annotation.Resource; import java.util.Collection; import java.util.List; import java.util.Map; import java.util.Optional; /** * 对于路线的各种操作 */ @Component public class TransportLineRepositoryImpl implements TransportLineRepository { @Resource private Neo4jClient neo4jClient; @Override public TransportLineNodeDTO findShortestPath(AgencyEntity start, AgencyEntity end) { return this.findShortestPath(start, end, 10); } @Override public TransportLineNodeDTO findShortestPath(AgencyEntity start, AgencyEntity end, int depth) { //获取网点数据在Neo4j中的类型 String type = AgencyEntity.class.getAnnotation(Node.class).value()[0]; //构造查询语句 String cypherQuery = StrUtil.format( "MATCH path = shortestPath((start:{}) -[*..{}]-> (end:{}))\n" + "WHERE start.bid = $startId AND end.bid = $endId AND start.status = true AND end.status = true\n" + "RETURN path", type, depth, type); Collection transportLineNodeDTOS = this.executeQueryPath(cypherQuery, start, end); if (CollUtil.isEmpty(transportLineNodeDTOS)) { return null; } for (TransportLineNodeDTO transportLineNodeDTO : transportLineNodeDTOS) { return transportLineNodeDTO; } return null; } private List executeQueryPath(String cypherQuery, AgencyEntity start, AgencyEntity end) { return ListUtil.toList(this.neo4jClient.query(cypherQuery) .bind(start.getBid()).to("startId") //设置参数 .bind(end.getBid()).to("endId") //设置参数 .fetchAs(TransportLineNodeDTO.class) //设置响应的类型 .mappedBy((typeSystem, record) -> { //对结果进行封装处理 PathValue pathValue = (PathValue) record.get(0); return TransportLineUtils.convert(pathValue); }).all()); } @Override public List findPathList(AgencyEntity start, AgencyEntity end, int depth, int limit) { //获取网点数据在Neo4j中的类型 String type = AgencyEntity.class.getAnnotation(Node.class).value()[0]; //构造查询语句 String cypherQuery = StrUtil.format( "MATCH path = (start:{}) -[*..{}]-> (end:{})\n" + "WHERE start.bid = $startId AND end.bid = $endId AND start.status = true AND end.status = true\n" + "UNWIND relationships(path) AS r\n" + "WITH sum(r.cost) AS cost, path\n" + "RETURN path ORDER BY cost ASC, LENGTH(path) ASC LIMIT {}", type, depth, type, limit); return this.executeQueryPath(cypherQuery, start, end); } @Override public Long queryCount(BaseEntity firstNode, BaseEntity secondNode) { String firstNodeType = firstNode.getClass().getAnnotation(Node.class).value()[0]; String secondNodeType = secondNode.getClass().getAnnotation(Node.class).value()[0]; String cypherQuery = StrUtil.format( "MATCH (m:{}) -[r]- (n:{})\n" + "WHERE m.bid = $firstBid AND n.bid = $secondBid\n" + "RETURN count(r) AS c", firstNodeType, secondNodeType); Optional optional = this.neo4jClient.query(cypherQuery) .bind(firstNode.getBid()).to("firstBid") .bind(secondNode.getBid()).to("secondBid") .fetchAs(Long.class) .mappedBy((typeSystem, record) -> Convert.toLong(record.get("c"))) .one(); return optional.orElse(0L); } @Override public Long create(BaseEntity firstNode, BaseEntity secondNode, TransportLine transportLine) { String firstNodeType = firstNode.getClass().getAnnotation(Node.class).value()[0]; String secondNodeType = secondNode.getClass().getAnnotation(Node.class).value()[0]; String cypherQuery = StrUtil.format( "MATCH (m:{} {bid : $firstBid})\n" + "WITH m\n" + "MATCH (n:{} {bid : $secondBid})\n" + "WITH m,n\n" + "CREATE\n" + " (m) -[r:IN_LINE {cost:$cost, number:$number, type:$type, name:$name, distance:$distance, time:$time, extra:$extra, startOrganId:$startOrganId, endOrganId:$endOrganId,created:$created, updated:$updated}]-> (n),\n" + " (m) <-[:OUT_LINE {cost:$cost, number:$number, type:$type, name:$name, distance:$distance, time:$time, extra:$extra, startOrganId:$endOrganId, endOrganId:$startOrganId, created:$created, updated:$updated}]- (n)\n" + "RETURN count(r) AS c", firstNodeType, secondNodeType); Optional optional = this.neo4jClient.query(cypherQuery) .bindAll(BeanUtil.beanToMap(transportLine)) .bind(firstNode.getBid()).to("firstBid") .bind(secondNode.getBid()).to("secondBid") .fetchAs(Long.class) .mappedBy((typeSystem, record) -> Convert.toLong(record.get("c"))) .one(); return optional.orElse(0L); } @Override public Long update(TransportLine transportLine) { String cypherQuery = "MATCH () -[r]-> ()\n" + "WHERE id(r) = $id\n" + "SET r.cost = $cost , r.number = $number, r.name = $name ,r.distance = $distance ,r.time = $time, r.startOrganId = $startOrganId, r.endOrganId = $endOrganId, r.updated = $updated , r.extra = $extra \n" + "RETURN count(r) AS c"; Optional optional = this.neo4jClient.query(cypherQuery) .bindAll(BeanUtil.beanToMap(transportLine)) .fetchAs(Long.class) .mappedBy((typeSystem, record) -> Convert.toLong(record.get("c"))) .one(); return optional.orElse(0L); } @Override public Long remove(Long lineId) { String cypherQuery = "MATCH () -[r]-> ()\n" + "WHERE id(r) = $lineId\n" + "DETACH DELETE r\n" + "RETURN count(r) AS c"; Optional optional = this.neo4jClient.query(cypherQuery) .bind(lineId).to("lineId") .fetchAs(Long.class) .mappedBy((typeSystem, record) -> Convert.toLong(record.get("c"))) .one(); return optional.orElse(0L); } @Override public PageResponse queryPageList(TransportLineSearchDTO transportLineSearchDTO) { int page = Math.max(transportLineSearchDTO.getPage(), 1); int pageSize = transportLineSearchDTO.getPageSize(); int skip = (page - 1) * pageSize; Map searchParam = BeanUtil.beanToMap(transportLineSearchDTO, false, true); MapUtil.removeAny(searchParam, "page", "pageSize"); //构建查询语句,第一个是查询数据,第二个是查询数量 String[] cyphers = this.buildPageQueryCypher(searchParam); String cypherQuery = cyphers[0]; //数据 List list = ListUtil.toList(this.neo4jClient.query(cypherQuery) .bind(skip).to("skip") .bind(pageSize).to("limit") .bindAll(searchParam) .fetchAs(TransportLine.class) .mappedBy((typeSystem, record) -> { //封装数据 return this.toTransportLine(record); }).all()); // 数据总数 String countCypher = cyphers[1]; Long total = this.neo4jClient.query(countCypher) .bindAll(searchParam) .fetchAs(Long.class) .mappedBy((typeSystem, record) -> Convert.toLong(record.get("c"))) .one().orElse(0L); PageResponse pageResponse = new PageResponse<>(); pageResponse.setPage(page); pageResponse.setPageSize(pageSize); pageResponse.setItems(list); pageResponse.setCounts(total); Long pages = Convert.toLong(PageUtil.totalPage(Convert.toInt(total), pageSize)); pageResponse.setPages(pages); return pageResponse; } private String[] buildPageQueryCypher(Map searchParam) { String queryCypher; String countCypher; if (CollUtil.isEmpty(searchParam)) { //无参数 queryCypher = "MATCH (m) -[r]-> (n) RETURN m,r,n ORDER BY id(r) DESC SKIP $skip LIMIT $limit"; countCypher = "MATCH () -[r]-> () RETURN count(r) AS c"; } else { //有参数 String cypherPrefix = "MATCH (m) -[r]-> (n)"; StringBuilder sb = new StringBuilder(); sb.append(cypherPrefix).append(" WHERE 1=1 "); for (String key : searchParam.keySet()) { Object value = searchParam.get(key); if (value instanceof String) { if (StrUtil.isNotBlank(Convert.toStr(value))) { sb.append(StrUtil.format("AND r.{} CONTAINS ${} \n", key, key)); } } else { sb.append(StrUtil.format("AND r.{} = ${} \n", key, key)); } } String cypher = sb.toString(); queryCypher = cypher + "RETURN m,r,n ORDER BY id(r) DESC SKIP $skip LIMIT $limit"; countCypher = cypher + "RETURN count(r) AS c"; } return new String[]{queryCypher, countCypher}; } @Override public List queryByIds(Long... ids) { String cypherQuery = "MATCH (m) -[r]-> (n)\n" + "WHERE id(r) in $ids\n" + "RETURN m,r,n"; return ListUtil.toList(this.neo4jClient.query(cypherQuery) .bind(ids).to("ids") .fetchAs(TransportLine.class) .mappedBy((typeSystem, record) -> { //封装数据 return this.toTransportLine(record); }).all()); } private TransportLine toTransportLine(Record record) { org.neo4j.driver.types.Node startNode = record.get("m").asNode(); org.neo4j.driver.types.Node endNode = record.get("n").asNode(); Relationship relationship = record.get("r").asRelationship(); Map map = relationship.asMap(); TransportLine transportLine = BeanUtil.toBeanIgnoreError(map, TransportLine.class); transportLine.setStartOrganName(startNode.get("name").asString()); transportLine.setStartOrganId(startNode.get("bid").asLong()); transportLine.setEndOrganName(endNode.get("name").asString()); transportLine.setEndOrganId(endNode.get("bid").asLong()); transportLine.setId(relationship.id()); return transportLine; } @Override public TransportLine queryById(Long id) { List transportLines = this.queryByIds(id); if (CollUtil.isNotEmpty(transportLines)) { return transportLines.get(0); } return null; } } ``` ## 7.3、路线Service ### 7.3.1、接口定义 ```java package com.sl.transport.service; import com.sl.transport.common.util.PageResponse; import com.sl.transport.domain.TransportLineNodeDTO; import com.sl.transport.domain.TransportLineSearchDTO; import com.sl.transport.entity.line.TransportLine; import java.util.List; /** * 计算路线相关业务 */ public interface TransportLineService { /** * 新增路线 * * @param transportLine 路线数据 * @return 是否成功 */ Boolean createLine(TransportLine transportLine); /** * 更新路线 * * @param transportLine 路线数据 * @return 是否成功 */ Boolean updateLine(TransportLine transportLine); /** * 删除路线 * * @param id 路线id * @return 是否成功 */ Boolean deleteLine(Long id); /** * 分页查询路线 * * @param transportLineSearchDTO 搜索参数 * @return 路线列表 */ PageResponse queryPageList(TransportLineSearchDTO transportLineSearchDTO); /** * 查询两个网点之间最短的路线,最大查询深度为:10 * * @param startId 开始网点id * @param endId 结束网点id * @return 路线 */ TransportLineNodeDTO queryShortestPath(Long startId, Long endId); /** * 查询两个网点之间成本最低的路线,最大查询深度为:10 * * @param startId 开始网点id * @param endId 结束网点id * @return 路线集合 */ TransportLineNodeDTO findLowestPath(Long startId, Long endId); /** * 根据调度策略查询路线 * * @param startId 开始网点id * @param endId 结束网点id * @return 路线 */ TransportLineNodeDTO queryPathByDispatchMethod(Long startId, Long endId); /** * 根据ids批量查询路线 * * @param ids id列表 * @return 路线列表 */ List queryByIds(Long... ids); /** * 根据id查询路线 * * @param id 路线id * @return 路线数据 */ TransportLine queryById(Long id); } ``` ### 7.3.2、接口实现 ```java package com.sl.transport.service.impl; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.bean.copier.CopyOptions; import cn.hutool.core.collection.CollUtil; import cn.hutool.core.convert.Convert; import cn.hutool.core.map.MapUtil; import cn.hutool.core.util.NumberUtil; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.StrUtil; import cn.hutool.json.JSONObject; import cn.hutool.json.JSONUtil; import com.itheima.em.sdk.EagleMapTemplate; import com.itheima.em.sdk.enums.ProviderEnum; import com.itheima.em.sdk.vo.Coordinate; import com.sl.transport.common.exception.SLException; import com.sl.transport.common.util.PageResponse; import com.sl.transport.domain.*; import com.sl.transport.entity.line.TransportLine; import com.sl.transport.entity.node.AgencyEntity; import com.sl.transport.entity.node.BaseEntity; import com.sl.transport.entity.node.OLTEntity; import com.sl.transport.entity.node.TLTEntity; import com.sl.transport.enums.DispatchMethodEnum; import com.sl.transport.enums.ExceptionEnum; import com.sl.transport.enums.TransportLineEnum; import com.sl.transport.repository.TransportLineRepository; import com.sl.transport.service.CostConfigurationService; import com.sl.transport.service.DispatchConfigurationService; import com.sl.transport.service.OrganService; import com.sl.transport.service.TransportLineService; import org.springframework.stereotype.Service; import javax.annotation.Resource; import java.util.List; import java.util.Map; import java.util.stream.Collectors; /** * 路线相关业务 * * @author zzj * @version 1.0 */ @Service public class TransportLineServiceImpl implements TransportLineService { @Resource private TransportLineRepository transportLineRepository; @Resource private EagleMapTemplate eagleMapTemplate; @Resource private OrganService organService; @Resource private DispatchConfigurationService dispatchConfigurationService; @Resource private CostConfigurationService costConfigurationService; // 新增路线业务规则:干线:起点终点无顺序,支线:起点必须是二级转运中心,接驳路线:起点必须是网点 @Override public Boolean createLine(TransportLine transportLine) { TransportLineEnum transportLineEnum = TransportLineEnum.codeOf(transportLine.getType()); if (null == transportLineEnum) { throw new SLException(ExceptionEnum.TRANSPORT_LINE_TYPE_ERROR); } if (ObjectUtil.equal(transportLine.getStartOrganId(), transportLine.getEndOrganId())) { //起点终点不能相同 throw new SLException(ExceptionEnum.TRANSPORT_LINE_ORGAN_CANNOT_SAME); } BaseEntity firstNode; BaseEntity secondNode; switch (transportLineEnum) { case TRUNK_LINE: { // 干线 firstNode = OLTEntity.builder().bid(transportLine.getStartOrganId()).build(); secondNode = OLTEntity.builder().bid(transportLine.getEndOrganId()).build(); break; } case BRANCH_LINE: { // 支线,起点必须是 二级转运中心 firstNode = TLTEntity.builder().bid(transportLine.getStartOrganId()).build(); secondNode = OLTEntity.builder().bid(transportLine.getEndOrganId()).build(); break; } case CONNECT_LINE: { // 接驳路线,起点必须是 网点 firstNode = AgencyEntity.builder().bid(transportLine.getStartOrganId()).build(); secondNode = TLTEntity.builder().bid(transportLine.getEndOrganId()).build(); break; } default: { throw new SLException(ExceptionEnum.TRANSPORT_LINE_TYPE_ERROR); } } if (ObjectUtil.hasEmpty(firstNode, secondNode)) { throw new SLException(ExceptionEnum.START_END_ORGAN_NOT_FOUND); } //判断路线是否已经存在 Long count = this.transportLineRepository.queryCount(firstNode, secondNode); if (count > 0) { throw new SLException(ExceptionEnum.TRANSPORT_LINE_ALREADY_EXISTS); } transportLine.setId(null); transportLine.setCreated(System.currentTimeMillis()); transportLine.setUpdated(transportLine.getCreated()); //补充信息 this.infoFromMap(firstNode, secondNode, transportLine); count = this.transportLineRepository.create(firstNode, secondNode, transportLine); return count > 0; } /** * 通过地图查询距离、时间,计算成本 * * @param firstNode 开始节点 * @param secondNode 结束节点 * @param transportLine 路线对象 */ private void infoFromMap(BaseEntity firstNode, BaseEntity secondNode, TransportLine transportLine) { //查询节点数据 OrganDTO startOrgan = this.organService.findByBid(firstNode.getBid()); if (ObjectUtil.hasEmpty(startOrgan, startOrgan.getLongitude(), startOrgan.getLatitude())) { throw new SLException("请先完善机构信息"); } OrganDTO endOrgan = this.organService.findByBid(secondNode.getBid()); if (ObjectUtil.hasEmpty(endOrgan, endOrgan.getLongitude(), endOrgan.getLatitude())) { throw new SLException("请先完善机构信息"); } //查询地图服务商 Coordinate origin = new Coordinate(startOrgan.getLongitude(), startOrgan.getLatitude()); Coordinate destination = new Coordinate(endOrgan.getLongitude(), endOrgan.getLatitude()); //设置高德地图参数,默认是不返回预计耗时的,需要额外设置参数 Map param = MapUtil.builder().put("show_fields", "cost").build(); String driving = this.eagleMapTemplate.opsForDirection().driving(ProviderEnum.AMAP, origin, destination, param); if (StrUtil.isEmpty(driving)) { return; } JSONObject jsonObject = JSONUtil.parseObj(driving); //时间,单位:秒 Long duration = Convert.toLong(jsonObject.getByPath("route.paths[0].cost.duration"), -1L); transportLine.setTime(duration); //距离,单位:米 Double distance = Convert.toDouble(jsonObject.getByPath("route.paths[0].distance"), -1d); transportLine.setDistance(NumberUtil.round(distance, 0).doubleValue()); // 总成本 = 每公里平均成本 * 距离(单位:米) / 1000 Double cost = costConfigurationService.findCostByType(transportLine.getType()); transportLine.setCost(NumberUtil.round(cost * distance / 1000, 2).doubleValue()); } @Override public Boolean updateLine(TransportLine transportLine) { // 先查后改 TransportLine transportLineData = this.queryById(transportLine.getId()); if (null == transportLineData) { throw new SLException(ExceptionEnum.TRANSPORT_LINE_NOT_FOUND); } //拷贝数据,忽略null值以及不能修改的字段 BeanUtil.copyProperties(transportLine, transportLineData, CopyOptions.create().setIgnoreNullValue(true) .setIgnoreProperties("type", "startOrganId", "startOrganName", "endOrganId", "endOrganName")); transportLineData.setUpdated(System.currentTimeMillis()); Long count = this.transportLineRepository.update(transportLineData); return count > 0; } @Override public Boolean deleteLine(Long id) { Long count = this.transportLineRepository.remove(id); return count > 0; } @Override public PageResponse queryPageList(TransportLineSearchDTO transportLineSearchDTO) { return this.transportLineRepository.queryPageList(transportLineSearchDTO); } @Override public TransportLineNodeDTO queryShortestPath(Long startId, Long endId) { AgencyEntity start = AgencyEntity.builder().bid(startId).build(); AgencyEntity end = AgencyEntity.builder().bid(endId).build(); if (ObjectUtil.hasEmpty(start, end)) { throw new SLException(ExceptionEnum.START_END_ORGAN_NOT_FOUND); } return this.transportLineRepository.findShortestPath(start, end); } @Override public TransportLineNodeDTO findLowestPath(Long startId, Long endId) { AgencyEntity start = AgencyEntity.builder().bid(startId).build(); AgencyEntity end = AgencyEntity.builder().bid(endId).build(); if (ObjectUtil.hasEmpty(start, end)) { throw new SLException(ExceptionEnum.START_END_ORGAN_NOT_FOUND); } List pathList = this.transportLineRepository.findPathList(start, end, 10, 1); if (CollUtil.isNotEmpty(pathList)) { return pathList.get(0); } return null; } /** * 根据调度策略查询路线 * * @param startId 开始网点id * @param endId 结束网点id * @return 路线 */ @Override public TransportLineNodeDTO queryPathByDispatchMethod(Long startId, Long endId) { //调度方式配置 DispatchConfigurationDTO configuration = this.dispatchConfigurationService.findConfiguration(); int method = configuration.getDispatchMethod(); //调度方式,1转运次数最少,2成本最低 if (ObjectUtil.equal(DispatchMethodEnum.SHORTEST_PATH.getCode(), method)) { return this.queryShortestPath(startId, endId); } else { return this.findLowestPath(startId, endId); } } @Override public List queryByIds(Long... ids) { return this.transportLineRepository.queryByIds(ids); } @Override public TransportLine queryById(Long id) { return this.transportLineRepository.queryById(id); } } ``` ## 7.4、路线成本 ### 7.4.1、需求 在后台系统中,可以针对路线成本进行设置: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1672739245626-9374c750-b539-4a28-9577-c504eba8a817.png#averageHue=%23fefbfb&clientId=ud4740818-830b-4&from=paste&height=394&id=u29ee4b71&name=image.png&originHeight=591&originWidth=810&originalType=binary&ratio=1&rotation=0&showTitle=false&size=23672&status=done&style=shadow&taskId=u612bd544-b621-4587-9924-df94b7e232b&title=&width=540) 计算路线成本:距离 * 每公里平均成本 ### 7.4.2、Controller ```java package com.sl.transport.controller; import com.sl.transport.domain.CostConfigurationDTO; import com.sl.transport.service.CostConfigurationService; import io.swagger.annotations.Api; import io.swagger.annotations.ApiOperation; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.*; import javax.annotation.Resource; import java.util.List; /** * 成本配置相关业务对外提供接口服务 */ @Api(tags = "成本配置") @RequestMapping("cost-configuration") @Validated @RestController public class CostConfigurationController { @Resource private CostConfigurationService costConfigurationService; @ApiOperation(value = "查询成本配置") @GetMapping public List findConfiguration() { return costConfigurationService.findConfiguration(); } @ApiOperation(value = "保存成本配置") @PostMapping public void saveConfiguration(@RequestBody List dto) { costConfigurationService.saveConfiguration(dto); } } ``` ### 7.4.3、Service ```java package com.sl.transport.service; import com.sl.transport.domain.CostConfigurationDTO; import java.util.List; /** * 成本配置相关业务 */ public interface CostConfigurationService { /** * 查询成本配置 * * @return 成本配置 */ List findConfiguration(); /** * 保存成本配置 * @param dto 成本配置 */ void saveConfiguration(List dto); /** * 查询成本根据类型 * @param type 类型 * @return 成本 */ Double findCostByType(Integer type); } ``` ### 7.4.4、ServiceImpl ```java package com.sl.transport.service.impl; import cn.hutool.core.convert.Convert; import cn.hutool.core.util.ObjectUtil; import com.sl.transport.common.exception.SLException; import com.sl.transport.domain.CostConfigurationDTO; import com.sl.transport.enums.ExceptionEnum; import com.sl.transport.enums.TransportLineEnum; import com.sl.transport.service.CostConfigurationService; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.stereotype.Service; import javax.annotation.Resource; import java.util.List; import java.util.Map; import java.util.stream.Collectors; /** * 成本配置相关业务 */ @Service public class CostConfigurationServiceImpl implements CostConfigurationService { /** * 成本配置 redis key */ private static final String SL_TRANSPORT_COST_REDIS_KEY = "SL_TRANSPORT_COST_CONFIGURATION"; /** * 默认成本配置 */ private static final Map DEFAULT_COST = Map.of( TransportLineEnum.TRUNK_LINE.getCode(), 0.8, TransportLineEnum.BRANCH_LINE.getCode(), 1.2, TransportLineEnum.CONNECT_LINE.getCode(), 1.5); @Resource private StringRedisTemplate stringRedisTemplate; /** * 查询成本配置 * * @return 成本配置 */ @Override public List findConfiguration() { Map entries = stringRedisTemplate.opsForHash().entries(SL_TRANSPORT_COST_REDIS_KEY); if (ObjectUtil.isEmpty(entries)) { // 使用默认值 entries = DEFAULT_COST; } // 返回 return entries.entrySet().stream() .map(v -> new CostConfigurationDTO(Convert.toInt(v.getKey()), Convert.toDouble(v.getValue()))) .collect(Collectors.toList()); } /** * 保存成本配置 * * @param dto 成本配置 */ @Override public void saveConfiguration(List dto) { Map map = dto.stream().collect(Collectors.toMap(v -> v.getTransportLineType().toString(), v -> v.getCost().toString())); stringRedisTemplate.opsForHash().putAll(SL_TRANSPORT_COST_REDIS_KEY, map); } /** * 查询成本根据类型 * * @param type 类型 * @return 成本 */ @Override public Double findCostByType(Integer type) { if (ObjectUtil.isEmpty(type)) { throw new SLException(ExceptionEnum.TRANSPORT_LINE_TYPE_ERROR); } // 查询redis Object o = stringRedisTemplate.opsForHash().get(SL_TRANSPORT_COST_REDIS_KEY, type.toString()); if (ObjectUtil.isNotEmpty(o)) { return Convert.toDouble(o); } // 返回默认值 return Convert.toDouble(DEFAULT_COST.get(type)); } } ``` ## ## 7.5、测试 新增路线: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661563052056-34141345-7bc5-41f2-a34d-a3b9beae46d5.png#averageHue=%23fdfdfd&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=312&id=u3fb6a21c&name=image.png&originHeight=515&originWidth=1416&originalType=binary&ratio=1&rotation=0&showTitle=false&size=49702&status=error&style=shadow&taskId=u75facf2f-54c9-4a16-8476-75f217ab7d0&title=&width=858.1817685802478) 新增成功: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661563089299-0d3cc317-2c7d-4854-9b2a-a250a9cb942d.png#averageHue=%23f3f9fd&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=293&id=u544fce89&name=image.png&originHeight=483&originWidth=718&originalType=binary&ratio=1&rotation=0&showTitle=false&size=23729&status=error&style=shadow&taskId=u863170e9-d353-4e4d-b720-ffe463467c5&title=&width=435.15149000043635) 同理可以测试其他类型路线。 查询路线列表: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661563794201-c7b75693-9647-42d4-be58-9ecba56c617d.png#averageHue=%23fdfdfd&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=419&id=uec442c85&name=image.png&originHeight=691&originWidth=1411&originalType=binary&ratio=1&rotation=0&showTitle=false&size=49703&status=error&style=shadow&taskId=u364dfcc4-b31f-4af8-9884-66640514563&title=&width=855.1514657250915) 查询到数据: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661563882190-ec560f70-da82-42bf-9a1a-ca00757f810a.png#averageHue=%23fbfbfb&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=433&id=u4d51d732&name=image.png&originHeight=714&originWidth=1242&originalType=binary&ratio=1&rotation=0&showTitle=false&size=133835&status=error&style=shadow&taskId=ud737ac06-6c74-4c8a-b93a-5af01142089&title=&width=752.7272292208106) # 8、综合测试 ## 8.1、功能测试 下面我们可以整合到后台管理系统中进行测试。 查询路线: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661567816771-4d0b6e4e-a281-47f7-8498-c0b6e1258e09.png#averageHue=%23e37858&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=498&id=u852f2d0f&name=image.png&originHeight=821&originWidth=1617&originalType=binary&ratio=1&rotation=0&showTitle=false&size=68566&status=error&style=shadow&taskId=ucee044c8-efdf-4878-870b-f2620c460a7&title=&width=979.9999433575288) 新增路线: ![image.png](https://cdn.nlark.com/yuque/0/2023/png/27683667/1673420573169-6f4dad60-3b47-4908-ab3a-3cad22fe83c7.png#averageHue=%238c4832&clientId=ufea784ab-bee4-4&from=paste&height=471&id=u9908e336&name=image.png&originHeight=706&originWidth=1214&originalType=binary&ratio=1&rotation=0&showTitle=false&size=59584&status=done&style=shadow&taskId=u463b67fb-6b6e-44ee-a10e-908cd476500&title=&width=809.3333333333334) :::danger 新增路线时路线的距离和成本系统会自动进行计算,距离是通过高德地图服务查询的实际距离,成本按照所设置的成本进行计算(同一标准在计算路线时是可行的,但是不能作为真实的成本进行利润计算),在编辑路线时可以修改距离和成本。 ::: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661568488182-5d633bbc-c8ab-4c4b-8689-c5b45d773e33.png#averageHue=%23fdfbfa&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=290&id=uddd76d10&name=image.png&originHeight=478&originWidth=1594&originalType=binary&ratio=1&rotation=0&showTitle=false&size=60281&status=error&style=shadow&taskId=ue84e740f-1f23-4509-9a36-26fcf3c4f0d&title=&width=966.06055022381) 完善下数据: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661571304268-b79030a8-ac45-4f06-a679-71ffb05d9669.png#averageHue=%23fdfbfb&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=416&id=u1bca9f09&name=image.png&originHeight=687&originWidth=1558&originalType=binary&ratio=1&rotation=0&showTitle=false&size=90879&status=error&style=shadow&taskId=u619283e6-f5db-4883-82eb-7d8c376b374&title=&width=944.242369666685) ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661571317227-0fe2f0b7-6eb6-40c7-9b64-9445e9100bf0.png#averageHue=%23f1f8fc&clientId=udb28364f-6735-4&errorMessage=unknown%20error&from=paste&height=237&id=ue964d422&name=image.png&originHeight=391&originWidth=698&originalType=binary&ratio=1&rotation=0&showTitle=false&size=30946&status=error&style=shadow&taskId=u77ff48c3-8ffd-4095-9b99-bbff6668ce6&title=&width=423.0302785798114) ## 8.2、Jenkins构建任务 如果在路线下没有service的构建任务,就需要创建一个构建任务: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661742167871-1c9d6179-14c1-4b64-96c1-26d53c01fcc6.png#averageHue=%234eb971&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=270&id=u41b0c6f5&name=image.png&originHeight=445&originWidth=1333&originalType=binary&ratio=1&rotation=0&showTitle=false&size=63500&status=error&style=shadow&taskId=u9a065527-1432-475d-92cd-31f5fbea29a&title=&width=807.8787411846541) 点击新建任务: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661742269482-69f4ad57-f803-4328-9cd1-63fc0a6b9ab9.png#averageHue=%23fefefe&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=429&id=uf03cdc8e&name=image.png&originHeight=708&originWidth=353&originalType=binary&ratio=1&rotation=0&showTitle=false&size=45089&status=error&style=shadow&taskId=ubc6b6799-5776-4bca-8c8b-8d3edc9e1ed&title=&width=213.9393815740307) 输入任务名称,名称与工程名一致: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661742353224-2f050d2c-7d38-4d41-b11f-395e1d57ded3.png#averageHue=%23edeef2&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=146&id=u8a96be72&name=image.png&originHeight=241&originWidth=855&originalType=binary&ratio=1&rotation=0&showTitle=false&size=15425&status=error&style=shadow&taskId=ue96b5539-4831-406e-aa86-1dfc6418afb&title=&width=518.1817882317174) 选择复制一个已有的任务: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661742375380-3885225a-e385-41a2-b0ce-3a862261f3a6.png#averageHue=%23e8eaef&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=240&id=u30c37070&name=image.png&originHeight=396&originWidth=1109&originalType=binary&ratio=1&rotation=0&showTitle=false&size=36869&status=error&style=shadow&taskId=ua1b8d6b6-4862-4465-a22d-c8b02dbe7cb&title=&width=672.1211732736546) 设置描述: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661743413625-decbbaaa-232c-4a3f-a1b7-7ff37dc24a59.png#averageHue=%23fefefe&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=233&id=u42ea34c4&name=image.png&originHeight=384&originWidth=1342&originalType=binary&ratio=1&rotation=0&showTitle=false&size=19042&status=error&style=shadow&taskId=u9f884d44-aaaf-42d1-b897-891ff3b868f&title=&width=813.3332863239355) 端口设置为:18083: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661743471714-42a66932-1409-4349-8bea-f143f4b97cfb.png#averageHue=%23fefefe&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=382&id=udf72d9ef&name=image.png&originHeight=630&originWidth=1220&originalType=binary&ratio=1&rotation=0&showTitle=false&size=30079&status=error&style=shadow&taskId=uc8123ba3-6b10-48d3-9abd-f68298cc568&title=&width=739.3938966581231) 设置名称: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661743509446-91f2dc94-de6e-447c-aadd-10735c9b558c.png#averageHue=%23fefefe&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=445&id=u2fc8bc1f&name=image.png&originHeight=735&originWidth=1200&originalType=binary&ratio=1&rotation=0&showTitle=false&size=39505&status=error&style=shadow&taskId=u26e338c7-381d-462e-b9c5-8046832c865&title=&width=727.2726852374981) 设置git地址: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661743547637-8c8f0415-81af-4450-86ab-9e3d53febf9d.png#averageHue=%23fefefe&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=388&id=uaa4203e5&name=image.png&originHeight=641&originWidth=1253&originalType=binary&ratio=1&rotation=0&showTitle=false&size=36381&status=error&style=shadow&taskId=u264402ca-dd04-4e3f-b745-fdc532b6c42&title=&width=759.3938955021542) 点击保存: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661743579353-4db2b383-38ea-41fd-b5bf-8076107686eb.png#averageHue=%23fefefe&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=204&id=u05f485a7&name=image.png&originHeight=337&originWidth=665&originalType=binary&ratio=1&rotation=0&showTitle=false&size=15429&status=error&style=shadow&taskId=u9c6671fb-ed42-4232-9cd6-13d7532f275&title=&width=403.0302797357802) 测试构建: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661744152310-f87b8eec-e8ae-4299-aafb-390eb8a94065.png#averageHue=%23f1f1f1&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=284&id=u47109b2f&name=image.png&originHeight=469&originWidth=1179&originalType=binary&ratio=1&rotation=0&showTitle=false&size=86729&status=error&style=shadow&taskId=u0d520419-44a1-49eb-8195-042b59eb282&title=&width=714.5454132458419) 测试:[http://192.168.150.101:18083/doc.html](http://192.168.150.101:18083/doc.html) 可以正常查询到数据: ![image.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1661744272620-42de1f69-f50e-45f0-bc5a-3be3e4f8e5cf.png#averageHue=%23fbfbfb&clientId=ufb7a4f96-8471-4&errorMessage=unknown%20error&from=paste&height=415&id=u2ec17425&name=image.png&originHeight=685&originWidth=1247&originalType=binary&ratio=1&rotation=0&showTitle=false&size=133523&status=error&style=shadow&taskId=u01232123-fe85-4c41-a9fb-77a49ce4547&title=&width=755.7575320759668) ## 8.3、导入数据 前面都是基于测试数据进行测试的,实际上我们已经构造了一些初始数据,可以直接导入使用,具体命令如下: ```shell #停止neo4j微服务 docker stop neo4j #将数据文件neo4j.dump上传到挂载目录下:/var/lib/docker/volumes/neo4j/_data #执行如下命令进行导入 #参数:--from:指定dump文件路径,--force:强制替换现有库 docker run -it --rm -v neo4j:/data neo4j:4.4.5 neo4j-admin load --from=/data/neo4j.dump --force #启动neo4j微服务 docker start neo4j #在管理工具中查询:match (n) return n #如果需要备份数据,可以用dump命令进行导出数据(导出也是需要先停止服务),示例如下: docker run -it --rm -v neo4j:/data neo4j:4.4.5 neo4j-admin dump --to=/data/neo4j2.dump --database=neo4j ``` 导入的数据如下: ![graph.png](https://cdn.nlark.com/yuque/0/2022/png/27683667/1666612951609-0fd6ba6a-803a-450a-a2a1-077e069350c1.png#averageHue=%23090504&clientId=ub06bfded-37c8-4&errorMessage=unknown%20error&from=ui&id=uf773c47a&name=graph.png&originHeight=1074&originWidth=906&originalType=binary&ratio=1&rotation=0&showTitle=false&size=220448&status=error&style=shadow&taskId=u006a2865-0d66-44a9-a1cf-9017a07adfc&title=) > 由于导入数据会覆盖之前插入的测试数据,会导致数据只存在权限系统中,出现脏数据,需要手动删除权限系统中对应的测试数据。 # 9、练习 ## 9.1、练习1 今日代码量较大,动手编写代码实现相关的业务功能。 ## 9.2、练习2 自己构造数据,进一步的完善数据和熟悉业务流程。 # 10、面试连环问 :::info 面试官问: - 你们物流项目中的路线规划是怎么做的? - 如何确定路线的成本和距离?成本计算规则是什么?该成本会计算到公司利润核算中吗? - 对于路线的往返你们是怎么设计的?为什么成对创建的? - 路线支持修改起点或终点机构吗?请说明理由。 :::