欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

10.Gradle自身源代碼編譯流程

一句話概括Gradle自身源代碼編譯流程-用gradle來編譯Gradle

創(chuàng)新互聯(lián)專業(yè)為企業(yè)提供湘陰網(wǎng)站建設(shè)、湘陰做網(wǎng)站、湘陰網(wǎng)站設(shè)計(jì)、湘陰網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計(jì)與制作、湘陰企業(yè)網(wǎng)站模板建站服務(wù),10年湘陰做網(wǎng)站經(jīng)驗(yàn),不只是建網(wǎng)站,更提供有價(jià)值的思路和整體網(wǎng)絡(luò)服務(wù)。


下面我們正式開始分析:


因?yàn)槲覀兡玫皆创a后,首先接觸的是gradlew.bat,也就是Gradle源代碼自身編譯的命令。所以,我們還是從這個腳本開始分析。


一. Eclipse打開源代碼

為了方便修改代碼,我選擇用Eclipse來打開這個工程。步驟是:

File->New->Java Project->Use default location去掉勾選->Browse選擇Gradle源代碼目錄->finish



二. gradlew.bat腳本

1. 還是從我們編譯Gradle源代碼的命令入手

gradlew.bat assemble

那首先來看下gradlw.bat :

@if "%DEBUG%" == "" @echo off
@rem ##########################################################################
@rem
@rem  Gradle startup script for Windows
@rem
@rem ##########################################################################

@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal

set DIRNAME=%~dp0
if "%DIRNAME%" == "" set DIRNAME=.
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%

@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS=-Xmx1024m -Dfile.encoding=UTF-8

@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome

set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if "%ERRORLEVEL%" == "0" goto init

echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.

goto fail

:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe

if exist "%JAVA_EXE%" goto init

echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.

goto fail

:init
@rem Get command-line arguments, handling Windows variants

if not "%OS%" == "Windows_NT" goto win9xME_args

:win9xME_args
@rem Slurp the command line arguments.
set CMD_LINE_ARGS=
set _SKIP=2

:win9xME_args_slurp
if "x%~1" == "x" goto execute

set CMD_LINE_ARGS=%*

:execute
@rem Setup the command line

set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar

@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%

:end
@rem End local scope for the variables with windows NT shell
if "%ERRORLEVEL%"=="0" goto mainEnd

:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
if  not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
exit /b 1

:mainEnd
if "%OS%"=="Windows_NT" endlocal

:omega

執(zhí)行g(shù)radlew.bat assemble時,首先來看看這個腳本里面的各個變量值:

CLASSPATH:gradle-3.1\\gradle\wrapper\gradle-wrapper.jar

表示的是gradle源代碼里面gralde\wrapper\目錄下gradle-wrapper.jar,這個jar也是待會要執(zhí)行的編譯操作要運(yùn)行的jar。

DEFAULT_JVM_OPTS:-Xmx1024m -Dfile.encoding=UTF-8 表示的是Java虛擬機(jī)配置

JAVA_OPTS:空

GRADLE_OPTS:空

CMD_LINE_ARGS:assemble 表示要執(zhí)行的gradle task名字

然后接下來,就會執(zhí)行重要的一句,啟動gradle-wrapper.jar里面的GradleWrapperMain.main函數(shù)。

@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%

那這里可能有個疑問,就是此時Gradle源代碼還沒有編譯出來,哪來的gradle-wrapper.jar。這個問題就像雞和蛋的問題,先有雞還是先有蛋。

Gradle的做法是先有雞后有蛋,那第一只雞哪來的呢? Gradle是自己給它造了一只雞。

請看gradle\wrapper\gradle-wrapper.jar

10. Gradle自身源代碼編譯流程

