成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Java9模塊化學(xué)習(xí)筆記二之模塊設(shè)計(jì)模式

李文鵬 / 801人閱讀

摘要:但是模塊化當(dāng)中,無(wú)法掃描只有模塊中可以使用有兩種解決方案定義一個(gè)專門的資源模塊,并使用提供的接口,實(shí)現(xiàn)它,并將這個(gè)實(shí)現(xiàn)注冊(cè)為服務(wù)。有兩種方式使用或包名,包名模塊名使用運(yùn)行時(shí)動(dòng)態(tài)。

模塊設(shè)計(jì)的原則:

1、防止出現(xiàn)編譯時(shí)循環(huán)依賴(主要是編譯器不支持),但運(yùn)行時(shí)是允許循環(huán)依賴的,比如GUI應(yīng)用
2、明確模塊的邊界

幾種模塊設(shè)計(jì):
API模塊,聚合模塊(比如java.base)

可選依賴

兩種方式:
1、可選的編譯時(shí)依賴(類似于maven的provided scope)聲明: requires static , requires transitive static
2、使用services模式,缺點(diǎn)就是需要使用侵入性的ServiceLoader API

使用編譯時(shí)可選依賴
module framework {
  requires static fastjsonlib;
}
public static void main(String... args) {
    try {
      Class clazz = Class.forName("javamodularity.fastjsonlib.FastJson");
      FastJson instance =
        (FastJson) clazz.getConstructor().newInstance();
      System.out.println("Using FastJson");
    } catch (ReflectiveOperationException e) {
      System.out.println("Oops, we need a fallback!");
    }
  }

注意,通過(guò)requires static聲明后,運(yùn)行時(shí),即使fastjsonlib模塊在模塊路徑中,仍然會(huì)跑到異常塊中,因?yàn)閞equies static聲明的模塊不會(huì)出現(xiàn)在模塊解析路徑上。除非你通過(guò)jlink打包時(shí),加入--add-modules fastjsonlib選項(xiàng)來(lái)顯式將其添加到模塊解析路徑(通過(guò)--add-modules也是作為一個(gè)root module).

使用Services模式的可選依賴

請(qǐng)參考之前的對(duì)于Services的探討

Versioned Modules

jar命令打包時(shí)可以通過(guò) --module-version=選項(xiàng)支持將版本添加到module-info.class中作為一個(gè)屬性。但是對(duì)于模塊解析而言,版本是沒有意義的,模塊解析過(guò)程中,只看模塊名,不支持版本。
所以如果需要版本化,還是得借助于Maven,Gradle之類的打包工具。

資源封裝

分模塊內(nèi)資源訪問、模塊間資源訪問

模塊內(nèi)資源訪問

firstresourcemodule/
├── javamodularity
│?? └── firstresourcemodule
│?? ├── ResourcesInModule.java
│?? ├── ResourcesOtherModule.java
│?? └── resource_in_package.txt 包內(nèi)資源
├── module-info.java
└── top_level_resource.txt 與module-info.java平級(jí)的資源

訪問方式有幾種,見下面代碼:

public class ResourcesInModule {

   public static void main(String... args) throws Exception {
      Class clazz = ResourcesInModule.class;
      InputStream cz_pkg = clazz.getResourceAsStream("resource_in_package.txt"); //<1> 
      URL cz_tl = clazz.getResource("/top_level_resource.txt"); //<2>

      Module m = clazz.getModule(); //<3>
      InputStream m_pkg = m.getResourceAsStream(
        "javamodularity/firstresourcemodule/resource_in_package.txt"); //<4>
      InputStream m_tl = m.getResourceAsStream("top_level_resource.txt"); //<5>

      assert Stream.of(cz_pkg, cz_tl, m_pkg, m_tl)
                   .noneMatch(Objects::isNull);
   }

}

在模塊化中,不推薦使用ClassLoder::getResource*
注意上面代碼中用到了Module API

跨模塊資源訪問

.
├── firstresourcemodule
│?? ├── javamodularity
│?? │?? └── firstresourcemodule
│?? │?? ├── ResourcesInModule.java
│?? │?? ├── ResourcesOtherModule.java
│?? │?? └── resource_in_package.txt
│?? ├── module-info.java
│?? └── top_level_resource.txt
└── secondresourcemodule

