从 jar 文件访问文件和图像
Accessing files and images from jar files
我在尝试访问 jar 文件中的文件和图像时遇到了一些问题 运行。该程序在被制成 jar 文件之前按预期工作。我已经创建了一个使用 ClassLoader 的 Resources 文件夹,但在尝试 运行 它可以工作的 jar 文件时在命令行上仍然出现错误,但并非所有信息都显示。
类型必须是文件,以便 databaseReader 可以读取它。
错误信息
java.io.FileNotFoundException: file:\C:\Users\Nicholas\IdeaProjects\MirrorMe\out\artifacts\MirrorMe_jar\MirrorMe.jar!\GeoLite2-City.mmdb (The filename, directory name, or volume label syntax is incorrect)
at java.io.RandomAccessFile.open0(Native Method)
at java.io.RandomAccessFile.open(Unknown Source)
at java.io.RandomAccessFile.<init>(Unknown Source)
at com.maxmind.db.BufferHolder.<init>(BufferHolder.java:19)
at com.maxmind.db.Reader.<init>(Reader.java:116)
at com.maxmind.geoip2.DatabaseReader.<init>(DatabaseReader.java:35)
at com.maxmind.geoip2.DatabaseReader.<init>(DatabaseReader.java:23)
at com.maxmind.geoip2.DatabaseReader$Builder.build(DatabaseReader.java:129)
at sample.LocateMyCity.<init>(LocateMyCity.java:60)
at sample.WeatherToday.getPersonLocationId(WeatherToday.java:102)
at sample.WeatherToday.<init>(WeatherToday.java:126)
at sample.Main.start(Main.java:37)
at com.sun.javafx.application.LauncherImpl.lambda$launchApplication12(LauncherImpl.java:863)
at com.sun.javafx.application.PlatformImpl.lambda$runAndWait5(PlatformImpl.java:326)
at com.sun.javafx.application.PlatformImpl.lambda$null3(PlatformImpl.java:295)
at java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.application.PlatformImpl.lambda$runLater4(PlatformImpl.java:294)
at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95)
at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
at com.sun.glass.ui.win.WinApplication.lambda$null8(WinApplication.java:191)
at java.lang.Thread.run(Unknown Source)
完整代码
public class LocateMyCity {
private String myCityLocation;
private String country;
public String getCountry() {
return country;
}
public String getmyCityLocation(){
return myCityLocation;
}
public LocateMyCity() {
try {
ClassLoader classLoader = getClass().getClassLoader();
File database = new File(classLoader.getResource("GeoLite2-City.mmdb").getFile());
URL whatismyip = new URL("http://checkip.amazonaws.com");
BufferedReader in = new BufferedReader(new InputStreamReader(
whatismyip.openStream()));
String ip = in.readLine(); //you get the IP as a String
System.out.println(ip);
// This creates the DatabaseReader object, which should be reused across
// lookups.
DatabaseReader reader = new DatabaseReader.Builder(database).build();
InetAddress ipAddress = InetAddress.getByName(ip);
// Replace "city" with the appropriate method for your database, e.g.,
// "country".
CityResponse response = reader.city(ipAddress);
City city = response.getCity();
System.out.println(city.getName()); // 'Minneapolis'
this.myCityLocation = city.getName();
Country country = response.getCountry();
System.out.println(country.getIsoCode()); // 'GB'
this.country = country.getIsoCode();
System.out.println(country.getName()); // 'United Kindom'
}catch (Exception e){
e.printStackTrace();
System.out.println("Tracing IP E");
}
}
}
提前致谢。
当您的应用程序被打包为 jar 文件时,资源不再是文件,而是存档(jar 文件)中的元素。对于桌面应用程序,应用程序通常 运行 不会从存档中提取这些元素。
如果你的数据库需要一个实际的文件,而不仅仅是一个它可以读取的流(如果你需要写入它,情况尤其如此),那么你不能使用存档中的资源并且将有使用文件系统上的文件。
您可以轻松地从存档中提取资源并将其内容写入本地文件系统。您如何执行此操作的具体细节取决于您需要的功能。例如,如果您将写入数据库作为应用程序功能的一部分,并希望这些更改在下次应用程序 运行 时持续存在,您只想在第一个 运行(或者如果用户稍后删除了文件)。通常您会通过将文件放在用户的主目录中来完成此操作。例如,您可以使用:
Path appDirectory = Paths.get(System.getProperty("user.home"), ".application-name");
Path databaseFile = appDirectory.resolve("GeoList2-City.mmdb");
if (! Files.exists(databaseFile)) {
try {
// create the app directory if it doesn't already exist:
Files.createDirectories(appDirectory);
InputSteam defaultDatabase = getClass().getClassLoader().getResourceAsStream("GeoLite2-City.mmdb");
Files.copy(defaultDatabase, databaseFile);
} catch (IOException exc) {
// handle exception here, e.g. if application can run without db,
// set flag indicating it must run in non-db mode
// otherwise this is probably a fatal exception, show message and exit...
exc.printStackTrace();
}
}
// ...
DatabaseReader reader = new DatabaseReader.Builder(databaseFile.toFile()).build();
如果您每次 运行 应用程序都需要一个新数据库,您可能会复制到一个临时文件,在应用程序退出时删除该文件。
我在尝试访问 jar 文件中的文件和图像时遇到了一些问题 运行。该程序在被制成 jar 文件之前按预期工作。我已经创建了一个使用 ClassLoader 的 Resources 文件夹,但在尝试 运行 它可以工作的 jar 文件时在命令行上仍然出现错误,但并非所有信息都显示。
类型必须是文件,以便 databaseReader 可以读取它。
错误信息
java.io.FileNotFoundException: file:\C:\Users\Nicholas\IdeaProjects\MirrorMe\out\artifacts\MirrorMe_jar\MirrorMe.jar!\GeoLite2-City.mmdb (The filename, directory name, or volume label syntax is incorrect)
at java.io.RandomAccessFile.open0(Native Method)
at java.io.RandomAccessFile.open(Unknown Source)
at java.io.RandomAccessFile.<init>(Unknown Source)
at com.maxmind.db.BufferHolder.<init>(BufferHolder.java:19)
at com.maxmind.db.Reader.<init>(Reader.java:116)
at com.maxmind.geoip2.DatabaseReader.<init>(DatabaseReader.java:35)
at com.maxmind.geoip2.DatabaseReader.<init>(DatabaseReader.java:23)
at com.maxmind.geoip2.DatabaseReader$Builder.build(DatabaseReader.java:129)
at sample.LocateMyCity.<init>(LocateMyCity.java:60)
at sample.WeatherToday.getPersonLocationId(WeatherToday.java:102)
at sample.WeatherToday.<init>(WeatherToday.java:126)
at sample.Main.start(Main.java:37)
at com.sun.javafx.application.LauncherImpl.lambda$launchApplication12(LauncherImpl.java:863)
at com.sun.javafx.application.PlatformImpl.lambda$runAndWait5(PlatformImpl.java:326)
at com.sun.javafx.application.PlatformImpl.lambda$null3(PlatformImpl.java:295)
at java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.application.PlatformImpl.lambda$runLater4(PlatformImpl.java:294)
at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95)
at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
at com.sun.glass.ui.win.WinApplication.lambda$null8(WinApplication.java:191)
at java.lang.Thread.run(Unknown Source)
完整代码
public class LocateMyCity {
private String myCityLocation;
private String country;
public String getCountry() {
return country;
}
public String getmyCityLocation(){
return myCityLocation;
}
public LocateMyCity() {
try {
ClassLoader classLoader = getClass().getClassLoader();
File database = new File(classLoader.getResource("GeoLite2-City.mmdb").getFile());
URL whatismyip = new URL("http://checkip.amazonaws.com");
BufferedReader in = new BufferedReader(new InputStreamReader(
whatismyip.openStream()));
String ip = in.readLine(); //you get the IP as a String
System.out.println(ip);
// This creates the DatabaseReader object, which should be reused across
// lookups.
DatabaseReader reader = new DatabaseReader.Builder(database).build();
InetAddress ipAddress = InetAddress.getByName(ip);
// Replace "city" with the appropriate method for your database, e.g.,
// "country".
CityResponse response = reader.city(ipAddress);
City city = response.getCity();
System.out.println(city.getName()); // 'Minneapolis'
this.myCityLocation = city.getName();
Country country = response.getCountry();
System.out.println(country.getIsoCode()); // 'GB'
this.country = country.getIsoCode();
System.out.println(country.getName()); // 'United Kindom'
}catch (Exception e){
e.printStackTrace();
System.out.println("Tracing IP E");
}
}
}
当您的应用程序被打包为 jar 文件时,资源不再是文件,而是存档(jar 文件)中的元素。对于桌面应用程序,应用程序通常 运行 不会从存档中提取这些元素。
如果你的数据库需要一个实际的文件,而不仅仅是一个它可以读取的流(如果你需要写入它,情况尤其如此),那么你不能使用存档中的资源并且将有使用文件系统上的文件。
您可以轻松地从存档中提取资源并将其内容写入本地文件系统。您如何执行此操作的具体细节取决于您需要的功能。例如,如果您将写入数据库作为应用程序功能的一部分,并希望这些更改在下次应用程序 运行 时持续存在,您只想在第一个 运行(或者如果用户稍后删除了文件)。通常您会通过将文件放在用户的主目录中来完成此操作。例如,您可以使用:
Path appDirectory = Paths.get(System.getProperty("user.home"), ".application-name");
Path databaseFile = appDirectory.resolve("GeoList2-City.mmdb");
if (! Files.exists(databaseFile)) {
try {
// create the app directory if it doesn't already exist:
Files.createDirectories(appDirectory);
InputSteam defaultDatabase = getClass().getClassLoader().getResourceAsStream("GeoLite2-City.mmdb");
Files.copy(defaultDatabase, databaseFile);
} catch (IOException exc) {
// handle exception here, e.g. if application can run without db,
// set flag indicating it must run in non-db mode
// otherwise this is probably a fatal exception, show message and exit...
exc.printStackTrace();
}
}
// ...
DatabaseReader reader = new DatabaseReader.Builder(databaseFile.toFile()).build();
如果您每次 运行 应用程序都需要一个新数据库,您可能会复制到一个临时文件,在应用程序退出时删除该文件。