所以這里有個小的細(xì)節(jié)要提醒下大家,大家修改完GradleWrapperMain這個類之后,比如打印了日志,如果要驗(yàn)證它的結(jié)果,需要首先執(zhí)行幾個步驟

  1. gradlew.bat assemble

  2. 進(jìn)行編譯把編譯出來的gradle-wrapper.jar覆蓋到gradle\wrapper\目錄下。

  3. 再執(zhí)行一個gradlew.bat assemble就可以在命令行里面驗(yàn)證。

三. GradleWrapperMain

文件路徑:

gradle-3.1\subprojects\wrapper\src\main\java\org\gradle\wrapper\GradleWrapperMain.java

GradleWrapperMain位置subprojects里面,Gradle源代碼把各個工程拆分成各個模塊,類似于插件的方式。

也就是說每個功能都拆分成一個插件,然后使用的時候進(jìn)行配置,比如某個插件需要依賴于哪幾個插件,那就直接配置上就可以。

配置的路徑在每個插件的jar包里面,名稱叫做xxx-classpath.properties,里面有個projects屬性,配置了這個插件依賴的插件(也可以叫項(xiàng)目或者模塊)。

這種設(shè)計(jì)思想可以讓整個項(xiàng)目層次清晰,同時便于多個團(tuán)隊(duì)間合作開發(fā)。

看下GradleWrapperMain的main函數(shù):

public class GradleUserHomeLookup {
    public static final String DEFAULT_GRADLE_USER_HOME = System.getProperty("user.home") + "/.gradle";
    public static final String GRADLE_USER_HOME_PROPERTY_KEY = "gradle.user.home";
    public static final String GRADLE_USER_HOME_ENV_KEY = "GRADLE_USER_HOME";

    public static File gradleUserHome() {
        String gradleUserHome;
        if ((gradleUserHome = System.getProperty(GRADLE_USER_HOME_PROPERTY_KEY)) != null) {
            return new File(gradleUserHome);
        }
        if ((gradleUserHome = System.getenv(GRADLE_USER_HOME_ENV_KEY)) != null) {
            return new File(gradleUserHome);
        }
        return new File(DEFAULT_GRADLE_USER_HOME);
    }
}


public static void main(String[] args) throws Exception {
        File wrapperJar = wrapperJar();
        System.out.println("wrapperJar: " + wrapperJar);
        File propertiesFile = wrapperProperties(wrapperJar);
        File rootDir = rootDir(wrapperJar);

		...
        
        File gradleUserHome = gradleUserHome(options);

        System.out.println("gradleUserHome: " + gradleUserHome + " rootDir: " + rootDir
        		+ " options: " + options+ "propertiesFile: " + propertiesFile);
        addSystemProperties(gradleUserHome, rootDir);

        Logger logger = logger(options);

        WrapperExecutor wrapperExecutor = WrapperExecutor.forWrapperPropertiesFile(propertiesFile);
        wrapperExecutor.execute(
                args,
                new Install(logger, new Download(logger, "gradlew", wrapperVersion()), new PathAssembler(gradleUserHome)),
                new BootstrapMainStarter());
    }

打印的log是:

wrapperJar: E:\work_space\gradle-source-from-csdn\gradle-3.1\gradle\wrapper\gradle-wrapper.jar
gradleUserHome: D:\gradle_jar_cache rootDir: E:\work_space\gradle-source-from-csdn\gradle-3.1 options: options: , extraArguments: 'assemble', removedOptions: propertiesFile: E:\work_space\gradle-source-from-csdn\gradle-3.1\gradle\wrapper\gradle-wrapper.properties

這個日志已經(jīng)很清楚的說明了各個變量的值。

需要說明的一點(diǎn)是gradleUserHome,這個是Gradle下載其他Jar包的存放地址,默認(rèn)是c盤的user/xxx/.gradle/目錄。

但是這個目錄是可以配置的,配置GRADLE_USER_HOME環(huán)境變量即可。

這點(diǎn)從上面代碼getUserHome可以清楚的看到。

另外,程序的最后面Gradle執(zhí)行WrapperExecutor.execute(xxxx)方法,這個比較關(guān)鍵。