├── META-INF
│?? └── resource_in_metainf.txt
├── foo
│?? └── foo.txt
├── javamodularity
│?? └── secondresourcemodule
│??     ├── A.java
│??     └── resource_in_package2.txt
├── module-info.java
└── top_level_resource2.txt

注意,下面代碼的前提是兩個(gè)模塊的包都沒暴露給對(duì)方

public class ResourcesOtherModule {

   public static void main(String... args) throws Exception {
      Optional otherModule = ModuleLayer.boot().findModule("secondresourcemodule"); //<1>

      otherModule.ifPresent(other -> {
         try {
            InputStream m_tl = other.getResourceAsStream("top_level_resource2.txt"); //<2>
            InputStream m_pkg = other.getResourceAsStream(
                "javamodularity/secondresourcemodule/resource_in_package2.txt"); //<3>
            InputStream m_class = other.getResourceAsStream(
                "javamodularity/secondresourcemodule/A.class"); //<4>
            InputStream m_meta = other.getResourceAsStream("META-INF/resource_in_metainf.txt"); //<5>
            InputStream cz_pkg =
              Class.forName("javamodularity.secondresourcemodule.A")
                   .getResourceAsStream("resource_in_package2.txt"); //<6>

            assert Stream.of(m_tl, m_class, m_meta)
                         .noneMatch(Objects::isNull);
            assert Stream.of(m_pkg, cz_pkg)
                         .allMatch(Objects::isNull);

         } catch (Exception e) {
            throw new RuntimeException(e);
         }
      });
   }

}

請(qǐng)注意<1>中的ModuleLayer.boot() API
<2>說(shuō)明了模塊中的top-level資源總是可以被其他模塊訪問的
<3>將得到null,因?yàn)槟K2的包沒有開放給模塊1,模塊包中的資源訪問遵循模塊的封裝原則
<4>將返回結(jié)果,上面提到資源訪問遵循模塊封裝原則,但對(duì)于.class文件除外。(想想也是,因?yàn)槭窃试S運(yùn)行時(shí)獲取到別的模塊封裝的Class對(duì)象,只是不允許反射調(diào)用相關(guān)方法)
<5>由于META-INF不是一個(gè)包,所以其不會(huì)遵循模塊封裝原則,換言之,也像top-level資源一樣,是可以被其他模塊訪問的。
<6>Class.forName會(huì)正常調(diào)用,不過(guò)接著調(diào)用的.getResourceAsStream會(huì)返回null,就像<3>說(shuō)明的一樣。

記住一個(gè)原則:資源封裝只針對(duì)包下的(除.class外,包下的.class文件也可以被其他模塊訪問),其余的不會(huì)有封裝。

那么問題來(lái)了,如果我真的很想公開包下的資源給其他模塊呢?
使用open module或者opens 包名,比如:

open module aaa{
    ...
}

module aaa{

    opens a.b.c
}
ResourceBundle

我們知道jdk有個(gè)i18n資源加載API: ResourceBundle。它的行為是掃描classpath中的所有資源,只要符合baseName和Local即可加載到。
但是java9模塊化當(dāng)中,無(wú)法掃描classpath,只有模塊中可以使用ResourceBundle::getBundle
有兩種解決方案:
1、定義一個(gè)專門的i18n資源模塊,并open module
2、使用java9提供的ResourceBundleProvider接口,實(shí)現(xiàn)它,并將這個(gè)實(shí)現(xiàn)注冊(cè)為服務(wù)。

Deep Reflection 與 三方框架

