博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
dubbo service export过程
阅读量:6684 次
发布时间:2019-06-25

本文共 7956 字,大约阅读时间需要 26 分钟。

hot3.png

1. AnnotationBean注册

AnnotationBean 的注册通过xml配置. Spring通过读取解析xml,而获得AnnotationBean. AnnotationBean实例化了BeanFactoryPostProcessor,所以在BeanFactory初始化后,会调用 AnnotationBean. postProcessBeanFactory. AnnotationBean通过反射,调用ClassPathBeanDefinitionScanner进行Bean扫描.并通过 IncludeFilter机制,让ClassPathBeanDefinitionScanner加载包含Service注解的Bean.

2. Annotation全局扫描

AnnotationBean 将扫描委托给ClassPathBeanDefinitionScanner实现.而AnnotationBean关心的Service注解,也是通过 IncludeFilter添加到ClassPathBeanDefinitionScanner.这里所介绍的内容更加接近于Spring的包扫描机 制.

ClassPathBeanDefinitionScanner.scan(String...basePackages)是包扫描的入口,而真正的实现则在doScan(String... basePackages)中.源码:

protected Set
 doScan(String... basePackages) { Set
 beanDefinitions = new LinkedHashSet
(); for (int i = 0; i < basePackages.length; i++) {//扫描包下所有的类,过滤出符合条件的类,生成BeanDefinition. Set
 candidates = findCandidateComponents(basePackages[i]); for (BeanDefinition candidate : candidates) { String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry); if (candidate instanceof AbstractBeanDefinition) { postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName); } ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate); if (checkCandidate(beanName, candidate)) {//使用BeanDefinitionHolder来装饰bean. BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName); definitionHolder = applyScope(definitionHolder, scopeMetadata); beanDefinitions.add(definitionHolder);//将bean注册到beanFactory中 registerBeanDefinition(definitionHolder, this.registry); } } } return beanDefinitions; }

 

3. Reference 自动注入

AnnotationBean实现了BeanPostProcessor,所以在每个bean 的初始化之前和之后,分别会调用AnnotationBean的postProcessBeforeInitialization和 postProcessAfterInitialization方法。

public Object postProcessBeforeInitialization(Object bean, String beanName)            throws BeansException {//判断此bean是否在包扫描路径下。如果不在路径下,则不做任何处理        if (! isMatchPackage(bean)) {            return bean;        }//分析bean所有的set方法,是否有Reference注解。如果有,则注入。        Method[] methods = bean.getClass().getMethods();        for (Method method : methods) {            String name = method.getName();            if (name.length() > 3 && name.startsWith("set")                    && method.getParameterTypes().length == 1                    && Modifier.isPublic(method.getModifiers())                    && ! Modifier.isStatic(method.getModifiers())) {                try {                	Reference reference = method.getAnnotation(Reference.class);                	if (reference != null) {	                	Object value = refer(reference, method.getParameterTypes()[0]);	                	if (value != null) {	                		method.invoke(bean, new Object[] {  });	                	}                	}                } catch (Throwable e) {                 }            }        }//分析bean所有的成员,如果有Reference注解,则注入。        Field[] fields = bean.getClass().getDeclaredFields();        for (Field field : fields) {            try {                if (! field.isAccessible()) {                    field.setAccessible(true);                }                Reference reference = field.getAnnotation(Reference.class);            	if (reference != null) {	                Object value = refer(reference, field.getType());	                if (value != null) {	                	field.set(bean, value);	                }            	}            } catch (Throwable e) {             }        }        return bean;}

 

 

通过以上的逻辑可以知道,Dubbo支持成员和set方法的注解。

public Object postProcessAfterInitialization(Object bean, String beanName)            throws BeansException {//判断所在包是否被扫描        if (! isMatchPackage(bean)) {            return bean;        }//如果Bean有Service注解,则实例化ServiceBean,承载此注解及注解描述的Bean。Service service = bean.getClass().getAnnotation(Service.class);        if (service != null) {            ServiceBean serviceConfig = new ServiceBean(service);            if (void.class.equals(service.interfaceClass())                    && "".equals(service.interfaceName())) {                if (bean.getClass().getInterfaces().length > 0) {                    serviceConfig.setInterface(bean.getClass().getInterfaces()[0]);                } else {                    throw new IllegalStateException("Failed to export remote service class " + bean.getClass().getName() + ", cause: The @Service undefined interfaceClass or interfaceName, and the service class unimplemented any interfaces.");                }            }            if (applicationContext != null) {                serviceConfig.setApplicationContext(applicationContext);             .................                if (service.protocol() != null && service.protocol().length > 0) {                    List
 protocolConfigs = new ArrayList
();                    for (String protocolId : service.registry()) {                        if (protocolId != null && protocolId.length() > 0) {                            protocolConfigs.add((ProtocolConfig)applicationContext.getBean(protocolId, ProtocolConfig.class));                        }                    }                    serviceConfig.setProtocols(protocolConfigs);                }                try {                    serviceConfig.afterPropertiesSet();                } catch (RuntimeException e) {                }            }            serviceConfig.setRef(bean);            serviceConfigs.add(serviceConfig);//export此bean。            serviceConfig.export();        }        return bean;}

 

 

由源码可以看出,servicebean是通过AnnotationBean实现export调用的。

4. ServiceBean export

ServiceBean 可以通过xml配置,也可以通过注解配置.所以,export方法的调用点也是两处:serviceBean的afterPropertiesSet接口 函数和AnnotationBean的postProcessAfterInitialization.这里介绍注解方式,所以从 AnnotationBean的切入点入手.在AnnotationBean中,根据ServiceBean(spring内注册bean)的信息,创建 一个新的ServiceConfigBean(dubbo内bean),并填充 provider,application,monitor, module, provider, protocol,ref等属性. 之后调用ServiceConfigBean.export().

Export的核心功能在doExportUrls中实现。核心代码:

 //配置为none不暴露        if (! Constants.SCOPE_NONE.toString().equalsIgnoreCase(scope)) {            //配置不是remote的情况下做本地暴露 (配置为remote,则表示只暴露远程服务)            if (!Constants.SCOPE_REMOTE.toString().equalsIgnoreCase(scope)) {                exportLocal(url);            }            //如果配置不是local则暴露为远程服务.(配置为local,则表示只暴露远程服务)            if (! Constants.SCOPE_LOCAL.toString().equalsIgnoreCase(scope) ){                if (registryURLs != null && registryURLs.size() > 0                        && url.getParameter("register", true)) {                    for (URL registryURL : registryURLs) {                        url = url.addParameterIfAbsent("dynamic", registryURL.getParameter("dynamic"));                        URL monitorUrl = loadMonitor(registryURL);                        if (monitorUrl != null) {                            url = url.addParameterAndEncoded(Constants.MONITOR_KEY, monitorUrl.toFullString());                        }                        Invoker
 invoker = proxyFactory.getInvoker(ref, (Class
) interfaceClass, registryURL.addParameterAndEncoded(Constants.EXPORT_KEY, url.toFullString()));                        Exporter
 exporter = protocol.export(invoker);                        exporters.add(exporter);                    }                } else {     Invoker
 invoker = proxyFactory.getInvoker(ref, (Class
) interfaceClass, url);                    Exporter
 exporter = protocol.export(invoker);                    exporters.add(exporter);                }            }        }

 

 