四. WrapperExecutor.execute


1. execute

public void execute(String[] args, Install install, BootstrapMainStarter bootstrapMainStarter) throws Exception {
        File gradleHome = install.createDist(config);
        bootstrapMainStarter.start(args, gradleHome);
    }

可以看到execute里面沒有什么東西,調(diào)用的是傳入的install.createDist和bootstrapMainStarter.start方法,所以,需要分析下這兩個方法。

2. Install.createDist

new Install(logger, new Download(logger, "gradlew", wrapperVersion()), new PathAssembler(gradleUserHome))



public File createDist(final WrapperConfiguration configuration) throws Exception {
        final URI distributionUrl = configuration.getDistribution();
        final String distributionSha256Sum = configuration.getDistributionSha256Sum();

        final PathAssembler.LocalDistribution localDistribution = pathAssembler.getDistribution(configuration);
        final File distDir = localDistribution.getDistributionDir();
        final File localZipFile = localDistribution.getZipFile();
        
        System.out.println("distributionUrl: " + distributionUrl + " distributionSha256Sum: " + distributionSha256Sum 
        		+ " localDistribution: " + localDistribution
        		+ " distDir: " + distDir
        		+ " localZipFile: " + localZipFile);

        return exclusiveFileAccessManager.access(localZipFile, new Callable<File>() {
            public File call() throws Exception {
                final File markerFile = new File(localZipFile.getParentFile(), localZipFile.getName() + ".ok");
                if (distDir.isDirectory() && markerFile.isFile()) {
                    return getAndVerifyDistributionRoot(distDir, distDir.getAbsolutePath());
                }

                boolean needsDownload = !localZipFile.isFile();

                if (needsDownload) {
                    File tmpZipFile = new File(localZipFile.getParentFile(), localZipFile.getName() + ".part");
                    tmpZipFile.delete();
                    logger.log("Downloading " + distributionUrl);
                    download.download(distributionUrl, tmpZipFile);
                    tmpZipFile.renameTo(localZipFile);
                }

                List<File> topLevelDirs = listDirs(distDir);
                for (File dir : topLevelDirs) {
                    logger.log("Deleting directory " + dir.getAbsolutePath());
                    deleteDir(dir);
                }

                verifyDownloadChecksum(configuration.getDistribution().toString(), localZipFile, distributionSha256Sum);

                logger.log("Unzipping " + localZipFile.getAbsolutePath() + " to " + distDir.getAbsolutePath());
                unzip(localZipFile, distDir);

                File root = getAndVerifyDistributionRoot(distDir, distributionUrl.toString());
                setExecutablePermissions(root);
                markerFile.createNewFile();

                return root;
            }
        });
    }

這是打印的日志:

distributionUrl: https://services.gradle.org/distributions/gradle-3.1-rc-1-bin.zip distributionSha256Sum: null localDistribution: org.gradle.wrapper.PathAssembler$LocalDistribution@4a574795 distDir: D:\gradle_jar_cache\wrapper\dists\gradle-3.1-rc-1-bin\3uhcvxvcic1j9jh0j26e3y151 localZipFile: D:\gradle_jar_cache\wrapper\dists\gradle-3.1-rc-1-bin\3uhcvxvcic1j9jh0j26e3y151\gradle-3.1-rc-1-bin.zip

這里有幾個問題:

a. 下載zip包

createDist其實(shí)就是去下載distributionUrl描述的zip包。

其實(shí)這還是個雞和蛋的問題,Gradle源代碼是用Gradle來編譯的,現(xiàn)在我們只有源代碼,那怎么編譯呢?

所以就只能先從服務(wù)器上把Gradle zip包下載下來。

b. distributionUrl定義位置

gradle源代碼根目錄/gradle/wrapper/gradle-wrapper.properties

也就是:

#Mon Sep 12 15:17:35 CEST 2016
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-3.1-rc-1-bin.zip