深度反射與淺反射的區(qū)別:淺反射只是獲取基本的類信息,比如字段名,方法上的注解等,而深度反射會(huì)進(jìn)行字段賦值,方法調(diào)用等。
模塊化強(qiáng)封裝帶來(lái)的問題就是,我們沒法使用深度反射,比如對(duì)一個(gè)exports包中的某個(gè)公開類的private域進(jìn)行反射調(diào)用,field.setAccessible(true)之類的就會(huì)出現(xiàn)異常;對(duì)非exports包中的類進(jìn)行任何深度反射都是非法的。
那么我們熟悉的ORM框架,IOC框架等都廣泛地使用了深度反射。這就會(huì)導(dǎo)致問題。如何解決?使用Services肯定是不行的,因?yàn)榭蚣鼙旧砀膭?dòng)成本就會(huì)很大,沒幾個(gè)愿意這么改。
有兩種方式: 1、使用open module或opens 包名, opens 包名 to 模塊名;2、使用Module::addOpens運(yùn)行時(shí)動(dòng)態(tài)open。
java9還為反射類添加了canAccess方法、trySetAccessible方法

使用open module或opens 包名

open允許對(duì)open的模塊或包進(jìn)行深度反射

還有個(gè)問題,假如我們想對(duì)三方提供的模塊進(jìn)行深度反射,那該怎么辦呢,總不能去拿到別人的代碼改module-info.java聲明吧。這個(gè)時(shí)候就要用到j(luò)ava命令行參數(shù) --add-opens /=. 比如我想深度反射java.base中的java.lang包,那么可以 --add-opens java.base/java.lang=mymodule,但是如果我不使用模塊化,而只是使用classpath-based,那么我們可以使用--add-opens java.base/java.lang=ALL_UNNAMED,指定想未命名ALL_UNNAMED的代碼開放。

反射的替代方案:

java9基于JEP193提供了反射的替代方案用于訪問非public元素MethodHandles (始于java7),VarHandles(始于java9)
示例:
src

├── application
│   ├── javamodularity
│   │   └── application
│   │       ├── Book.java
│   │       └── Main.java
│   └── module-info.java
└── ormframework
    ├── javamodularity
    │   └── ormframework
    │       └── OrmFramework.java
    └── module-info.java
    

Book是一個(gè)POJO,里面有個(gè)private title字段
OrmFramework是一個(gè)模擬orm行為的demo,內(nèi)容如下:

ublic class OrmFramework {

  private Lookup lookup;

  public OrmFramework(Lookup lookup) { this.lookup = lookup; }

  public  T loadfromDatabase(String query, Class clazz) {
     try {
       MethodHandle ctor = lookup.findConstructor(clazz, MethodType.methodType(void.class));
       T entity =  (T) ctor.invoke();

       Lookup privateLookup = MethodHandles.privateLookupIn?(clazz, lookup);
       VarHandle title = privateLookup.findVarHandle(clazz, "title", String.class); // Name/type presumably found in some orm mapping config
       title.set(entity, "Loaded from database!");
       return entity;
     } catch(Throwable e) {
       throw new RuntimeException(e);
     }

  }

Main類內(nèi)容如下:

public static void main(String... args) {
    Lookup lookup = MethodHandles.lookup();
    OrmFramework ormFramework = new OrmFramework(lookup);
    Book book = ormFramework.loadfromDatabase("/* query */", Book.class);
    System.out.println(book.getTitle());
  }

你可能要問,為什么OrmFramework需要傳入Lookup,因?yàn)橹挥衋pplication模塊的Lookup才能有權(quán)限訪問那個(gè)模塊的非public元素,而OrmFramework模塊自己生成的Lookup是沒有權(quán)限訪問的。
所以使用MethodHandles與VarHandles時(shí)需要注意Lookup的權(quán)限

利用module相關(guān)api進(jìn)行反射

java.lang.module提供了三種類型的能力:1、查詢模塊屬性(主要基于module-info.java的內(nèi)容);2、運(yùn)行時(shí)動(dòng)態(tài)修改模塊的行為;3、模塊內(nèi)資源訪問
類圖:

1、查詢模塊屬性(主要基于module-info.java的內(nèi)容)

public class Introspection {

