前言
本章主要做了Eureka的服务注册和发现,实现了负载均衡,虽然理论上来说应该很快就能做完,因为代码实际上添加只有几行,但又被硬控一个下午加半个晚上,因为按照黑马的视频来做出现各种问题,同时他的源码因为后面替换成了nacos所以关于eureka的部分有删减,且我不好直接复刻,好在最后还是折腾出来了,如果只是想查看Eureka错误解决方案可以直接到文章末尾部分。(黑马的视频讲解确实很详细,但会经典白学以及其代码距离现在有些时日,经常会报奇怪的错)
以及后续章节使用为nacos完成服务注册和发现,所以不想白学也可直接看下一章节。
本章代码基于第1章项目,前置源码可在第1章博客下载,博客链接如下:
从零搭建微服务项目(微服务学习——第1章)-优快云博客https://blog.youkuaiyun.com/wlf2030/article/details/145229577https://blog.youkuaiyun.com/wlf2030/article/details/145229577https://blog.youkuaiyun.com/wlf2030/article/details/145229577https://blog.youkuaiyun.com/wlf2030/article/details/145229577简要介绍前置项目流程:order-service以及user-service两服务分别连接order-db以及user-db两数据库,order-db中仅有user-id,user-info存在user-db中,为提供完整order-info,order-service通过ip加端口(硬编码)发现user-service服务地址并调用其端口拿取user-info结合从order-db中拿取的信息返回给前端。
本项目源码链接如下:
wlf728050719/SpringCloudBase2https://github.com/wlf728050719/SpringCloudBase2https://github.com/wlf728050719/SpringCloudBase2https://github.com/wlf728050719/SpringCloudBase2https://github.com/wlf728050719/SpringCloudBase2以及本专栏会持续更新微服务项目,每一章的项目都会基于前一章项目进行功能的完善,欢迎小伙伴们关注!同时如果只是对单章感兴趣也不用从头看,只需下载前一章项目即可,每一章都会有前置项目准备部分,跟着操作就能实现上一章的最终效果,当然如果是一直跟着做可以直接跳过这一部分。专栏目录链接如下,其中Base篇为基础微服务搭建,Pro篇为复杂模块实现。
从零搭建微服务项目(全)-优快云博客https://blog.youkuaiyun.com/wlf2030/article/details/145799620
一、前置项目准备
1.从github下载前一章的项目解压,重命名文件夹为Base2打开
2.重命名模块名为Base2
3.重命名后检测maven是否变成下面结构
4.配置两服务数据源,换成你自己的密码(文件夹里有sql文件,可以创建对应数据库)
5.启动测试
6.测试成功
二、Eureka服务注册
1.新建maven模块
2.eureka的pom.xml导入依赖,即添加eureka服务器的依赖
3.创建resources目录yml文件以及主类
4.主类内容如下
package cn.bit.eurekaserver;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.server.EnableEurekaServer;
@EnableEurekaServer
@SpringBootApplication
public class EurekaApplication {
public static void main(String[] args) {
SpringApplication.run(EurekaApplication.class, args);
}
}
5.配置order-service和user-service的pom.xml,即添加eureka客户端的依赖
6.配置Base2的pom文件,为每种环境添加eureka的端口
7.eureka的yml内容如下(记得替换数据源密码)
server:
port: @eureka.server.port@
spring:
application:
name: eurekaserver
datasource:
url: jdbc:mysql://localhost:3306/db_order?useSSL=false
username: root
password: 15947035212
driver-class-name: com.mysql.jdbc.Driver
eureka:
client:
fetch-registry: false
service-url:
defaultZone: http://127.0.0.1:@eureka.server.port@/eureka
8.order-service的yml内容如下(记得替换数据源密码)
server:
port: @order.service.port@
eureka:
client:
service-url:
defaultZone: http://127.0.0.1:@eureka.server.port@/eureka
fetch-registry: true
registryFetchIntervalSeconds: 5
spring:
application:
name: order-service
datasource:
url: jdbc:mysql://localhost:3306/db_order?useSSL=false
username: root
password: 15947035212
driver-class-name: com.mysql.jdbc.Driver
mybatis:
mapper-locations: classpath:mapper/*Mapper.xml
type-aliases-package: cn.bit.orderservice.mapper
9.user-service的yml内容如下:(记得替换数据源密码)
server:
port: @user.service.port@
eureka:
client:
service-url:
defaultZone: http://127.0.0.1:@eureka.server.port@/eureka
fetch-registry: true
registryFetchIntervalSeconds: 5
spring:
application:
name: user-service
datasource:
url: jdbc:mysql://localhost:3306/db_user?useSSL=false
username: root
password: 15947035212
driver-class-name: com.mysql.jdbc.Driver
mybatis:
type-aliases-package: cn.bit.userservice.bean.po
configuration:
map-underscore-to-camel-case: true
10.修改OrderServiceApplicarion类,其实也就加了个@LoadBalanced
package cn.bit.orderservice;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
import org.springframework.context.annotation.Bean;
import org.springframework.web.client.RestTemplate;
@SpringBootApplication
@MapperScan("cn.bit.orderservice.mapper")
public class OrderServiceApplication {
public static void main(String[] args) {
SpringApplication.run(OrderServiceApplication.class, args);
}
@Bean
@LoadBalanced
RestTemplate restTemplate() {
return new RestTemplate();
}
}
11.启动三个服务(启动失败可参考文末错误集锦部分)
12.点击访问eureka的端口(三个服务都能看见,但不代表稳了,可能访问不了)
三、Eureka服务相互调用
1.更改原来order-serviece服务的serviceImpl方法(其实也就替换了url)
package cn.bit.orderservice.service.impl;
import cn.bit.orderservice.bean.dto.UserBaseInfoDTO;
import cn.bit.orderservice.bean.po.OrderPO;
import cn.bit.orderservice.bean.vo.OrderInfoVO;
import cn.bit.orderservice.bean.vo.R;
import cn.bit.orderservice.mapper.OrderMapper;
import cn.bit.orderservice.service.OrderService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.ParameterizedTypeReference;
import org.springframework.http.HttpMethod;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
private OrderMapper orderMapper;
@Autowired
private RestTemplate restTemplate;
@Override
public OrderInfoVO getOrderInfoById(int orderId) {
OrderPO orderPO = orderMapper.getOrderPOById(orderId);
if (orderPO == null)
return null;
OrderInfoVO orderInfoVO = new OrderInfoVO();
orderInfoVO.setOrderId(orderPO.getId());
orderInfoVO.setAmount(orderPO.getAmount());
orderInfoVO.setCreateTime(orderPO.getCreateTime());
orderInfoVO.setGetLocation(orderPO.getGetLocation());
try {
// 获取买家信息
String url1 = "http://user-service/user/baseInfo/" + orderPO.getBuyerId();
R<UserBaseInfoDTO> response1 = restTemplate.exchange(url1, HttpMethod.GET, null, new ParameterizedTypeReference<R<UserBaseInfoDTO>>() {
}).getBody();
UserBaseInfoDTO userBaseInfo1 = response1.getData();
orderInfoVO.setBuyerName(userBaseInfo1.getUsername());
// 获取卖家信息
String url2 = "http://user-service/user/baseInfo/" + orderPO.getSellerId();
R<UserBaseInfoDTO> response2 = restTemplate.exchange(url2, HttpMethod.GET, null, new ParameterizedTypeReference<R<UserBaseInfoDTO>>() {
}).getBody();
UserBaseInfoDTO userBaseInfo2 = response2.getData();
orderInfoVO.setSellerName(userBaseInfo2.getUsername());
} catch (Exception e) {
System.out.println("user-service connect error");
e.printStackTrace();
}
return orderInfoVO;
}
}
2.启动服务,等待5s,最好10s(信我先别急,不然等会报错你就舒服了)
3.出现getting all instance registry后就可以进入order-service测试了(出现各种报错或buyerName为null可见文末错误集锦)
四、负载均衡测试
1.在user-service的controller中添加一句调试输出
2.复制实例,右键UserServiceApplication选择复制配置
3.添加虚拟机选项
4.更改名称,并添加-Dserver.port=6666覆盖其配置文件端口
5.启动!
6.查看注册
7.访问后多次刷新
8.能够看到两个服务均输出get request表明负载均衡生效
五、Eureka服务注册发现错误集锦以及解决方法
只能说黑马课程这一段简直是顶级折磨了,甚至由于他后面换成了nacos导致我几乎很难直接将他的源码还原为eureka版本的,其中遇到各种错误,硬控我三个小时,我甚至感觉微服务本身不难,难的是环境配置(插一句,我实习前三天部署公司项目一直报错,尝试手段包括更换mysql版本,重装java,idea换成2022版本,更换maven版本,甚至包括换了一台电脑重装系统从0开始部署,公司三个大佬折腾三天,最后终于发现是nacos里的yml配置文件有误,不过换了之后我能部署了,之前另外能运行的哥们又不行了,真的难蚌)
1.Failed to configure a DataSource:XXX
eureka服务无法启动,报如下错误。翻译就是datasource的url属性未指定,没有配置的数据源。
解决方法:在yml中配置数据源,因为user-service和order-service本来就需要连接数据库,所以只需要补充eureka服务的yml文件即可,这里是把order-service的数据源直接粘贴过去了。
2.java.lang.ExceptionInInitializerError/java.lang.reflect.InaccessibleObjectException
eureka服务无法启动,报如下异常信息。
java.lang.ExceptionInInitializerError: null
at com.thoughtworks.xstream.XStream.setupConverters(XStream.java:990) ~[xstream-1.4.11.1.jar:1.4.11.1]
at com.thoughtworks.xstream.XStream.<init>(XStream.java:593) ~[xstream-1.4.11.1.jar:1.4.11.1]
at com.thoughtworks.xstream.XStream.<init>(XStream.java:515) ~[xstream-1.4.11.1.jar:1.4.11.1]
at com.thoughtworks.xstream.XStream.<init>(XStream.java:484) ~[xstream-1.4.11.1.jar:1.4.11.1]
at com.thoughtworks.xstream.XStream.<init>(XStream.java:430) ~[xstream-1.4.11.1.jar:1.4.11.1]
at com.thoughtworks.xstream.XStream.<init>(XStream.java:397) ~[xstream-1.4.11.1.jar:1.4.11.1]
at com.netflix.discovery.converters.XmlXStream.<init>(XmlXStream.java:51) ~[eureka-client-1.9.25.jar:1.9.25]
at com.netflix.discovery.converters.XmlXStream.<clinit>(XmlXStream.java:42) ~[eureka-client-1.9.25.jar:1.9.25]
at com.netflix.discovery.converters.wrappers.CodecWrappers$XStreamXml.<init>(CodecWrappers.java:358) ~[eureka-client-1.9.25.jar:1.9.25]
at com.netflix.discovery.converters.wrappers.CodecWrappers.create(CodecWrappers.java:133) ~[eureka-client-1.9.25.jar:1.9.25]
at com.netflix.discovery.converters.wrappers.CodecWrappers.getEncoder(CodecWrappers.java:75) ~[eureka-client-1.9.25.jar:1.9.25]
at com.netflix.discovery.converters.wrappers.CodecWrappers.getEncoder(CodecWrappers.java:66) ~[eureka-client-1.9.25.jar:1.9.25]
at com.netflix.discovery.provider.DiscoveryJerseyProvider.<init>(DiscoveryJerseyProvider.java:77) ~[eureka-client-1.9.25.jar:1.9.25]
at com.netflix.discovery.provider.DiscoveryJerseyProvider.<init>(DiscoveryJerseyProvider.java:64) ~[eureka-client-1.9.25.jar:1.9.25]
at java.base/jdk.internal.reflect.DirectConstructorHandleAccessor.newInstance(DirectConstructorHandleAccessor.java:62) ~[na:na]
at java.base/java.lang.reflect.Constructor.newInstanceWithCaller(Constructor.java:501) ~[na:na]
at java.base/java.lang.reflect.ReflectAccess.newInstance(ReflectAccess.java:132) ~[na:na]
at java.base/jdk.internal.reflect.ReflectionFactory.newInstance(ReflectionFactory.java:259) ~[na:na]
at java.base/java.lang.Class.newInstance(Class.java:804) ~[na:na]
at com.sun.jersey.core.spi.component.ComponentConstructor._getInstance(ComponentConstructor.java:193) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.component.ComponentConstructor.getInstance(ComponentConstructor.java:180) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.component.ProviderFactory.__getComponentProvider(ProviderFactory.java:166) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.component.ProviderFactory._getComponentProvider(ProviderFactory.java:159) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.component.ProviderFactory.getComponentProvider(ProviderFactory.java:153) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.component.ProviderServices.getComponent(ProviderServices.java:278) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.component.ProviderServices.getProviders(ProviderServices.java:151) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.factory.MessageBodyFactory.initReaders(MessageBodyFactory.java:175) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.core.spi.factory.MessageBodyFactory.init(MessageBodyFactory.java:162) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.server.impl.application.WebApplicationImpl._initiate(WebApplicationImpl.java:1338) ~[jersey-server-1.19.1.jar:1.19.1]
at com.sun.jersey.server.impl.application.WebApplicationImpl.access$700(WebApplicationImpl.java:180) ~[jersey-server-1.19.1.jar:1.19.1]
at com.sun.jersey.server.impl.application.WebApplicationImpl$13.f(WebApplicationImpl.java:799) ~[jersey-server-1.19.1.jar:1.19.1]
at com.sun.jersey.server.impl.application.WebApplicationImpl$13.f(WebApplicationImpl.java:795) ~[jersey-server-1.19.1.jar:1.19.1]
at com.sun.jersey.spi.inject.Errors.processWithErrors(Errors.java:193) ~[jersey-core-1.19.1.jar:1.19.1]
at com.sun.jersey.server.impl.application.WebApplicationImpl.initiate(WebApplicationImpl.java:795) ~[jersey-server-1.19.1.jar:1.19.1]
at com.sun.jersey.server.impl.application.WebApplicationImpl.initiate(WebApplicationImpl.java:790) ~[jersey-server-1.19.1.jar:1.19.1]
at com.sun.jersey.spi.container.servlet.ServletContainer.initiate(ServletContainer.java:509) ~[jersey-servlet-1.19.1.jar:1.19.1]
at com.sun.jersey.spi.container.servlet.ServletContainer$InternalWebComponent.initiate(ServletContainer.java:339) ~[jersey-servlet-1.19.1.jar:1.19.1]
at com.sun.jersey.spi.container.servlet.WebComponent.load(WebComponent.java:605) ~[jersey-servlet-1.19.1.jar:1.19.1]
at com.sun.jersey.spi.container.servlet.WebComponent.init(WebComponent.java:207) ~[jersey-servlet-1.19.1.jar:1.19.1]
at com.sun.jersey.spi.container.servlet.ServletContainer.init(ServletContainer.java:394) ~[jersey-servlet-1.19.1.jar:1.19.1]
at com.sun.jersey.spi.container.servlet.ServletContainer.init(ServletContainer.java:744) ~[jersey-servlet-1.19.1.jar:1.19.1]
at org.apache.catalina.core.ApplicationFilterConfig.initFilter(ApplicationFilterConfig.java:270) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.ApplicationFilterConfig.<init>(ApplicationFilterConfig.java:106) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.StandardContext.filterStart(StandardContext.java:4566) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5203) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:183) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1384) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1374) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at java.base/java.util.concurrent.FutureTask.run(FutureTask.java:317) ~[na:na]
at org.apache.tomcat.util.threads.InlineExecutorService.execute(InlineExecutorService.java:75) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at java.base/java.util.concurrent.AbstractExecutorService.submit(AbstractExecutorService.java:148) ~[na:na]
at org.apache.catalina.core.ContainerBase.startInternal(ContainerBase.java:909) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.StandardHost.startInternal(StandardHost.java:843) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:183) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1384) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1374) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at java.base/java.util.concurrent.FutureTask.run(FutureTask.java:317) ~[na:na]
at org.apache.tomcat.util.threads.InlineExecutorService.execute(InlineExecutorService.java:75) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at java.base/java.util.concurrent.AbstractExecutorService.submit(AbstractExecutorService.java:148) ~[na:na]
at org.apache.catalina.core.ContainerBase.startInternal(ContainerBase.java:909) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.StandardEngine.startInternal(StandardEngine.java:262) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:183) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.StandardService.startInternal(StandardService.java:434) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:183) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.core.StandardServer.startInternal(StandardServer.java:930) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:183) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.apache.catalina.startup.Tomcat.start(Tomcat.java:486) ~[tomcat-embed-core-9.0.43.jar:9.0.43]
at org.springframework.boot.web.embedded.tomcat.TomcatWebServer.initialize(TomcatWebServer.java:123) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.embedded.tomcat.TomcatWebServer.<init>(TomcatWebServer.java:104) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory.getTomcatWebServer(TomcatServletWebServerFactory.java:440) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory.getWebServer(TomcatServletWebServerFactory.java:193) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.createWebServer(ServletWebServerApplicationContext.java:178) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.onRefresh(ServletWebServerApplicationContext.java:158) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:545) ~[spring-context-5.2.13.RELEASE.jar:5.2.13.RELEASE]
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.refresh(ServletWebServerApplicationContext.java:143) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:758) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:750) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.refreshContext(SpringApplication.java:405) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.run(SpringApplication.java:315) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1237) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1226) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at cn.bit.eurekaserver.EurekaApplication.main(EurekaApplication.java:11) ~[classes/:na]
Caused by: java.lang.reflect.InaccessibleObjectException: Unable to make field private final java.util.Comparator java.util.TreeMap.comparator accessible: module java.base does not "opens java.util" to unnamed module @2362f559
at java.base/java.lang.reflect.AccessibleObject.throwInaccessibleObjectException(AccessibleObject.java:388) ~[na:na]
at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:364) ~[na:na]
at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:312) ~[na:na]
at java.base/java.lang.reflect.Field.checkCanSetAccessible(Field.java:183) ~[na:na]
at java.base/java.lang.reflect.Field.setAccessible(Field.java:177) ~[na:na]
at com.thoughtworks.xstream.core.util.Fields.locate(Fields.java:40) ~[xstream-1.4.11.1.jar:1.4.11.1]
at com.thoughtworks.xstream.converters.collections.TreeMapConverter.<clinit>(TreeMapConverter.java:50) ~[xstream-1.4.11.1.jar:1.4.11.1]
... 82 common frames omitted
2025-01-19 14:06:01.923 ERROR 10176 --- [ main] o.apache.catalina.core.StandardContext : One or more Filters failed to start. Full details will be found in the appropriate container log file
2025-01-19 14:06:01.923 ERROR 10176 --- [ main] o.apache.catalina.core.StandardContext : Context [] startup failed due to previous errors
2025-01-19 14:06:01.958 INFO 10176 --- [ main] o.apache.catalina.core.StandardService : Stopping service [Tomcat]
2025-01-19 14:06:01.960 WARN 10176 --- [ main] ConfigServletWebServerApplicationContext : Exception encountered during context initialization - cancelling refresh attempt: org.springframework.context.ApplicationContextException: Unable to start web server; nested exception is org.springframework.boot.web.server.WebServerException: Unable to start embedded Tomcat
2025-01-19 14:06:01.964 INFO 10176 --- [ main] ConditionEvaluationReportLoggingListener :
Error starting ApplicationContext. To display the conditions report re-run your application with 'debug' enabled.
2025-01-19 14:06:01.966 ERROR 10176 --- [ main] o.s.boot.SpringApplication : Application run failed
org.springframework.context.ApplicationContextException: Unable to start web server; nested exception is org.springframework.boot.web.server.WebServerException: Unable to start embedded Tomcat
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.onRefresh(ServletWebServerApplicationContext.java:161) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:545) ~[spring-context-5.2.13.RELEASE.jar:5.2.13.RELEASE]
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.refresh(ServletWebServerApplicationContext.java:143) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:758) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:750) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.refreshContext(SpringApplication.java:405) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.run(SpringApplication.java:315) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1237) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1226) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at cn.bit.eurekaserver.EurekaApplication.main(EurekaApplication.java:11) ~[classes/:na]
Caused by: org.springframework.boot.web.server.WebServerException: Unable to start embedded Tomcat
at org.springframework.boot.web.embedded.tomcat.TomcatWebServer.initialize(TomcatWebServer.java:142) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.embedded.tomcat.TomcatWebServer.<init>(TomcatWebServer.java:104) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory.getTomcatWebServer(TomcatServletWebServerFactory.java:440) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory.getWebServer(TomcatServletWebServerFactory.java:193) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.createWebServer(ServletWebServerApplicationContext.java:178) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.onRefresh(ServletWebServerApplicationContext.java:158) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
... 9 common frames omitted
Caused by: java.lang.IllegalStateException: StandardEngine[Tomcat].StandardHost[localhost].TomcatEmbeddedContext[] failed to start
at org.springframework.boot.web.embedded.tomcat.TomcatWebServer.rethrowDeferredStartupExceptions(TomcatWebServer.java:187) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
at org.springframework.boot.web.embedded.tomcat.TomcatWebServer.initialize(TomcatWebServer.java:126) ~[spring-boot-2.3.9.RELEASE.jar:2.3.9.RELEASE]
... 14 common frames omitted
进程已结束,退出代码为 1
AI解释原因:(说人话java版本太高了)
(1)InaccessibleObjectException
:这个异常通常是由于 Java 模块系统的访问控制引起的。在 Java 9 及更高版本中,某些类和字段的访问权限变得更加严格。您看到的错误信息中提到 java.util.TreeMap.comparator
字段无法访问,这意味着代码试图反射访问一个不允许访问的字段。
(2)XStream 和 Eureka:您使用的 XStream 库(版本 1.4.11.1)可能不兼容当前的 Java 版本,特别是在 Java 9 及更高版本中。XStream 可能试图访问 Java 的内部实现细节,这在新的模块系统中被限制了。
(3)Spring Boot 和 Tomcat:由于 XStream 初始化失败,导致 Spring Boot 的嵌入式 Tomcat 无法启动。这是因为 Tomcat 依赖于 XStream 进行某些操作,而 XStream 的初始化失败导致整个应用程序的启动失败。
解决方法:
1.选择项目结构
2.全局sdk设置为1.8
3.每个模块的sdk也选1.8
4.最后别忘记点应用
5.启动成功
6.查看eureka页面
3.SpringBoot项目构建时显示找不到或无法加载主类
报错信息只有两行红字(忘记截图了),在添加完eureka的springboot项目后出现。
解决方法:
maven clean后install再点击运行
4.Cannot execute request on any known server
解决方法:
在eureka服务的yml文件中增加fetch-registry: false,别忘记空格。
5.Eureka的网页上显示服务已注册,但无法访问。
具体体现为order-service无法调用user-service的接口
但同时又能在Eureka网页上看到服务已经注册
这种报错分成两种情况
5.1I/O error on GET request for "xxx" UnknownHostException
解决方法:
RestTemplate的bean上添加负载均衡的注解(即使你只有一个实例)
5.2No instances available for userservice
(这个报错是最搞心态的,如果你没有使用下面解决方法你可能会遇见这样的情况,如果写死访问url"http://user-service/user/info/1",能偶尔正常调用,是的没错偶尔,但如果使用字符串拼接的方式比如"http://user-service/user/info/"+orderId(一个变量),那么对不起偶尔的机会也没有,当时遇到这个错误我人都要懵了,程序的确定性不存在了?)
解决方法:
orderservice的yml文件中设置 fetch-registry: true registryFetchIntervalSeconds: 5
原因如下:
没有 fetch-registry服务不会从eureka服务拿取服务列表,所以会找不到实例,但不知道为什么写死url反而又能拿取实例,同时设置拿取间隔为5秒,如果设置为30秒,这边服务列表你还没拿到了,它自然找不到实例,这也解释为啥有时候能够成功有时候不能成功(属性有个默认值,如果很急在默认值之前就测试就会报错)
(我又仔细确认了黑马的视频和源码确实没有这两句,我也不知道为啥他的就能调用)
6.Public Key Retrieval is not allowed
这个错在项目运行成功几天后再次运行时出现。
解决方法:从报错能够看出问题出现在数据库,点击右侧数据库,然后随便找个数据库右键属性
发现有个数据源报红,发现其驱动程序和另外两个不一样(不知道为啥它自己变了)
将驱动程序从MySQL for 5.1改成MySql,重新输入用户密码并测试连接,最好对另外两个数据库也重新输入一遍用户密码并测试连接
启动服务后访问ok,问题解决。虽然也不知道为啥浏览器展现json也变成这样了。
最后
关于eureka的部分就到这里,我在网上搜了一下eureka好像作为一个old school的东西存在一些问题,所以主流使用都是nacos,同时我目前实习所在公司用的也是nacos,所以这部分的实践就浅尝辄止,后面nacos的用法可能会有更多内容。(吐槽一句明明这个寒假明明有50天,但要实习就缩成10天了,emo)