配置文件位置可以在WrapperExecutor的構(gòu)造函數(shù)看出來:

ublic static WrapperExecutor forProjectDirectory(File projectDir) {
        return new WrapperExecutor(new File(projectDir, "gradle/wrapper/gradle-wrapper.properties"), new Properties());
    }

c. 根據(jù)md5計(jì)算出zip文件的存放目錄

distDir: D:\gradle_jar_cache\wrapper\dists\gradle-3.1-rc-1-bin\3uhcvxvcic1j9jh0j26e3y151

distDir目錄,也就是下載下來的zip文件存放目錄有一串字符串,這是根據(jù)md5算出來的,保證唯一性,代碼如下:

文件路徑:

subprojects\wrapper\src\main\java\org\gradle\wrapper\PathAssembler.java

/**
     * This method computes a hash of the provided {@code string}.
     * <p>
     * The algorithm in use by this method is as follows:
     * <ol>
     *    <li>Compute the MD5 value of {@code string}.</li>
     *    <li>Truncate leading zeros (i.e., treat the MD5 value as a number).</li>
     *    <li>Convert to base 36 (the characters {@code 0-9a-z}).</li>
     * </ol>
     */
    private String getHash(String string) {
        try {
            MessageDigest messageDigest = MessageDigest.getInstance("MD5");
            byte[] bytes = string.getBytes();
            messageDigest.update(bytes);
            return new BigInteger(1, messageDigest.digest()).toString(36);
        } catch (Exception e) {
            throw new RuntimeException("Could not hash input string.", e);
        }
    }

d. 執(zhí)行時機(jī)

在第一次執(zhí)行g(shù)radlew.bat assemble的時候會去下載,然后解壓。

所以第一次執(zhí)行g(shù)radlew.bat assemble會出現(xiàn)這樣日志:

Downloading xxxx.......

Unzipping.....

3. bootstrapMainStarter.start

文件路徑:

subprojects\wrapper\src\main\java\org\gradle\wrapper\BootstrapMainStarter.java

public void start(String[] args, File gradleHome) throws Exception {
    	System.out.println("BootstrapMainStarter gradleHome: " + gradleHome);
    	if (args != null) {
			for(int i = 0; i< args.length; i++) {
				System.out.println("args[" + i+"]= " + args[i]);
			}
		}
    	
        File gradleJar = findLauncherJar(gradleHome);
        URLClassLoader contextClassLoader = new URLClassLoader(new URL[]{gradleJar.toURI().toURL()}, ClassLoader.getSystemClassLoader().getParent());
        Thread.currentThread().setContextClassLoader(contextClassLoader);
        Class<?> mainClass = contextClassLoader.loadClass("org.gradle.launcher.GradleMain");
        Method mainMethod = mainClass.getMethod("main", String[].class);
        mainMethod.invoke(null, new Object[]{args});
        if (contextClassLoader instanceof Closeable) {
            ((Closeable) contextClassLoader).close();
        }
    }

    private File findLauncherJar(File gradleHome) {
        for (File file : new File(gradleHome, "lib").listFiles()) {
            if (file.getName().matches("gradle-launcher-.*\\.jar")) {
                return file;
            }
        }
        throw new RuntimeException(String.format("Could not locate the Gradle launcher JAR in Gradle distribution '%s'.", gradleHome));
    }

打印日志如下:

BootstrapMainStarter gradleHome: D:\gradle_jar_cache\wrapper\dists\gradle-3.1-rc-1-bin\3uhcvxvcic1j9jh0j26e3y151\gradle-3.1-rc-1
args[0]= assemble

那么現(xiàn)在來解釋下上面這段代碼是在干什么:

a. 從gradle bin下載解壓目錄的lib文件夾找到gradle-launcher-.*\\.jar。

b. 然后執(zhí)行l(wèi)auncher-xxx.jar包里面的org.gradle.launcher.GradleMain.main函數(shù)。

c. 同時,把我們輸入的參數(shù)assemble傳入進(jìn)去。