  public static void main(String... args) {
    Module module = String.class.getModule();

    String name1 = module.getName(); // Name as defined in module-info.java
    System.out.println("Module name: " + name1);

    Set packages1 = module.getPackages(); // Lists all packages in the module
    System.out.println("Packages in module: " + packages1);

    // The methods above are convenience methods that return
    // information from the Module"s ModuleDescriptor:
    ModuleDescriptor descriptor = module.getDescriptor();
    String name2 = descriptor.name(); // Same as module.getName();
    System.out.println("Module name from descriptor: " + name2);

    Set packages2 = descriptor.packages(); // Same as module.getPackages();
    System.out.println("Packages from descriptor: " + packages2);

    // Through ModuleDescriptor, all information from module-info.java is exposed:
    Set exports = descriptor.exports(); // All exports, possibly qualified
    System.out.println("Exports: " + exports);

    Set uses = descriptor.uses(); // All services used by this module
    System.out.println("Uses: " + uses);
  }

}

2、運(yùn)行時(shí)動(dòng)態(tài)修改模塊的行為
比如動(dòng)態(tài)exports

Module target=...
Module current=getClass().getModule();
current.addExports("com.test.in.Hello",target);

看了這段代碼,你可能要問,第二行,假如我是在別的模塊中調(diào)用,那么是不是任何模塊都可以修改其他模塊的exports,opens等屬性呢,非也,JVM運(yùn)行時(shí)會(huì)判斷Module對(duì)象的調(diào)用上下文,如果檢測(cè)到調(diào)用時(shí)非當(dāng)前模塊,那么就會(huì)出現(xiàn)異常。這種行為叫做Caller Sensitive

Caller Sensitive
jdk定義了很多caller sensitive的方法,只要是caller sensitive的方法都會(huì)被注解@CallerSensitive標(biāo)注,比如剛剛提到的Module::addExports,Field::setAccessible

Module API中可修改運(yùn)行時(shí)行為的幾個(gè)方法:
addExports(String pkgName, Module other)
addOpens(String pkgName, Module other)
addReads(Module other)

模塊上也可以加注解

@Deprecated
module m{
}

你也可以自定義模塊注解
注意:@Target(value={PACKAGE, MODULE})

@Retention(RetentionPolicy.RUNTIME)
@Target(value={PACKAGE, MODULE})
public @interface CustomAnnotation {

}
容器應(yīng)用模式 Layers And Configurations

ModuleLayer API、boot layer、layer的父子關(guān)系、一個(gè)layer可以有多個(gè)父layer
一個(gè)layer包含了當(dāng)前root模塊的解析圖(module resolution graph),一個(gè)應(yīng)用中可以有多個(gè)layer,但是只有一個(gè)boot layer,啟動(dòng)時(shí)的boot layer是java給你自動(dòng)創(chuàng)建的,你也可以手動(dòng)創(chuàng)建layer,那么這個(gè)創(chuàng)建的layer的parent就是boot layer。 只有boot layer才能解析platform module,但children layer可以共享boot layer中的Platform module,但是如果boot layer中沒有加載到的platform module,children module是無(wú)法使用的。

public static void main(String... args) {
    Driver driver = null; // We reference java.sql.Driver to see "java.sql" gets resolved
    ModuleLayer.boot().modules().forEach(m -> System.out.println(m.getName() + ", loader: " + m.getClassLoader()));
    System.out.println("System classloader: " + ClassLoader.getSystemClassLoader());
  }

創(chuàng)建ModuleLayer的示例:

ModuleFinder finder=ModuleFinder.of(Paths.get("../modules"));
ModuleLayer  bootLayer=ModuleLayer.boot();
//第二個(gè)Finder參數(shù)是在第一個(gè)finder中找不到模塊時(shí)才會(huì)去第二個(gè)finder中找,還有個(gè)resolveAndBind方法,區(qū)別在于,后者還會(huì)解析services provides/uses
Configuration config=bootLayer.configuration().resolve(finder,ModuleFinder.of(), Set.of("rootmodule")); 
ClassLoader cl=ClassLoader.getSystemClassLoader();
ModuleLayer newLayer=bootLayer.defineModulesWithOneLoader(config,cl);

上面的Configuration除了resolve方法外,還有個(gè)resolveAndBind方法,區(qū)別在于,后者還會(huì)解析services provides/uses

ClassLoaders in Layer


引入模塊化以后,去掉了之前的ExtClassLoader,引入了PlatformClassLoader

如果我們?yōu)槊總€(gè)layer都傳入不同的ClassLoader,那么則允許不同layer中存在相同的全限定類,這樣可以做到隔離與相互不干擾。

