困难在于它应该是跨平台的。Windows 2000,XP,Vista,OSX,Linux和其他Unix变体。我正在寻找可以针对所有平台完成此操作的代码片段,以及一种检测平台的方法。
现在,你应该知道的错误4787931是user.home
不能正常工作,所以请不要为我提供教科书的答案,我可以在手册中找到这些我自己。
困难在于它应该是跨平台的。Windows 2000,XP,Vista,OSX,Linux和其他Unix变体。我正在寻找可以针对所有平台完成此操作的代码片段,以及一种检测平台的方法。
现在,你应该知道的错误4787931是user.home
不能正常工作,所以请不要为我提供教科书的答案,我可以在手册中找到这些我自己。
Answers:
实际上,对于Java 8,正确的方法是使用:
System.getProperty("user.home");
错误JDK-6519127已得到修复,发行说明的 “ JDK 8和JDK 7之间的不兼容性”部分指出:
区域:核心库/ java.lang
概要
用于确定Windows上用户主目录的步骤已更改为遵循Microsoft建议的方法。在Windows的较早版本上或将注册表设置或环境变量设置为其他目录的情况下,此更改可能可见。不兼容的性质
behavioral RFE 6519127
尽管问题很老,我还是留给以后参考。
System.getProperty("user.home");
参见JavaDoc。
对于Windows,HOME目录的概念似乎有点含糊。如果环境变量(HOMEDRIVE / HOMEPATH / USERPROFILE)不够用,您可能不得不通过JNI或JNA使用本机函数。SHGetFolderPath允许您检索特殊文件夹,例如“ 我的文档”(CSIDL_PERSONAL)或“ 本地设置\应用程序数据”(CSIDL_LOCAL_APPDATA)。
样本JNA代码:
public class PrintAppDataDir {
public static void main(String[] args) {
if (com.sun.jna.Platform.isWindows()) {
HWND hwndOwner = null;
int nFolder = Shell32.CSIDL_LOCAL_APPDATA;
HANDLE hToken = null;
int dwFlags = Shell32.SHGFP_TYPE_CURRENT;
char[] pszPath = new char[Shell32.MAX_PATH];
int hResult = Shell32.INSTANCE.SHGetFolderPath(hwndOwner, nFolder,
hToken, dwFlags, pszPath);
if (Shell32.S_OK == hResult) {
String path = new String(pszPath);
int len = path.indexOf('\0');
path = path.substring(0, len);
System.out.println(path);
} else {
System.err.println("Error: " + hResult);
}
}
}
private static Map<String, Object> OPTIONS = new HashMap<String, Object>();
static {
OPTIONS.put(Library.OPTION_TYPE_MAPPER, W32APITypeMapper.UNICODE);
OPTIONS.put(Library.OPTION_FUNCTION_MAPPER,
W32APIFunctionMapper.UNICODE);
}
static class HANDLE extends PointerType implements NativeMapped {
}
static class HWND extends HANDLE {
}
static interface Shell32 extends Library {
public static final int MAX_PATH = 260;
public static final int CSIDL_LOCAL_APPDATA = 0x001c;
public static final int SHGFP_TYPE_CURRENT = 0;
public static final int SHGFP_TYPE_DEFAULT = 1;
public static final int S_OK = 0;
static Shell32 INSTANCE = (Shell32) Native.loadLibrary("shell32",
Shell32.class, OPTIONS);
/**
* see http://msdn.microsoft.com/en-us/library/bb762181(VS.85).aspx
*
* HRESULT SHGetFolderPath( HWND hwndOwner, int nFolder, HANDLE hToken,
* DWORD dwFlags, LPTSTR pszPath);
*/
public int SHGetFolderPath(HWND hwndOwner, int nFolder, HANDLE hToken,
int dwFlags, char[] pszPath);
}
}
其他人已经回答了我面前的问题,但是一个打印所有可用属性的有用程序是:
for (Map.Entry<?,?> e : System.getProperties().entrySet()) {
System.out.println(String.format("%s = %s", e.getKey(), e.getValue()));
}
在搜索Scala版本时,我只能找到上面的McDowell的JNA代码。我在这里包括我的Scala端口,因为目前没有任何合适的端口。
import com.sun.jna.platform.win32._
object jna {
def getHome: java.io.File = {
if (!com.sun.jna.Platform.isWindows()) {
new java.io.File(System.getProperty("user.home"))
}
else {
val pszPath: Array[Char] = new Array[Char](WinDef.MAX_PATH)
new java.io.File(Shell32.INSTANCE.SHGetSpecialFolderPath(null, pszPath, ShlObj.CSIDL_MYDOCUMENTS, false) match {
case true => new String(pszPath.takeWhile(c => c != '\0'))
case _ => System.getProperty("user.home")
})
}
}
}
与Java版本一样,您需要添加 Java Native Access(包括两个jar文件)添加到引用的库中。
很高兴看到JNA现在比发布原始代码时更容易做到这一点。
如果您想要在Windows上运行良好的工具,则有一个名为WinFoldersJava的软件包,该软件包包装了本地调用以获取Windows上的“特殊”目录。我们经常使用它,并且效果很好。