Druid(二)——Druid中用到的一些技术

Guice框架

Guice是Google开发的一个轻量级的DI框架,Guice在2008年获得了软件界的奥斯卡--Jolt奖,核心的功能就是依赖反转。Guice的依赖反转比较简单,在Guice中有一个Module的接口,实现Module接口在configure方法中进行bind-to的操作,就完成了一个类型对应实现的绑定。Druid中大量的使用了Guice,要搞清楚代码究竟干了什么,首先先了解下Druid中是如何使用Guice的。

常量绑定

在Guice中可以把一个值与一个常量绑定。

public class A {
    ...
    @Inject
    @Named("aConstant")
    private String aConstant;
}

上面的代码中有一个字段是通过注入来设置值的,对于种类型的bind是这样的:

public class AModule implements Module{
    public void configure(Binder binder){
        binder.bindConstant().annotatedWith(Names.named("aConstant")).to("aConstant")
    }
}

上述代码的意思是绑定一个用Named注解修饰的常量,且注解的值是aConstant注入值为aConstant。

直连绑定

接口绑定就是最基础的常用的绑定,如果对SpringMVC比较熟悉的话接口绑定应该是最常用的绑定的方式了。举个简单的例子,我们想对接口TestInterface绑定一个实现可以使用直接绑定

binder.bind(TestInterface.class).to(TestInterfaceImpl.class).in(Scopes.SINGLETON);
绑定注解

有的时候对于同一个接口可能有多个实现,这个时候可以用一个额外的注解来进行限定。比如对于上个例子的TestInterface有一个其他的实现,这个时候可以创建一个绑定注解:

@BindingAnnotation
@Retention(RUNTIME)
@Target({FIELD, METHOD, TYPE})
public @interface Test {
}

然后我们在Module中就可以这样绑定:

binder.bind(TestInterface.class)
                .annotatedWith(Test.class)
                .to(TestAnnotationImpl.class)
                .in(Scopes.SINGLETON);

获取有两种常见的方式,第一种是使用Test注解:

    @Inject
    public C(B b, @Test TestInterface testInterface) {
        this.testInterface = testInterface;
        this.b = b;
    }

另一种方式是通过Key去获取:

TestInterface annotationInterface = injector.getInstance(Key.get(TestInterface.class, Test.class));
Provides注解

当你需要一定的代码来创建一个对象的时候可以使用@Providers注解,注解修饰一个方法,这个方法必须放到Module中,同时可以结合上面绑定注解一起使用。对于上个如果使用Providers注解的话首先把绑定的代码去掉就是binder.bind那里,其次在任意一个Module中provide一个对象:

    @Test
    @Provides
    @Singleton
    public TestInterface provideTest(){
        return new TestProvidersImpl();
    }

获取的方式是不变的。

隐式绑定

Druid中使用的隐式绑定和文档中的不太一样,看起来Druid中的隐式绑定更多的是为了注入,Druid中隐式绑定是直接写一个类,然后在bind的时候并没有声明target。这样做的好处是有些东西并不需要写接口,但是你可能要注入。

public class A {

    private B b;

    @Inject
    public A(B b) {
        this.b = b;
    }

    public void doBTest(){
        b.doB();
    }
}

绑定的时候:

binder.bind(A.class).in(Scopes.SINGLETON);
Provider接口

当@Providers的方法开始变得复杂了,你想把它移动到自己的类中,那么可以通过实现Provider接口来达到目的。如下:

public class TestProvider implements Provider<TestInterface> {

    @Override
    public TestInterface get() {
        return new TestInterface() {
            public void test() {
                System.out.println("provider test");
            }
        };
    }

}

绑定的部分:

binder.bind(TestInterface.class).toProvider(TestProvider.class).in(Scopes.SINGLETON);
MapBinder

MapBinder顾名思义就是注入一个实现的Map

MapBinder<String, CheckHandler> mapBinder = MapBinder.newMapBinder(binder(), String.class, CheckHandler.class);
mapBinder.addBinding("Hello").to(InstalledCheckHandler.class);
JsonConfigProvider