Plug-in 架構(gòu)

比如Eclipse,IDEA都是基于插件的應(yīng)用
在Java9中,我們有兩種方式來(lái)實(shí)現(xiàn)插件化:1、仍然利用以前的Services能力;2、結(jié)合ModuleLayer+Services實(shí)現(xiàn)封裝性更強(qiáng)的插件

public class PluginHostMain {

  public static void main(String... args) {
    if (args.length < 1) {
      System.out.println("Please provide plugin directories");
      return;
    }

    System.out.println("Loading plugins from " + Arrays.toString(args));

    Stream pluginLayers = Stream
      .of(args)
      .map(dir -> createPluginLayer(dir)); //<1>

    pluginLayers
      .flatMap(layer -> toStream(ServiceLoader.load(layer, Plugin.class))) // <2>
      .forEach(plugin -> {
         System.out.println("Invoking " + plugin.getName());
         plugin.doWork(); // <3>
      });
  }

  static ModuleLayer createPluginLayer(String dir) {
    ModuleFinder finder = ModuleFinder.of(Paths.get(dir));

    Set pluginModuleRefs = finder.findAll();
    Set pluginRoots = pluginModuleRefs.stream()
             .map(ref -> ref.descriptor().name())
             .filter(name -> name.startsWith("plugin")) // <1>
             .collect(Collectors.toSet());

    ModuleLayer parent = ModuleLayer.boot();
    Configuration cf = parent.configuration()
      .resolve(finder, ModuleFinder.of(), pluginRoots); // <2>

    ClassLoader scl = ClassLoader.getSystemClassLoader();
    ModuleLayer layer = parent.defineModulesWithOneLoader(cf, scl); // <3>

    return layer;
  }

  static  Stream toStream(Iterable iterable) {
    return StreamSupport.stream(iterable.spliterator(), false);
  }

}

Container架構(gòu)

比如tomcat,Jetty就是基于Container的應(yīng)用,支持運(yùn)行時(shí)動(dòng)態(tài)depoy和undeploy應(yīng)用。

與Plugin-in架構(gòu)的區(qū)別:1、Container支持運(yùn)行時(shí)deploy和undeploy;2、Plugin-in是用的是Services思路,而Container模式不應(yīng)該使用Services。這種情況下,就需要使用模塊的open功能,但是我們又不應(yīng)該強(qiáng)制應(yīng)用open,那么這就需要用到ModuleLayer.Controller::addOpens了,與Module::addOpens是Caller Sensitive不同,它可以實(shí)現(xiàn)跨模塊調(diào)用來(lái)修改模塊屬性。然后利用Deep reflection來(lái)實(shí)例化應(yīng)用類

private static void deployApp(int appNo) {
    AppDescriptor appDescr = apps[appNo];//AppDescriptor是自定義的類
    System.out.println("Deploying " + appDescr);

    ModuleLayer.Controller appLayerCtrl = createAppLayer(appDescr);
    Module appModule = appLayerCtrl.layer()
      .findModule(appDescr.rootmodule)
      .orElseThrow(() -> new IllegalStateException(appDescr.rootmodule + " missing"));

    appLayerCtrl.addOpens(appModule, appDescr.appClassPkg,
      Launcher.class.getModule());

    ContainerApplication app = instantiateApp(appModule, appDescr.appClass);
    deployedApps[appNo] = app;
    app.startApp();
  }

private static ModuleLayer.Controller createAppLayer(AppDescriptor appDescr) {
    ModuleFinder finder = ModuleFinder.of(Paths.get(appDescr.appDir));
    ModuleLayer parent = ModuleLayer.boot();

    Configuration cf = parent.configuration()
       .resolve(finder, ModuleFinder.of(), Set.of(appDescr.rootmodule));

    ClassLoader scl = ClassLoader.getSystemClassLoader();
    ModuleLayer.Controller layerCtrl =
      ModuleLayer.defineModulesWithOneLoader(cf, List.of(parent), scl);

    return layerCtrl;
  }

