Using bash, how do you make a classpath out of all files in a directory?
New Answer
(October 2012)
There's no need to manually build the classpath list. Java supports a convenient wildcard syntax for directories containing jar files.
java -cp "$LIB/*"
(Notice that the *
is inside the quotes.)
Explanation from man java
:
As a special convenience, a class path element containing a basename of * is considered equivalent to specifying a list of all the files in the directory with the extension
.jar
or.JAR
(a java program cannot tell the difference between the two invocations).For example, if directory foo contains
a.jar
andb.JAR
, then the class path elementfoo/*
is expanded to aA.jar:b.JAR
, except that the order of jar files is unspecified. All jar files in the specified directory, even hidden ones, are included in the list. A classpath entry consisting simply of*
expands to a list of all the jar files in the current directory. TheCLASSPATH
environment variable, where defined, will be similarly expanded. Any classpath wildcard expansion occurs before the Java virtual machine is started — no Java program will ever see unexpanded wildcards except by querying the environment.
Old Answer
Good
Simple but not perfect solution:
CLASSPATH=$(echo "$LIB"/*.jar | tr ' ' ':')
There's a slight flaw in that this will not handle file names with spaces correctly. If that matters try this slightly more complicated version:
Better
CLASSPATH=$(find "$LIB" -name '*.jar' -printf '%p:' | sed 's/:$//')
This only works if your find command supports -printf
(as GNU find
does).
If you don't have GNU find
, as on Mac OS X, you can use xargs
instead:
CLASSPATH=$(find "." -name '*.jar' | xargs echo | tr ' ' ':')
Best?
Another (weirder) way to do it is to change the field separator variable $IFS
. This is very strange-looking but will behave well with all file names and uses only shell built-ins.
CLASSPATH=$(JARS=("$LIB"/*.jar); IFS=:; echo "${JARS[*]}")
Explanation:
-
JARS
is set to an array of file names. -
IFS
is changed to:
. - The array is echoed, and
$IFS
is used as the separator between array entries. Meaning the file names are printed with colons between them.
All of this is done in a sub-shell so the change to $IFS
isn't permanent (which would be baaaad).
for i in $LIB/*.jar; do
CLASSPATH=$CLASSPATH:$i
done
CLASSPATH=`echo $CLASSPATH | cut -c2-`
Here's another variation:
printf -v CLASSPATH "$LIB/%s:" *.jar; CLASSPATH=${CLASSPATH%:}
printf -v
is somewhat like sprintf
. The brace expansion removes the extra colon from the end.