├── MultidexSample
├── .gitignore
├── app
│ ├── .gitignore
│ ├── build.gradle
│ ├── multidex.keep
│ └── src
│ │ └── main
│ │ ├── AndroidManifest.xml
│ │ ├── java
│ │ └── com
│ │ │ └── github
│ │ │ └── mmadev
│ │ │ └── multidexsample
│ │ │ ├── MainActivity.java
│ │ │ └── utils
│ │ │ └── MultiDexUtils.java
│ │ └── res
│ │ ├── layout
│ │ ├── activity_main.xml
│ │ └── empty_view.xml
│ │ ├── mipmap-hdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-mdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xhdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xxhdpi
│ │ └── ic_launcher.png
│ │ ├── values-v21
│ │ └── styles.xml
│ │ ├── values-w820dp
│ │ └── dimens.xml
│ │ └── values
│ │ ├── dimens.xml
│ │ ├── strings.xml
│ │ └── styles.xml
├── build.gradle
├── gradle.properties
├── gradle
│ └── wrapper
│ │ ├── gradle-wrapper.jar
│ │ └── gradle-wrapper.properties
├── gradlew
├── gradlew.bat
└── settings.gradle
└── README.md
/MultidexSample/.gitignore:
--------------------------------------------------------------------------------
1 | .gradle
2 | /local.properties
3 | /.idea/workspace.xml
4 | /.idea/libraries
5 | .DS_Store
6 | /build
7 | /captures
8 |
--------------------------------------------------------------------------------
/MultidexSample/app/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/MultidexSample/app/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 |
3 | android {
4 | compileSdkVersion 23
5 | buildToolsVersion "23.0.1"
6 |
7 | defaultConfig {
8 | applicationId "com.github.mmadev.multidexsample"
9 | minSdkVersion 16
10 | targetSdkVersion 23
11 | versionCode 1
12 | versionName "1.0"
13 |
14 | // Enabling multidex support.
15 | multiDexEnabled true
16 | }
17 | buildTypes {
18 | release {
19 | minifyEnabled true
20 | proguardFiles getDefaultProguardFile('proguard-android.txt')
21 | }
22 | }
23 | }
24 |
25 | dependencies {
26 | compile fileTree(dir: 'libs', include: ['*.jar'])
27 | compile 'com.android.support:multidex:1.0.1'
28 | compile "com.android.support:support-v4:23.1.0"
29 | }
30 |
31 | android.applicationVariants.all { variant ->
32 | task "fix${variant.name.capitalize()}MainDexClassList" << {
33 | logger.info "Fixing main dex keep file for $variant.name"
34 | File keepFile = new File("$buildDir/intermediates/multi-dex/$variant.buildType.name/maindexlist.txt")
35 |
36 | keepFile.withWriterAppend { w ->
37 | // Get a reader for the input file
38 | w.append('\n')
39 | new File("${projectDir}/multidex.keep").withReader { r ->
40 | // And write data from the input into the output
41 | w << r << '\n'
42 | }
43 | logger.info "Updated main dex keep file for ${keepFile.getAbsolutePath()}\n$keepFile.text"
44 | }
45 | }
46 | }
47 |
48 | tasks.whenTaskAdded { task ->
49 | android.applicationVariants.all { variant ->
50 | if (task.name == "create${variant.name.capitalize()}MainDexClassList") {
51 | task.finalizedBy "fix${variant.name.capitalize()}MainDexClassList"
52 | }
53 | }
54 | }
55 |
56 | tasks.withType(com.android.build.gradle.tasks.Dex) { dexTask ->
57 | def command = [] as List
58 | command << '--minimal-main-dex'
59 | dexTask.setAdditionalParameters(command)
60 | }
61 |
--------------------------------------------------------------------------------
/MultidexSample/app/multidex.keep:
--------------------------------------------------------------------------------
1 | android/support/v4/util/ArrayMap.class
2 | android/support/v4/util/ContainerHelpers.class
3 | android/support/v4/util/LruCache.class
4 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
11 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/java/com/github/mmadev/multidexsample/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.github.mmadev.multidexsample;
2 |
3 | import android.app.ListActivity;
4 | import android.os.Bundle;
5 | import android.widget.ArrayAdapter;
6 | import android.widget.TextView;
7 |
8 | import com.github.mmadev.multidexsample.utils.MultiDexUtils;
9 |
10 | import java.util.List;
11 |
12 | public class MainActivity extends ListActivity {
13 |
14 | @Override
15 | protected void onCreate(Bundle savedInstanceState) {
16 | super.onCreate(savedInstanceState);
17 | setContentView(R.layout.activity_main);
18 | try {
19 | final List externalClasses = new MultiDexUtils().getExternalDexClasses(this);
20 | if(externalClasses != null) {
21 | final TextView header = new TextView(this);
22 | header.setText(R.string.header);
23 | getListView().addHeaderView(header);
24 | getListView().setAdapter(new ArrayAdapter<>(this, android.R.layout.simple_list_item_1, externalClasses));
25 | }
26 | } catch (Exception e) {
27 | e.printStackTrace();
28 | }
29 | }
30 | }
31 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/java/com/github/mmadev/multidexsample/utils/MultiDexUtils.java:
--------------------------------------------------------------------------------
1 | package com.github.mmadev.multidexsample.utils;
2 |
3 | import android.content.Context;
4 | import android.content.SharedPreferences;
5 | import android.content.pm.ApplicationInfo;
6 | import android.content.pm.PackageManager;
7 | import android.os.Build;
8 |
9 | import java.io.File;
10 | import java.io.IOException;
11 | import java.util.ArrayList;
12 | import java.util.Enumeration;
13 | import java.util.List;
14 |
15 | import dalvik.system.DexFile;
16 |
17 | public class MultiDexUtils {
18 | private static final String EXTRACTED_NAME_EXT = ".classes";
19 | private static final String EXTRACTED_SUFFIX = ".zip";
20 |
21 | private static final String SECONDARY_FOLDER_NAME = "code_cache" + File.separator +
22 | "secondary-dexes";
23 |
24 | private static final String PREFS_FILE = "multidex.version";
25 | private static final String KEY_DEX_NUMBER = "dex.number";
26 |
27 | private SharedPreferences getMultiDexPreferences(Context context) {
28 | return context.getSharedPreferences(PREFS_FILE,
29 | Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB
30 | ? Context.MODE_PRIVATE
31 | : Context.MODE_PRIVATE | Context.MODE_MULTI_PROCESS);
32 | }
33 |
34 | /**
35 | * get all the dex path
36 | *
37 | * @param context the application context
38 | * @return all the dex path
39 | * @throws PackageManager.NameNotFoundException
40 | * @throws IOException
41 | */
42 | public List getSourcePaths(Context context) throws PackageManager.NameNotFoundException, IOException {
43 | final ApplicationInfo applicationInfo = context.getPackageManager().getApplicationInfo(context.getPackageName(), 0);
44 | final File sourceApk = new File(applicationInfo.sourceDir);
45 | final File dexDir = new File(applicationInfo.dataDir, SECONDARY_FOLDER_NAME);
46 |
47 | final List sourcePaths = new ArrayList<>();
48 | sourcePaths.add(applicationInfo.sourceDir); //add the default apk path
49 |
50 | //the prefix of extracted file, ie: test.classes
51 | final String extractedFilePrefix = sourceApk.getName() + EXTRACTED_NAME_EXT;
52 | //the total dex numbers
53 | final int totalDexNumber = getMultiDexPreferences(context).getInt(KEY_DEX_NUMBER, 1);
54 |
55 | for (int secondaryNumber = 2; secondaryNumber <= totalDexNumber; secondaryNumber++) {
56 | //for each dex file, ie: test.classes2.zip, test.classes3.zip...
57 | final String fileName = extractedFilePrefix + secondaryNumber + EXTRACTED_SUFFIX;
58 | final File extractedFile = new File(dexDir, fileName);
59 | if (extractedFile.isFile()) {
60 | sourcePaths.add(extractedFile.getAbsolutePath());
61 | //we ignore the verify zip part
62 | } else {
63 | throw new IOException("Missing extracted secondary dex file '" +
64 | extractedFile.getPath() + "'");
65 | }
66 | }
67 |
68 | return sourcePaths;
69 | }
70 |
71 | /**
72 | * get all the external classes name in "classes2.dex", "classes3.dex" ....
73 | *
74 | * @param context the application context
75 | * @return all the classes name in the external dex
76 | * @throws PackageManager.NameNotFoundException
77 | * @throws IOException
78 | */
79 | public List getExternalDexClasses(Context context) throws PackageManager.NameNotFoundException, IOException {
80 | final List paths = getSourcePaths(context);
81 | if(paths.size() <= 1) {
82 | // no external dex
83 | return null;
84 | }
85 | // the first element is the main dex, remove it.
86 | paths.remove(0);
87 | final List classNames = new ArrayList<>();
88 | for (String path : paths) {
89 | try {
90 | DexFile dexfile = null;
91 | if (path.endsWith(EXTRACTED_SUFFIX)) {
92 | //NOT use new DexFile(path), because it will throw "permission error in /data/dalvik-cache"
93 | dexfile = DexFile.loadDex(path, path + ".tmp", 0);
94 | } else {
95 | dexfile = new DexFile(path);
96 | }
97 | final Enumeration dexEntries = dexfile.entries();
98 | while (dexEntries.hasMoreElements()) {
99 | classNames.add(dexEntries.nextElement());
100 | }
101 | } catch (IOException e) {
102 | throw new IOException("Error at loading dex file '" +
103 | path + "'");
104 | }
105 | }
106 | return classNames;
107 | }
108 |
109 | /**
110 | * Get all loaded external classes name in "classes2.dex", "classes3.dex" ....
111 | * @param context
112 | * @return get all loaded external classes
113 | */
114 | public List getLoadedExternalDexClasses(Context context) {
115 | try {
116 | final List externalDexClasses = getExternalDexClasses(context);
117 | if (externalDexClasses != null && !externalDexClasses.isEmpty()) {
118 | final ArrayList classList = new ArrayList<>();
119 | final java.lang.reflect.Method m = ClassLoader.class.getDeclaredMethod("findLoadedClass", new Class[]{String.class});
120 | m.setAccessible(true);
121 | final ClassLoader cl = context.getClassLoader();
122 | for (String clazz : externalDexClasses) {
123 | if (m.invoke(cl, clazz) != null) {
124 | classList.add(clazz.replaceAll("\\.", "/").replaceAll("$", ".class"));
125 | }
126 | }
127 | return classList;
128 | }
129 | } catch (Exception e) {
130 | e.printStackTrace();
131 | }
132 | return null;
133 | }
134 | }
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
7 |
11 |
12 |
14 |
15 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/layout/empty_view.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mmadev/multidex-sample/cc2d0ca9c441025467f262596e4aa3ac5794b1d4/MultidexSample/app/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mmadev/multidex-sample/cc2d0ca9c441025467f262596e4aa3ac5794b1d4/MultidexSample/app/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mmadev/multidex-sample/cc2d0ca9c441025467f262596e4aa3ac5794b1d4/MultidexSample/app/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mmadev/multidex-sample/cc2d0ca9c441025467f262596e4aa3ac5794b1d4/MultidexSample/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/values-v21/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
5 |
6 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/values-w820dp/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 | 64dp
6 |
7 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/values/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 16dp
4 | 16dp
5 |
6 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | MultidexSample
3 | No class stored in the external dex
4 | List of classes stored in the external dex
5 |
6 |
--------------------------------------------------------------------------------
/MultidexSample/app/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/MultidexSample/build.gradle:
--------------------------------------------------------------------------------
1 | // Top-level build file where you can add configuration options common to all sub-projects/modules.
2 |
3 | buildscript {
4 | repositories {
5 | jcenter()
6 | }
7 | dependencies {
8 | classpath 'com.android.tools.build:gradle:1.3.1'
9 |
10 | // NOTE: Do not place your application dependencies here; they belong
11 | // in the individual module build.gradle files
12 | }
13 | }
14 |
15 | task wrapper(type: Wrapper) {
16 | gradleVersion = '2.8'
17 | }
18 |
19 |
20 | allprojects {
21 | repositories {
22 | jcenter()
23 | }
24 | }
25 |
--------------------------------------------------------------------------------
/MultidexSample/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
--------------------------------------------------------------------------------
/MultidexSample/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mmadev/multidex-sample/cc2d0ca9c441025467f262596e4aa3ac5794b1d4/MultidexSample/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/MultidexSample/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Tue Nov 24 04:47:08 PST 2015
2 | distributionBase=GRADLE_USER_HOME
3 | distributionPath=wrapper/dists
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.8-bin.zip
7 |
--------------------------------------------------------------------------------
/MultidexSample/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # For Cygwin, ensure paths are in UNIX format before anything is touched.
46 | if $cygwin ; then
47 | [ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
48 | fi
49 |
50 | # Attempt to set APP_HOME
51 | # Resolve links: $0 may be a link
52 | PRG="$0"
53 | # Need this for relative symlinks.
54 | while [ -h "$PRG" ] ; do
55 | ls=`ls -ld "$PRG"`
56 | link=`expr "$ls" : '.*-> \(.*\)$'`
57 | if expr "$link" : '/.*' > /dev/null; then
58 | PRG="$link"
59 | else
60 | PRG=`dirname "$PRG"`"/$link"
61 | fi
62 | done
63 | SAVED="`pwd`"
64 | cd "`dirname \"$PRG\"`/" >&-
65 | APP_HOME="`pwd -P`"
66 | cd "$SAVED" >&-
67 |
68 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
69 |
70 | # Determine the Java command to use to start the JVM.
71 | if [ -n "$JAVA_HOME" ] ; then
72 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
73 | # IBM's JDK on AIX uses strange locations for the executables
74 | JAVACMD="$JAVA_HOME/jre/sh/java"
75 | else
76 | JAVACMD="$JAVA_HOME/bin/java"
77 | fi
78 | if [ ! -x "$JAVACMD" ] ; then
79 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
80 |
81 | Please set the JAVA_HOME variable in your environment to match the
82 | location of your Java installation."
83 | fi
84 | else
85 | JAVACMD="java"
86 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
87 |
88 | Please set the JAVA_HOME variable in your environment to match the
89 | location of your Java installation."
90 | fi
91 |
92 | # Increase the maximum file descriptors if we can.
93 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
94 | MAX_FD_LIMIT=`ulimit -H -n`
95 | if [ $? -eq 0 ] ; then
96 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
97 | MAX_FD="$MAX_FD_LIMIT"
98 | fi
99 | ulimit -n $MAX_FD
100 | if [ $? -ne 0 ] ; then
101 | warn "Could not set maximum file descriptor limit: $MAX_FD"
102 | fi
103 | else
104 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
105 | fi
106 | fi
107 |
108 | # For Darwin, add options to specify how the application appears in the dock
109 | if $darwin; then
110 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
111 | fi
112 |
113 | # For Cygwin, switch paths to Windows format before running java
114 | if $cygwin ; then
115 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
116 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
117 |
118 | # We build the pattern for arguments to be converted via cygpath
119 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
120 | SEP=""
121 | for dir in $ROOTDIRSRAW ; do
122 | ROOTDIRS="$ROOTDIRS$SEP$dir"
123 | SEP="|"
124 | done
125 | OURCYGPATTERN="(^($ROOTDIRS))"
126 | # Add a user-defined pattern to the cygpath arguments
127 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
128 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
129 | fi
130 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
131 | i=0
132 | for arg in "$@" ; do
133 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
134 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
135 |
136 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
137 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
138 | else
139 | eval `echo args$i`="\"$arg\""
140 | fi
141 | i=$((i+1))
142 | done
143 | case $i in
144 | (0) set -- ;;
145 | (1) set -- "$args0" ;;
146 | (2) set -- "$args0" "$args1" ;;
147 | (3) set -- "$args0" "$args1" "$args2" ;;
148 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
149 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
150 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
151 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
152 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
153 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
154 | esac
155 | fi
156 |
157 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
158 | function splitJvmOpts() {
159 | JVM_OPTS=("$@")
160 | }
161 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
162 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
163 |
164 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
165 |
--------------------------------------------------------------------------------
/MultidexSample/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/MultidexSample/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':app'
2 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # multidex-sample
2 | Sample app to demonstrate multidex
3 |
--------------------------------------------------------------------------------