private static ContainerApplication instantiateApp(Module appModule, String appClassName) {
    try {
      ClassLoader cl = appModule.getClassLoader();
      Class appClass = cl.loadClass(appClassName);

      if(ContainerApplication.class.isAssignableFrom(appClass)) {
        return ((Class) appClass).getConstructor().newInstance();
      } else {
        System.out.println("WARNING: " + appClassName + " doesn"t implement ContainerApplication, cannot be started");
      }
    } catch (ReflectiveOperationException roe) {
      System.out.println("Could not start " + appClassName);
      roe.printStackTrace();
    }

注意點(diǎn):只有jvm啟動(dòng)時(shí)的boot layer才能解析platform module,在這里就是Container的root layer,但children layer可以共享boot layer中的Platform module,但是如果boot layer中沒有加載到的platform module,children module是無(wú)法使用的。所以Container啟動(dòng)時(shí)可以指定參數(shù)--add-modules ALL-SYSTEM這樣便可以解析所有的platform module到layer module graph中

總之:不管是Plugin-in還是Container模式,我們都需要適應(yīng)新的ModuleLayer API就像以前的ClassLoader API一樣

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/71464.html

相關(guān)文章

  • Java9模塊學(xué)習(xí)筆記三之遷移到Java9

    摘要:命令行參數(shù)文件鑒于遷移到后可能需要很長(zhǎng)的命令行參數(shù),有些會(huì)限制命令行長(zhǎng)度,支持定義一個(gè)命令行參數(shù)文件。已有三分庫(kù)可以自動(dòng)轉(zhuǎn)成模塊,只要在啟動(dòng)時(shí)將放在指定路徑中,便會(huì)自動(dòng)變成。 java[c]命令行參數(shù)文件 鑒于遷移到j(luò)ava9后可能需要很長(zhǎng)的命令行參數(shù),有些os會(huì)限制命令行長(zhǎng)度,java9支持定義一個(gè)命令行參數(shù)文件。使用方式: java @arguments.txt arguments...

    NeverSayNever 評(píng)論0 收藏0
  • Java9模塊學(xué)習(xí)筆記一之快速入門

    摘要:如果你想查看運(yùn)行時(shí)模塊的加載過(guò)程輸出結(jié)果表示為模塊,由于我限制了不再往下輸出了,而我們模塊又沒有別的額外依賴,所以僅有這行輸出。 jdk9模塊快速入門 列出自帶模塊:java --list-modulesmac多版本jdk共存:http://adolphor.com/blog/2016...模塊規(guī)則示意圖:showImg(https://segmentfault.com/img/bVb...

    cjie 評(píng)論0 收藏0
  • 《Java應(yīng)用架構(gòu)設(shè)計(jì):模塊模式與OSGi》讀書筆記

    摘要:本書概括以軟件系統(tǒng)為例,重點(diǎn)講解了應(yīng)用架構(gòu)中的物理設(shè)計(jì)問題,即如何將軟件系統(tǒng)拆分為模塊化系統(tǒng)。容器獨(dú)立模塊不依賴于具體容器,采用輕量級(jí)容器,如獨(dú)立部署模塊可獨(dú)立部署可用性模式發(fā)布接口暴露外部配置使用獨(dú)立的配置文件用于不同的上下文。 本文為讀書筆記,對(duì)書中內(nèi)容進(jìn)行重點(diǎn)概括,并將書中的模塊化結(jié)合微服務(wù)、Java9 Jigsaw談?wù)劺斫狻?本書概括 以Java軟件系統(tǒng)為例,重點(diǎn)講解了應(yīng)用架構(gòu)...

    seanHai 評(píng)論0 收藏0
  • Java9的新特性

    摘要:新特性概述系列一安裝及使用系列二運(yùn)行系列三模塊系統(tǒng)精要系列四更新系列五系列六系列七系列八系列九與的區(qū)別遷移注意事項(xiàng)參數(shù)遷移相關(guān)選項(xiàng)解析使用構(gòu)建實(shí)例使用示例帶你提前了解中的新特性 Java語(yǔ)言特性系列 Java5的新特性 Java6的新特性 Java7的新特性 Java8的新特性 Java9的新特性 Java10的新特性 Java11的新特性 Java12的新特性 Java13的新特性...

    ddongjian0000 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<