背景:在Eclipse中运行的时候,一切正常。把Java工程导成了可运行Java文件,通过java -jar XX.jar命令运行的时候,会报异常说配置文件路径找不到。
解决:
虽然不太确定,但是web工程的配置文件的路径查找和可运行Jar文件的路径查找方式不同。下面描述的是在可运行Jar文件中的配置文件的查找方法。
1. 我的工程结构如下:
注:
a. EnnDataDownloader.ini是工程的一下配置信息。log4j.properties是log4j的配置文件。
b. 请注意这两个文件在工程的根目录,不是在src下。
2. 读取EnnDataDownloader.ini,如下:
package com.ecity.enn.sap.download.util;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Properties;
public class ConfigUtil {
private static final String INI_FILE_NAME = "/EnnDataDownloader.ini";
private static HashMap<String, String> propertyMap = null;
static {
propertyMap = new HashMap<String, String>();
FileInputStream inStream = null;
try {
String path = URLUtil.getProjectHomePath() + INI_FILE_NAME;
inStream = new FileInputStream(path);
LogUtil.info("Loaded init file.");
} catch (FileNotFoundException e) {
LogUtil.error(e);
}
Properties properties = new Properties();
try {
properties.load(inStream);
} catch (IOException e) {
LogUtil.error(e);
}
@SuppressWarnings("unchecked")
Enumeration<String> keys = (Enumeration<String>) properties.propertyNames();
while (keys.hasMoreElements()) {
String key = keys.nextElement();
propertyMap.put(key.toLowerCase(), properties.getProperty(key));
}
}
public static String get(String key) {
return propertyMap.get(key.toLowerCase());
}
public static int getInt(String key) {
String valueStr = propertyMap.get(key.toLowerCase());
int value = 0;
try {
value = Integer.valueOf(valueStr);
} catch (Exception e) {
LogUtil.error(e);
value = 0;
}
return value;
}
public static void set(String key, String value) {
if (get(key) == null) {
propertyMap.put(key, value);
}
}
}
3. 读取log4j.properties。参考LogUtil.java和log4j.properties。如下:
package com.ecity.enn.sap.download.util;
import org.apache.log4j.Logger;
import org.apache.log4j.PropertyConfigurator;
public class LogUtil {
private static final Logger logger;
private static final String TAG = "EnnDataDownloader";
private static final String INI_FILE_NAME = "/log4j.properties";
static {
String path = URLUtil.getProjectHomePath() + INI_FILE_NAME;
PropertyConfigurator.configure(path);
logger = Logger.getLogger(TAG);
}
public static void debug(Object message) {
logger.info(message);
}
public static void debug(String message, Throwable e) {
logger.info(message, e);
}
public static void info(String message) {
logger.info(message);
}
public static void info(String message, Throwable e) {
logger.info(message, e);
}
public static void error(String message) {
logger.error(message);
}
public static void warn(Object message, Throwable e) {
logger.warn(message, e);
}
public static void warn(Throwable e) {
logger.warn("", e);
}
public static void error(Object message, Throwable e) {
logger.error(message, e);
}
public static void error(Throwable e) {
logger.error("", e);
}
}
log4j.rootLogger=INFO, ServerDailyRollingFile, stdout
log4j.appender.ServerDailyRollingFile=org.apache.log4j.DailyRollingFileAppender
log4j.appender.ServerDailyRollingFile.DatePattern='.'yyyy-MM-dd
log4j.appender.ServerDailyRollingFile.File=EnnDataDownloader.log
log4j.appender.ServerDailyRollingFile.layout=org.apache.log4j.PatternLayout
log4j.appender.ServerDailyRollingFile.layout.ConversionPattern=%d - %m%n
log4j.appender.ServerDailyRollingFile.Append=true
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} %p [%c] %m%n
注:两个java文件中,都用到了URLUtil.getProjectHomePath()。实现如下:
/**
* Used to get locating directory of a runnable jar file. Not for web project.
* @return Directory of the running file.
*/
public static String getProjectHomePath() {
return System.getProperty("user.dir");
}
4. 运行时的文件结构如下:
注:
a. start.bat内容是一个cmd命令,来启动控制台。方便直接运行run_jar.bat。
b. run_jar.bat内容是一个java -jar EnnDataDownloader.jar命令,来运行导出的可运行jar包。
c. 请注意ConfigUtil和LogUtil是如何获取文件路径的。getProjectHomePath方法是关键。
d. 用上面log4j.properties的配置,log文件也生成在同一个目录。