Druid的代码中大量的使用了JsonConfigProvider,这个类主要是用于读取配置,实际上它干的事儿跟Springboot中@Configuration很像。它就是将配置文件中一些固定前缀的配置项映射到一个配置类中。如果没有使用过Springboot的话,对于配置类还是比较陌生的,配置类其实就是在用Java去写配置,比如一个Server,你可能需要一个ServerConfig类,这个类中有一些字段都是Server的配置,如host,port等等。当你需要使用ServerConfig的时候只需要调用ServerConfig中相应的方法就可以。看一下Druid中例子:

JsonConfigProvider.bind(binder, "druid.indexer.task", TaskConfig.class);

这样调用的结果是将druid.indexer.task为前缀的property项目都映射到TaskConfig的同名字段中。比如druid.indexer.task.baseDir会映射到TaskConfig的baseDir字段,前提是类中需要一些Jackson的注解的设置。
这个类注入了一个Properties对象,这个对象包含配置文件中的配置,同时还会注入一个JsonConfigurator。JsonConfigProvider是一个实现了Provider<Supplier<T>>接口的泛型类。看一下bind方法:

public static <T> void bind(
        Binder binder,
        String propertyBase,
        Class<T> clazz,
        Key<T> instanceKey,
        Key<Supplier<T>> supplierKey) {
    binder.bind(supplierKey).toProvider((Provider) of(propertyBase, clazz)).in(LazySingleton.class);
    binder.bind(instanceKey).toProvider(new SupplierProvider<T>(supplierKey));
    }

这个bind方法是把Supplier<T>绑定一个Provider,这个Provider就是这个JsonProvider。它的get方法:

    public Supplier<T> get() {
        if (retVal != null) {
            return retVal;
        }

        try {
            final T config = configurator.configurate(props, propertyBase, classToProvide);
            retVal = Suppliers.ofInstance(config);
        } catch (RuntimeException e) {
            retVal = Suppliers.ofInstance(null);
            throw e;
        }
        return retVal;
    }

可以看出是通过JsonConfigurator生成一个配置类的实例,然后包一个Supplier返回回来。获取实例的部分就是把配置文件放到一个Map里面,然后结合类中Jackson注解通过Jackson框架来进行赋值处理。
上面的bind方法除了把Supplier<T>绑定了一个Provider之外,又把T绑定了一个SupplierProvider,看下SupplierProvider的代码:

public class SupplierProvider<T> implements Provider<T> {
    private final Key<Supplier<T>> supplierKey;

    private Provider<Supplier<T>> supplierProvider;

    public SupplierProvider(Key<Supplier<T>> supplierKey) {
        this.supplierKey = supplierKey;
    }

    @Inject
    public void configure(Injector injector) {
        this.supplierProvider = injector.getProvider(supplierKey);
    }

    @Override
    public T get() {
        return supplierProvider.get().get();
    }
}

构造器中传入的是一个supplierKey,然后注入的时候通过SupplierKey拿到SupplierPovider,然后get方法是supplierProvider.get().get()

PolyBind

PolyBind主要是利用MapBinder来创建一个可选的Binding,比如Overlord的运行模式有两个,一个是本地,一个是远程。通过配置文件来指定本地模式运行或者远程模式运行,这个时候就用PolyBind来实现这个功能。首先TaskRunner是通过TaskRunnerFactory生产出来的,所以两种不同的实现对应两种不同的Factory。所以对于TaskRunner就变成了根据配置文件不同的属性来选择不同的Factory。

PolyBind.createChoice(
        binder,
        "druid.indexer.runner.type",
        Key.get(TaskRunnerFactory.class),
        Key.get(ForkingTaskRunnerFactory.class));

看一下createChoice的实现:

    public static <T> ScopedBindingBuilder createChoiceWithDefault(
            Binder binder,
            String property,
            Key<T> interfaceKey,
            Key<? extends T> defaultKey,
            String defaultPropertyValue) {
        return binder.bind(interfaceKey).toProvider(new ConfiggedProvider<T>(interfaceKey, property, defaultKey, defaultPropertyValue));
    }

