原创 吴就业 83 0 2019-11-10
本文为博主原创文章,未经博主允许不得转载。
本文链接:https://www.wujiuye.com/article/48cbc4f48fa740ef98ef96b70280a79b
作者:吴就业
链接:https://www.wujiuye.com/article/48cbc4f48fa740ef98ef96b70280a79b
来源:吴就业的网络日记
本文为博主原创文章,未经博主允许不得转载。
本篇文章写于2019年11月10日,从公众号|掘金|CSDN手工同步过来(博客搬家),本篇为原创文章。
dubbo通过BeanFactoryPostProcessor与BeanPostProcessor分别完成ServiceBean的注册与被@Reference注释的属性的依赖注入,通过BeanPostProcessor完成配置文件与相关配置类bean的属性绑定。
@Service(dubbo的)注释的bean会交由spring创建管理,同时注册一个持有该bean的引用(beanName)ServiceBean。@Service配置的属性最终会绑定到ServiceBean的属性上,ServiceBean通过监听spring事件完成服务的导出(暴露)。
@Reference声明的属性,最终注入的是接口的代理,由ReferenceBean创建,ReferenceBean是一个FactoryBean,在getObject方法中调用get方法,即ReferenceBean的get方法完成服务的引入。
在消费端调用接口的方法,会走两层代理,第一层是jdk动态代理,被代理对象是ReferenceBean,走到invoke方法时,最终调用ReferenceBean的get返回的代理类的方法,代理类完成服务的远程调用。
还记得上篇说到的SPI吗,自适应扩展。在dubbo中,URL是将所有层次衔接起来的桥梁。所有配置最终都是绑定到URL上,通过URL传递从而发挥其作用。URL相当于数据总线,但缺点也很明显,配置通过url传递,每次rpc调用携带的参数都非常多,导致数据包增大。
dubbo版本:apache-dubbo-2.7.2
demo:使用源码提供的demo
实际开发中我是借助dubbo-spring-boot-starter完成dubbo的自动配置,starter做的事情很简单,真正提供与spring整合功能的在dubbo源码的dubbo-config模块:dubbo-config-spring。本篇重点是分析dubbo-config-spring的源码,看看dubbo是如何实现服务暴露与服务发现的。
我学习源码的方法就是带着问题看源码,这其实也是我们看源码的目的之一,不就是想通过看源码解答我们心中的疑问吗?那么,问题来了,我为什么要去学习 dubbo是如何与spring整合的呢,为什么只看注解方式的整合呢?肯定是因为我使用了注解方式,而我想配置连接数,但在yml中可以配置,在注解上可以配置,那么到底哪个是起作用的。
首先,我们看下官方提供的demo,了解如果通过注解方式使用dubbo,这也是我们分析源码的入口。demo源码在dubbo-demo模块,使用注解配置方式的demo是dubbo-demo-annotation,dubbo-demo-annotation下有两个例子,一个是服务提供者dubbo-demo-annotation-provider,一个是服务消费者dubbo-demo-annotation-consumer。
无论是消费端还是服务提供端,想要使用dubbo就必须要添加jar包依赖。上篇提到,Dubbo总体分为业务层、RPC层、Remote层,而RPC层又可细分为代理层、注册中心层、集群负载层、监视器层、协议层,Remote层又可细分为信息交换层、传输层、序列化层。根据dubbo的分层我们很容易理解demo中的pom依赖配置文件。
抽象接口,真正项目开发中,需要每个业务模块的负责人约定协议,这与前后端分离约定接口上行请求与下行返回是一样的。假设服务A是提供者,服务B是消费者,那么服务A需要知道服务B想要返回什么数据,而服务B需要告诉服务A,想要获取数据需要传递哪些参数,这对应到interface的方法定义上。服务B和服务A约定好接口后,通过依赖同一个接口的jar包,双方就可以同步开发。
public interface DemoService {
String sayHello(String name);
}
服务提供方实现接口
@Service
public class DemoServiceImpl implements DemoService {
private static final Logger logger = LoggerFactory.getLogger(DemoServiceImpl.class);
@Override
public String sayHello(String name) {
logger.info("Hello " + name + ", request from consumer: " + RpcContext.getContext().getRemoteAddress());
return "Hello " + name + ", response from provider: " + RpcContext.getContext().getLocalAddress();
}
}
服务消费方调用接口
@Component("demoServiceComponent")
public class DemoServiceComponent implements DemoService {
@Reference
private DemoService demoService;
@Override
public String sayHello(String name) {
return demoService.sayHello(name);
}
}
就像发送http请求一样,消费端只要知道服务提供方的ip地址和端口号,就可以发起远程rpc调用,但如果服务提供方服务器迁移或者增加节点,又或者某个节点故障维修,那会导致消费端不可用,所以需要一个注册中心来统一管理。消费端从注册中心拿到正常提供服务的所有服务提供者,服务提供者向注册中心注册,并保持心跳,当某个提供者不可用时,将提供者剔除,并通知所有消费者,或者消费者定时询问注册中心。Monitor暂时不讨论。
服务提供方向注册中心注册,暴露服务
@Configuration
@EnableDubbo(scanBasePackages = "org.apache.dubbo.demo.provider")
@PropertySource("classpath:/spring/dubbo-provider.properties")
static class ProviderConfiguration {
@Bean
public RegistryConfig registryConfig() {
RegistryConfig registryConfig = new RegistryConfig();
registryConfig.setAddress("multicast://224.5.6.7:1234");
return registryConfig;
}
}
服务消费方通过注册中心发现服务
@Configuration
@EnableDubbo(scanBasePackages = "org.apache.dubbo.demo.consumer.comp")
@PropertySource("classpath:/spring/dubbo-consumer.properties")
@ComponentScan(value = {"org.apache.dubbo.demo.consumer.comp"})
static class ConsumerConfiguration {
}
从官方提供的demo中可以看出,无论是消费端还是提供端,都是通过@EnableDubbo注解实现自动配置的。所以@EnableDubbo就是我们分析dubbo与spring整合源码的入口。
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
@EnableDubboConfig
@DubboComponentScan
public @interface EnableDubbo {
@AliasFor(annotation = DubboComponentScan.class, attribute = "basePackages")
String[] scanBasePackages() default {};
@AliasFor(annotation = DubboComponentScan.class, attribute = "basePackageClasses")
Class<?>[] scanBasePackageClasses() default {};
@AliasFor(annotation = EnableDubboConfig.class, attribute = "multiple")
boolean multipleConfig() default true;
}
@EnableDubbo注解的属性有:scanBasePackages:配置需要扫描的包,这与spring配置的包扫描不同,dubbo的包扫描是扫描被org.apache.dubbo.config.annotation.Service注解的类,由dubbo向spring容器中注册bean。
scanBasePackageClasses:配置扫描的类,与scanBasePackages作用相同。前者是指定扫描的包,而后面则可以直接指定类,最后拿到这些类的包名,在我看来是多余的。
@EnableDubbo注解上有@EnableDubboConfig与@DubboComponentScan注解。scanBasePackages与scanBasePackageClasses属性上都有@AliasFor注解,作用是指定将这个属性的值赋给@DubboComponentScan注解。那么很明显,@DubboComponentScan注解就是完成包扫描的入口。所以接下来我们继续分析@DubboComponentScan注解。
在分析@DubboComponentScan之前,你是否会有疑惑。在demo中,服务消费端和服务提供端都配置了@EnableDubbo(scanBasePackages =“xxxx”)。但实际上消费端配置的scanBasePackages是多余的,并不会用到。那dubbo是怎么知道哪些bean中有被org.apache.dubbo.config.annotation.Reference注解的属性,以及在spring初始化bean阶段需要完成属性的注入时,dubbo怎么告诉spring这个属性的值由它提供?带着问题我们继续看源码。
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import(DubboComponentScanRegistrar.class)
public @interface DubboComponentScan {
String[] value() default {};
String[] basePackages() default {};
Class<?>[] basePackageClasses() default {};
}
分析完@EnableDubbo注解之后,我们知道,@DubboComponentScan的属性basePackages的值是从@EnableDubbo的scanBasePackages属性传递过来的。@DubboComponentScan注解上有一个@Import注解,如果对 spring还不是很了解的,可能需要先了解下@Import注解。@Import注解导入了一个配置类DubboComponentScanRegistrar.class。接着我们看DubboComponentScanRegistrar的registerBeanDefinitions方法。
Set<String> packagesToScan = getPackagesToScan(importingClassMetadata);
registerServiceAnnotationBeanPostProcessor(packagesToScan, registry);
registerReferenceAnnotationBeanPostProcessor(registry);
getPackagesToScan方法:从@DubboComponentScan注解中拿到需要扫描的包。
registerServiceAnnotationBeanPostProcessor:向spring的BeanDefinitionRegistry注册dubbo的Service注解bean工厂前置处理器。
BeanDefinitionBuilder builder = rootBeanDefinition(ServiceAnnotationBeanPostProcessor.class);
builder.addConstructorArgValue(packagesToScan);
builder.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
AbstractBeanDefinition beanDefinition = builder.getBeanDefinition();
BeanDefinitionReaderUtils.registerWithGeneratedName(beanDefinition, registry);
该方法是将ServiceAnnotationBeanPostProcessor解析成一个BeanDefinition,注册到BeanDefinitionRegistry,在spring中,bean是通过BeanDefinition创建的,可以翻下我的往期分析spring源码的文章。
而ServiceAnnotationBeanPostProcessor实现BeanDefinitionRegistryPostProcessor接口,该接口继承BeanFactoryPostProcessor接口,所以ServiceAnnotationBeanPostProcessor实际上是一个bean工厂的前置处理器。对spring源码有了解的,看到这里就明白ServiceAnnotationBeanPostProcessor的作用了。BeanFactoryPostProcessor是实现spring容器功能扩展的重要接口,例如修改bean属性值,实现bean动态代理等。
BeanDefinitionRegistryPostProcessor是BeanFactoryPostProcessor的子类,postProcessBeanDefinitionRegistry会在postProcessBeanFactory方法之前被调用。感兴趣可以看下org.springframework.context.support.PostProcessorRegistrationDelegate的invokeBeanFactoryPostProcessors源码。到此,我不想继续展开细说ServiceAnnotationBeanPostProcessor,我来总结下ServiceAnnotationBeanPostProcessor做的事情。
扯远了。ServiceAnnotationBeanPostProcessor做的事情就是通过扫描指定的包,获取所有被org.apache.dubbo.config.annotation.Service注解注释的类,生成bean并注册到spring容器,让spring管理。但这里就有意思了,实际上会生成一个 Service的代理对象ServiceBean。
private void registerServiceBeans(Set<String> packagesToScan, BeanDefinitionRegistry registry) {
DubboClassPathBeanDefinitionScanner scanner =
new DubboClassPathBeanDefinitionScanner(registry, environment, resourceLoader);
BeanNameGenerator beanNameGenerator = resolveBeanNameGenerator(registry);
scanner.setBeanNameGenerator(beanNameGenerator);
for (String packageToScan : packagesToScan) {
// Registers @Service Bean first
scanner.scan(packageToScan);
// Finds all BeanDefinitionHolders of @Service whether @ComponentScan scans or not.
Set<BeanDefinitionHolder> beanDefinitionHolders =
findServiceBeanDefinitionHolders(scanner, packageToScan, registry, beanNameGenerator);
if (!CollectionUtils.isEmpty(beanDefinitionHolders)) {
for (BeanDefinitionHolder beanDefinitionHolder : beanDefinitionHolders) {
registerServiceBean(beanDefinitionHolder, registry, scanner);
}
}
}
}
在registerServiceBeans方法中先调用org.springframework.context.annotation.ClassPathBeanDefinitionScanner的scan方法,先将被org.apache.dubbo.config.annotation.Service注解注释的类交给spring处理,而dubbo要做的是给这些bean封装一层代理,也就是ServiceBean,这些ServiceBean持有Service的引用,即Service的beanName。所以也就为什么被org.apache.dubbo.config.annotation.Service注解的bean跟普通的bean没什么区别的原因。
关于ServiceBean我们一会再分析,先分析完DubboComponentScanRegistrar的registerBeanDefinitions方法。
接着我们看registerReferenceAnnotationBeanPostProcessor方法,从方法名中可以看出,这里处理服务消费端的@Reference引用的。
registerReferenceAnnotationBeanPostProcessor(registry);
实际上也并没有使用到@EnableDubbo配置的扫描包,所以消费端并不需要包扫描,那么dubbo怎么知道哪些bean中有被@Reference引用的属性呢。该方法是向spring中注册一个ReferenceAnnotationBeanPostProcessor。
BeanPostProcessor是spring的bean前置处理器,在Spring容器的创建bean过程createBean中会回调BeanPostProcessor中定义的两个方法。
public interface BeanPostProcessor {
Object postProcessBeforeInitialization(Object var1, String var2) throws BeansException;
Object postProcessAfterInitialization(Object var1, String var2) throws BeansException;
}
而InstantiationAwareBeanPostProcessor接口继承BeanPostProcessor接口。InstantiationAwareBeanPostProcessor接口的主要作用在于目标对象的实例化过程中需要处理的事情,包括实例化对象的前后过程以及实例的属性设置。
public interface InstantiationAwareBeanPostProcessor extends BeanPostProcessor {
Object postProcessBeforeInstantiation(Class<?> var1, String var2) throws BeansException;
boolean postProcessAfterInstantiation(Object var1, String var2) throws BeansException;
PropertyValues postProcessPropertyValues(PropertyValues var1, PropertyDescriptor[] var2, Object var3, String var4) throws BeansException;
}
这属于spring源码的范围,这里不做深入分析,感兴趣可以根据我给的思路去找源码看。
这是spring4.3.16的源码,dubbo2.7.2依赖的版本
.......
org.springframework.beans.factory.support.DefaultListableBeanFactory#resolveNamedBean(java.lang.Class<T>)
org.springframework.beans.factory.support.DefaultListableBeanFactory#resolveNamedBean(java.lang.Class<T>, java.lang.Object...)
org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#createBean(java.lang.String, org.springframework.beans.factory.support.RootBeanDefinition, java.lang.Object[])
org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#populateBean
这几个方法的执行顺序是
我总感觉我被带沟里去了。我们重点看ReferenceAnnotationBeanPostProcessor的doGetInjectedBean方法,该方法就是spring创建bean阶段,给bean属性赋值时,调用postProcessPropertyValues方法,最后绕啊绕,如果属性被@Reference注解注释,则会调用ReferenceAnnotationBeanPostProcessor的doGetInjectedBean方法。根据属性类型,生成一个代理bean,ReferenceBean?。
我们继续分析完@DubboComponentScan注解,再分析ReferenceBean和ServiceBean。那么@DubboComponentScan已经分析,还剩下一个@EnableDubboConfig。
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
@Import(DubboConfigConfigurationRegistrar.class)
public @interface EnableDubboConfig {
boolean multiple() default true;
}
@EnableDubboConfig有一个属性multiple,配置是否使用多注册中心,这里我们忽略。接着看DubboConfigConfigurationRegistrar。DubboConfigConfigurationRegistrar的registerBeanDefinitions方法主要是从@EnableDubboConfig注解获取注解上的multiple属性的值。我们忽略多注册中心相关的逻辑。
registerBeans(registry, DubboConfigConfiguration.Single.class);
向BeanDefinitionRegistry中注册一个BeanDefinition,其实就是注册bean。所以重点看下DubboConfigConfiguration.Single.class是什么。
@EnableDubboConfigBindings({
@EnableDubboConfigBinding(prefix = "dubbo.application", type = ApplicationConfig.class),
@EnableDubboConfigBinding(prefix = "dubbo.module", type = ModuleConfig.class),
@EnableDubboConfigBinding(prefix = "dubbo.registry", type = RegistryConfig.class),
@EnableDubboConfigBinding(prefix = "dubbo.protocol", type = ProtocolConfig.class),
@EnableDubboConfigBinding(prefix = "dubbo.monitor", type = MonitorConfig.class),
@EnableDubboConfigBinding(prefix = "dubbo.provider", type = ProviderConfig.class),
@EnableDubboConfigBinding(prefix = "dubbo.consumer", type = ConsumerConfig.class),
@EnableDubboConfigBinding(prefix = "dubbo.config-center", type = ConfigCenterBean.class),
@EnableDubboConfigBinding(prefix = "dubbo.metadata-report", type = MetadataReportConfig.class)
})
public static class Single {
}
就是将@PropertySource(“classpath:/spring/dubbo-provider.properties”)的配置信息自动与ApplicationConfig、ModuleConfig、RegistryConfig等类型的bean的属性绑定,且会自动注册bean。也就是说,我们可以也可以通过ApplicationContext.getBean(xxxConfig.class)获取到相应的配置。
实现方式还是通过BeanPostProcessor在bean初始化阶段给bean的属性赋值,具体实现可看DubboConfigBindingBeanPostProcessor。
@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
if (beanName.equals(this.beanName) && bean instanceof AbstractConfig) {
AbstractConfig dubboConfig = (AbstractConfig) bean;
bind(prefix, dubboConfig);
customize(beanName, dubboConfig);
}
return bean;
}
所有继承AbstractConfig的配置类,都可通过@EnableDubboConfigBinding完成自动绑定配置。关于@EnableDubboConfig就说这么多。
ServiceBean是对被@Service注解注释的bean的代理,持有对被@Service注解注释的bean的引用beanName。
@Service配置的属性,是在ServiceBean中才使用,ServiceBean完成服务的暴露工作,将服务注册到注册中心。我们从ServiceBean的创建说起。
回到ServiceAnnotationBeanPostProcessor,看buildServiceBeanDefinition方法。buildServiceBeanDefinition方法生成一个ServiceBean的BeanDefinition,将@Service配置的属性赋给ServiceBean的属性。
BeanDefinitionBuilder builder = rootBeanDefinition(ServiceBean.class);
AbstractBeanDefinition beanDefinition = builder.getBeanDefinition();
MutablePropertyValues propertyValues = beanDefinition.getPropertyValues();
String[] ignoreAttributeNames = of("provider", "monitor", "application", "module", "registry", "protocol",
"interface", "interfaceName", "parameters");
propertyValues.addPropertyValues(new AnnotationPropertyValuesAdapter(service, environment, ignoreAttributeNames));
// References "ref" property to annotated-@Service Bean
addPropertyReference(builder, "ref", annotatedServiceBeanName);
// Set interface
builder.addPropertyValue("interface", interfaceClass.getName());
// Convert parameters into map
builder.addPropertyValue("parameters", convertParameters(service.parameters()));
// ProviderConfig
// MonitorConfig
// ApplicationConfig
// ModuleConfig
// RegistryConfig
// ProtocolConfig
return builder.getBeanDefinition();
每个BeanDefinition都有一个MutablePropertyValues,在创建bean的时候,会把MutablePropertyValues描述的属性对应的value为bean注入属性值。在这里我还是想啰嗦一句,因为我发现到处都有BeanDefinition。BeanDefinition是对bean的描述,描述怎么创建一个bean,比如说我们在淘宝上定制衣服,在下单时描述我要的是一件短袖,白色的、升高172cm的男孩子穿的、在衣服上印上“xxx”字样,这样卖家就会根据你的描述给你生产一件衣服。ok,这里的MutablePropertyValues就相当于记录:
颜色:白色 花文:印“xxx”字 等
new AnnotationPropertyValuesAdapter(service, environment, ignoreAttributeNames)
排除ignoreAttributeNames指定的属性,将service注解所有的属性转为PropertyValues。
其它的*ProviderConfig、MonitorConfig*等前面分析过了,由于这些属性是一个个bean,只能通过RuntimeBeanReference持有一个引用,因为当前引用的bean未被创建,只有在创建ServiceBean时,如果PropertyValue的value是一个RuntimeBeanReference,这是才会去找到bean赋值给ServiceBean实例的属性,如果被引用的bean未创建,则会去创建被引用的bean。
回到ServiceAnnotationBeanPostProcessor的registerServiceBean方法,通过buildServiceBeanDefinition方法拿到的BeanDefinition,最后是通过registerServiceBean注册到spring的。一个应用可能会存在多个ServiceBean,那如何确保为每个@Service的bean生成的代理ServiceBean是不同的bean呢,就是通过beanName区分的。
beanName由interfaceClass+group+version组成。当接收到远程调用时,就可以根据请求uri携带的interfaceClass、gourp、version参数拿到目标代理ServiceBean实例。
解答疑惑,timeout在AbstractConfig,由@Service注解配置得来,如果没有配置,则为null,dubbo会使用默认值1000ms。connections在AbstractInterfaceConfig,也是由@Service注解配置得来。所以我们看@Service注解配置的timeout、connections等起不起作用,就看这写属性有没有被用到。
那ServiceBean是何时调用export方法暴露服务的呢?一个是在接收到事件时onApplicationEvent方法调用export导出服务,或是在ServiceBean初始化阶段InitializingBean的afterPropertiesSet方法被调用。
afterPropertiesSet方法
if (!supportedApplicationListener) {
export();
}
所以afterPropertiesSet方法中只有在不支持ApplicationListener的情况下,才会在这个时机导出服务。
public void export() {
super.export();
// Publish ServiceBeanExportedEvent
publishExportEvent();
}
export方法调用父类的方法完成导出,同时会发布一个事件,这个事件是干嘛用的呢,解决local(本应用内)的消费者,同一个进程内的对象通过@Reference获取服务提供者的问题。这里我们不用去关心,同一个应用内还用@Reference干嘛。
我们继续看父类org.apache.dubbo.config.ServiceConfig的export方法。
if (shouldDelay()) {
delayExportExecutor.schedule(this::doExport, getDelay(), TimeUnit.MILLISECONDS);
} else {
doExport();
}
判断是否需要延迟暴露服务,如果延迟,则提交一个定时任务,延时指定时间后再调用doExport方法,否则直接调用doExport方法。
if (exported) {
return;
}
exported = true;
if (StringUtils.isEmpty(path)) {
path = interfaceName;
}
doExportUrls();
关于doExport就分析到这,继续分析就超出本篇的范围了。
我们回到前面分析的ReferenceAnnotationBeanPostProcessor的doGetInjectedBean方法说起。doGetInjectedBean为被@Reference注释的bean的属性赋值一个对象。但是该对象也是交给spring管理的。
这个对象是ReferenceBean吗?ReferenceBean是一个FactoryBean,所以我们最关心的还是它的getObject方法返回的是什么对象。我们先分析下doGetInjectedBean方法。
@Override
protected Object doGetInjectedBean(Reference reference, Object bean, String beanName, Class<?> injectedType,
InjectionMetadata.InjectedElement injectedElement) throws Exception {
String referencedBeanName = buildReferencedBeanName(reference, injectedType);
ReferenceBean referenceBean = buildReferenceBeanIfAbsent(referencedBeanName, reference, injectedType, getClassLoader());
cacheInjectedReferenceBean(referenceBean, injectedElement);
return buildProxy(referencedBeanName, referenceBean, injectedType);
}
referencedBeanName同ServiceBean一样,通过引用的接口类型名、group、version生成的一个字符串,所以一个应用中,有多个类对同一个接口使用@Reference注解,其依赖的都是同一个对象。那么问题来了,对于同一个接口,每个@Reference配置的timeout等属性都不同,最后每个的配置都起作用吗?还是只有其中一个?
我们继续分析doGetInjectedBean方法,接着看buildReferenceBeanIfAbsent方法。
ReferenceAnnotationBeanPostProcessor#buildReferenceBeanIfAbsent
private ReferenceBean buildReferenceBeanIfAbsent(
String referencedBeanName, Reference reference,
Class<?> referencedType, ClassLoader classLoader)throws Exception {
ReferenceBean<?> referenceBean = referenceBeanCache.get(referencedBeanName);
if (referenceBean == null) {
ReferenceBeanBuilder beanBuilder = ReferenceBeanBuilder
.create(reference, classLoader, applicationContext)
.interfaceClass(referencedType);
referenceBean = beanBuilder.build();
referenceBeanCache.put(referencedBeanName, referenceBean);
}
return referenceBean;
}
所以,同一个接口,同一个group、version,最后用的都是同一个ReferenceBean,准确的说,一个应用中,有多个类对同一个接口使用@Reference注解,其依赖的都是同一个对象ReferenceBean的getObject返回的对象。
接着看AbstractAnnotationConfigBeanBuilder#build
public final B build() throws Exception {
checkDependencies();
B bean = doBuild();
configureBean(bean);
return bean;
}
AbstractAnnotationConfigBeanBuilder#configureBean
protected void configureBean(B bean) throws Exception {
preConfigureBean(annotation, bean);
configureRegistryConfigs(bean);
configureMonitorConfig(bean);
configureApplicationConfig(bean);
configureModuleConfig(bean);
postConfigureBean(annotation, bean);
}
@Reference注解配置的属性,在preConfigureBean中为ReferenceBean属性赋值。详细过程就不分析了。
回到doGetInjectedBean方法的最后一行,buildProxy。
private Object buildProxy(String referencedBeanName, ReferenceBean referenceBean, Class<?> injectedType) {
InvocationHandler handler = buildInvocationHandler(referencedBeanName, referenceBean);
return Proxy.newProxyInstance(getClassLoader(), new Class[]{injectedType}, handler);
}
使用jdk动态代理生成一个代理对象,所以最终为@Reference注释的bean的属性赋值的不是ReferenceBean的getObject返回的对象,而是ReferenceBean的getObject返回的对象的代理。
看下ReferenceBeanInvocationHandler。
private static class ReferenceBeanInvocationHandler implements InvocationHandler {
private final ReferenceBean referenceBean;
private Object bean;
private ReferenceBeanInvocationHandler(ReferenceBean referenceBean) {
this.referenceBean = referenceBean;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result;
try {
if (bean == null) { // If the bean is not initialized, invoke init()
// issue: https://github.com/apache/dubbo/issues/3429
init();
}
result = method.invoke(bean, args);
} catch (InvocationTargetException e) {
// re-throws the actual Exception.
throw e.getTargetException();
}
return result;
}
private void init() {
this.bean = referenceBean.get();
}
}
ReferenceBeanInvocationHandler持有ReferenceBean,注意看init方法,前面提到,ReferenceBean是一个FactoryBean,我们要关系它的getObject方法。而getObject方法就是返回一个接口的动态实现类,封装了远程调用逻辑。所以,当我们在消费端调用一个接口的方法时,ReferenceBeanInvocationHandler的invoke方法就是入口。
那ReferenceBean的getObject方法返回的对象是什么呢?
@Override
public Object getObject() {
return get();
}
org.apache.dubbo.config.ReferenceConfig#get
public synchronized T get() {
checkAndUpdateSubConfigs();
if (destroyed) {
throw new IllegalStateException("The invoker of ReferenceConfig(" + url + ") has already destroyed!");
}
if (ref == null) {
init();
}
return ref;
}
org.apache.dubbo.config.ReferenceConfig#init
.....
Map<String, String> map = new HashMap<String, String>();
.....
appendParameters(map, this);
.....
ref = createProxy(map);
.....
appendParameters(map, this);方法就是将ReferenceBean的所有属性写入map中,最后调用createProxy传入map创建一个org.apache.dubbo.rpc.Invoker。Invoker的创建依赖URL,在createProxy方法中,通过map创建一个URL,所以URL中保存了map中的所有信息。
继续往下,就是服务引入阶段,本篇不再继续分析。
声明:公众号、CSDN、掘金的曾用名:“Java艺术”,因此您可能看到一些早期的文章的图片有“Java艺术”的水印。
并发数上升,到底是哪个服务处理能力到了瓶颈,还是Redis性能到了瓶颈,只有找出是哪里的性能问题,才能对症下药。所以,了解redis的一些运维知识能够帮助我们快速判定是否Redis集群的性能问题。
Dubbo默认使用随机负载均衡策略,据笔者了解,目前Dubbo一共提供了四种可选的负载均衡策略,有关于负载均衡策略的实现,如果不怕阅读源码枯燥的,笔者推荐阅读官网的源码导读部份的文档。
dubbo随机负载均衡的权重很少会用到吗?之前我想给随机负载均衡策略配置权重,各种搜索都找不到答案,包括翻阅官方文档。而且我们项目中用的还是最新的Nacos注册中心,非常无奈,最后只能在源码中寻找答案。
本篇介绍使用ES与使用Redis实现的IP库范围查询谁性能更强,以及远程http调用ip服务耗时如何降低到0ms。
IP信息库是按区间存储的,拿到一个ip得要知道它所属的范围才能知道它对应哪条记录。本篇介绍如何使用Redis的Sorted Set数据结构实现支持范围查找的IP库缓存方案。
订阅
订阅新文章发布通知吧,不错过精彩内容!
输入邮箱,提交后我们会给您发送一封邮件,您需点击邮件中的链接完成订阅设置。