protocol.export(invoker)实现了service的注册中心注册。 Protocol基于dubboProtocol扩展,利用扩展点的方式实现切面。调用框 架:ProtocolFilterWrapper->ProtocolListenerWrapper ->RegistryProtocol。真正的注册方法在RegistryProtocol中实现。

扩展点的思想贯穿dubbo的整个设计,也是dubbo扩展各种功能的基础。之后会进行详细介绍。

http://my.oschina.net/u/347227/blog/367703

转载于:https://my.oschina.net/xiaominmin/blog/551081

你可能感兴趣的文章
ASP.NET 生成图片验证码
查看>>
ToString()转换格式;DateTime.ToString()用法详解
查看>>
如何让Apache不显示服务器信息
查看>>
VSS配置
查看>>
数据分页存储过程
查看>>
事件流
查看>>
HDU-1532-Drainage Ditches
查看>>
angularJS的学习资源,巨全
查看>>
计算最长英语单词链
查看>>
LIS和LCS算法分析
查看>>
Error prompt:“xxx is not in the sudoers file”----Solution
查看>>
基于Silverlight的新浪微博客户端 - LightBus
查看>>
C#多维数组与嵌套数组
查看>>
Sublime text3的一些操作
查看>>
MySQL-ProxySQL中间件(二)| Admin Schemas介绍
查看>>
Python 设置 IP 代理 访问网页 ( 用户名密码验证代理 )
查看>>
从点云中提取杆状物算法
查看>>
sqlplus用户登录
查看>>
更轻松的获取APK文件安装时间
查看>>
Web标准中用于改善Web应用程序性能的各种方法总结
查看>>