这个ConfiggedProvider是干什么的呢?

    static class ConfiggedProvider<T> implements Provider<T> {
        private final Key<T> key;
        private final String property;
        private final Key<? extends T> defaultKey;
        private final String defaultPropertyValue;

        private Injector injector;
        private Properties props;

        ConfiggedProvider(
                Key<T> key,
                String property,
                Key<? extends T> defaultKey,
                String defaultPropertyValue
        ) {
            this.key = key;
            this.property = property;
            this.defaultKey = defaultKey;
            this.defaultPropertyValue = defaultPropertyValue;
        }

        @Inject
        void configure(Injector injector, Properties props) {
            this.injector = injector;
            this.props = props;
        }

        @Override
        @SuppressWarnings("unchecked")
        public T get() {
            final ParameterizedType mapType = Types.mapOf(
                    String.class, Types.newParameterizedType(Provider.class, key.getTypeLiteral().getType())
            );

            final Map<String, Provider<T>> implsMap;
            if (key.getAnnotation() != null) {
                implsMap = (Map<String, Provider<T>>) injector.getInstance(Key.get(mapType, key.getAnnotation()));
            } else if (key.getAnnotationType() != null) {
                implsMap = (Map<String, Provider<T>>) injector.getInstance(Key.get(mapType, key.getAnnotation()));
            } else {
                implsMap = (Map<String, Provider<T>>) injector.getInstance(Key.get(mapType));
            }

            String implName = props.getProperty(property);
            if (implName == null) {
                implName = defaultPropertyValue;
            }
            final Provider<T> provider = implsMap.get(implName);

            if (provider == null) {
                if (defaultKey == null) {
                    throw new ProvisionException(
                            String.format("Unknown provider[%s] of %s, known options[%s]", implName, key, implsMap.keySet())
                    );
                }
                return injector.getInstance(defaultKey);
            }

            return provider.get();
        }
    }

可以看到,这个ConfiggedProvider首先注入了一个properties,然后再get方法中,首先先拿到接口的MapBind,然后用出入的property作为key去properties中取获取implName,如果implName是null则用默认的,这里默认的也可能为null。如果从BindMap中没有取到对应的实现,则通过defaultKey直接获取对应的实例,然后返回。如果获取到对应的Provider,则调用Provider的get方法。
CreateChoiceWithDefault之后,这个接口就对应了一个ConfiggedProvider的实现,如果想继续添加选项可以配合使用PolyBind.optionBinder和addBinding来处理。

final MapBinder<String, TaskRunnerFactory> biddy = PolyBind.optionBinder(
        binder,
        Key.get(TaskRunnerFactory.class)
);
biddy.addBinding("local").to(ForkingTaskRunnerFactory.class);

这个就是很简单了,就是创建一个对应接口的MapBind,然后通过addBind向Map里面添加值。总结起来就是这个接口对应一个Provider,Provider的get方法会去找基于这个接口的MapBind,然后从MapBind中找Prop中指定的实现,如果没有就用默认的实现。MapBind可以通过OptionBinder来创建,通过addBInder方法添加Key-Impl。

LifeCycle

LifeCycle就是类似于Spring的postadd和predestroy的东西,想想当年Twell没用Spring也搞过类似的东西。注册到LifeCycle里的东西必须有start和close方法或者有@LifeCycleStart注解或者@LifeCycleStop注解修饰的方法。LifeCycle中有State的概念,分为Normal和Last两个值,start过程中先执行Normal的,都是Normal的按照加入的顺序去执行,然后执行Last的,在关闭的过程中就按照完全相反的顺序去执行。

Jetty+Jersey+guice

Druid的服务使用的是一个内置的Jetty,rest接口使用的是Jersey。至少看起来是这一个样子的。Druid不同的节点会有不同的Server配置。这个是通过实现JettyServerInitializer接口实现的,什么时候调用的initialize方法呢?是在JettyServerModule中调用的,通过@Provides注解来实现的。同时这个Module还对GuiceContainer进行了Bind,DruidGuiceContainer继承了GuiceContainer,通过Guice将JSR311Resource注解修饰的类都注入进来作为Resource,Resource上就是http接口。所以想要增加http接口只需要写一个Resource类,然后通过Jerseys.addResource添加进来就可以。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容