那么接下去執(zhí)行的就是我們下載的gradle里面的launcher-xxx.jar的函數(shù)了,不是我們源代碼里面的,這點(diǎn)要特別區(qū)分清楚。

執(zhí)行下載的gradle里面的代碼來編譯Gradle源代碼的過程,就像gradle編譯其他程序一樣。

所以才說Gradle源代碼的編譯過程就是用gradle來編譯Gradle.

這個地方就是之前我們說過的gradlew.bat和gradle.bat的區(qū)別。

d. gradlew.bat和gradle.bat的區(qū)別

gradlew.bat是Gradle源代碼自身編譯時候的bat腳本

加載的是當(dāng)前目錄下 gralde/wrapper/gradle-wrapper.jar包,執(zhí)行的是GradleWrapperMain.main方法。

然后去下載gradle bin,再解壓。然后執(zhí)行g(shù)radle lib里面的gradle-launcher-xxx.jar里面的GradleMain.main函數(shù)。

gradle.bat是gradle編譯其他程序的bat腳本

加載的是gradle lib里面gradle-launcher-xxx.jar里面的GradleMain.main函數(shù)。

e. 關(guān)于gradlew.bat命名的思考

我在思考為什么編譯Gradle自身源代碼的腳本要叫g(shù)radlew.bat呢?和編譯其他程序的腳本gradle.bat只有一字之差?"w"代表的意思是什么呢?

gradlew.bat做的事情是編譯Gradle源代碼自身,那么叫叫g(shù)radle_compile_self.bat比較直觀點(diǎn)?

也許,我們可以理解"w"是wrapper的縮寫,因?yàn)閮蓚€腳本加載的Jar就是這樣的區(qū)別;而且wapper做的事情真的就是包裝的工作。比如它只是去下載gradle bin,然后就直接調(diào)用gradle bin的grale-launcher-xxx.jar了。

所以,也許這個就是Gradle團(tuán)隊(duì)對于"w"的理解吧。。

接下來的話,Gradle自身源代碼的編譯過程就和一般程序的編譯過程一樣了;我們再分析gradle編譯應(yīng)用程序過程。

文章標(biāo)題:10.Gradle自身源代碼編譯流程
分享網(wǎng)址:http://aaarwkj.com/article22/pdeejc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供外貿(mào)建站、面包屑導(dǎo)航、Google、網(wǎng)頁設(shè)計(jì)公司、做網(wǎng)站商城網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

網(wǎng)站優(yōu)化排名
亚洲女同在线免费观看| 国产区av中文字幕在线观看| av天堂官网在线人妻| 丰满少妇亚洲中文字幕| 日本黄色中文字幕网站| 五月天丁香婷婷一区二区| 国产 亚洲 一区 二区| 伊人亚洲中文一区二区| 成年黄网站免费视频大全| 日产一区二区三区网站| 日韩一区二区三区不卡| 亚洲国产欧美精品综合在线| 懂色av中文字幕一区| 亚洲一区二区天堂av| 在线欧美日韩一区二区三区| 有码精品视频在线观看| 91最新精品丝袜国产在线| 欧美中文字幕在线精品| 91激情黑丝在线观看| 精品色欧美色国产一区国产| 综合av在线一区天堂| 亚洲热妇热女久久精品| av在线中文字幕剧情| 亚洲精品黄色片中文字幕| 99精品国产综合久久麻豆| 男人一插就想射的原因| 国产精品日本在线观看| 不卡一区二区黄色av| 少妇高潮特黄在线观看| 成人av高清在线观看| 一起草视频在线观看视频| 国产欧美日韩精品一区| 国产一级r内射视频播放| 日本高清有码中文字幕| 久久精品国产精品亚洲片| 中文字幕日本人妻影视| 成人福利网站午夜一区| 日本午夜理论视频在线播放| 日韩中文字幕资源一区| 久久精品国产普通话对白| 亚洲激情在线观看一区|