在web环境下有上下文环境的地方,可以轻松获取服务器上服务器路径的真实地址。
但是在没有上下文环境的地方获取服务器的地址可以通过类所在的路径来获取,通常采用以下办法:
需注意在不同中间件服务器下,获取的结果可能不同,这里需要自己的多次测试。
Class.getResource(String path)
path不以'/'开头时,默认是从此类所在的包下取资源;path以'/'开头时,则是从项目的ClassPath根下获取资源。在这里'/'表示ClassPath
JDK设置这样的规则,是很好理解的,path不以'/'开头时,我们就能获取与当前类所在的路径相同的资源文件,而以'/'开头时可以获取ClassPath根下任意路径的资源。
如下所示的例子:
1
2
3
4
5
6
7
8
|
public
class
Test
{
public
static
void
main
(
String
[
]
args
)
{
System
.
out
.
println
(
Test
.
class
.
getResource
(
""
)
)
;
System
.
out
.
println
(
Test
.
class
.
getResource
(
"/"
)
)
;
}
}
|
运行结果为:
file:/D:/work_space/java/bin/net/swiftlet/
file:/D:/work_space/java/bin/
Class.getClassLoader().getResource(String path)
path不能以'/'开头时,path是指类加载器的加载范围,在资源加载的过程中,使用的逐级向上委托的形式加载的,'/'表示Boot ClassLoader中的加载范围,因为这个类加载器是C++实现的,所以加载范围为null。如下所示:
1
2
3
4
5
6
7
8
|
public
class
Test
{
public
static
void
main
(
String
[
]
args
)
{
System
.
out
.
println
(
Test
.
class
.
getClassLoader
(
)
.
getResource
(
""
)
)
;
System
.
out
.
println
(
Test
.
class
.
getClassLoader
(
)
.
getResource
(
"/"
)
)
;
}
}
|
运行结果为:
file:/D:/work_space/java/bin/
null
从上面可以看出:
class.getResource("/") == class.getClassLoader().getResource("")
其实,Class.getResource和ClassLoader.getResource本质上是一样的,都是使用ClassLoader.getResource加载资源的。下面请看一下jdk的Class源码:
1
2
3
4
5
6
7
8
9
10
11
|
public
java
.
net
.
URL
getResource
(
String
name
)
{
name
=
resolveName
(
name
)
;
ClassLoader
cl
=
getClassLoader0
(
)
;
if
(
cl
==
null
)
{
// A system class.
return
ClassLoader
.
getSystemResource
(
name
)
;
}
return
cl
.
getResource
(
name
)
;
}
|
从上面就可以看才出来:Class.getResource和ClassLoader.getResource本质上是一样的,并且Class.getResource的实现时考虑类加载器为null的情况,直接使用更方便。至于为什么Class.getResource(String path)中path可以'/'开头,是因为在name = resolveName(name);进行了处理:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
private
String
resolveName
(
String
name
)
{
if
(
name
==
null
)
{
return
name
;
}
if
(
!
name
.
startsWith
(
"/"
)
)
{
Class
c
=
this
;
while
(
c
.
isArray
(
)
)
{
c
=
c
.
getComponentType
(
)
;
}
String
baseName
=
c
.
getName
(
)
;
int
index
=
baseName
.
lastIndexOf
(
'.'
)
;
if
(
index
!=
-
1
)
{
name
=
baseName
.
substring
(
0
,
index
)
.
replace
(
'.'
,
'/'
)
+
"/"
+
name
;
}
}
else
{
//如果是以"/"开头,则去掉
name
=
name
.
substring
(
1
)
;
}
return
name
;
}
|
在获取properties的路径之后,在进行加载,properties的方法
Properties类的重要方法Properties 类存在于胞 Java.util 中,该类继承自 Hashtable
1. getProperty ( String key) , 用指定的键在此属性列表中搜索属性。也就是通过参数 key ,得到 key 所对应的 value。
2. load ( InputStream inStream) ,从输入流中读取属性列表(键和元素对)。通过对指定的文件(比如说上面的 test.properties 文件)进行装载来获取该文件中的所有键 - 值对。以供 getProperty ( String key) 来搜索。
3. setProperty ( String key, String value) ,调用 Hashtable 的方法 put 。他通过调用基类的put方法来设置 键 - 值对。
4. store ( OutputStream out, String comments) , 以适合使用 load 方法加载到 Properties 表中的格式,将此 Properties 表中的属性列表(键和元素 对)写入输出流。与 load 方法相反,该方法将键 - 值对写入到指定的文件中去。
5. clear () ,清除所有装载的 键 - 值对。该方法在基类中提供。
关于文件读取及加载
web工程中, 文件在工程中的位置$app/WEB-INF/classes/db.properties在web有两种情形,一种是Servlet、一种是非Servlet
在Servlet中有2种方式:
1、
- InputStream in = this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");
- Properties prop = new Properties();
- prop.load(in);
- prop.getProperty("username");
- String path = this.getServletContext().getRealPath("/WEB-INF/classes/db.properties");
- FileInputStream in = new FileInputStream(path);
- Properties prop = new Properties();
- prop.load(in);
- prop.getProperty("username");
1.直接将文件装载到内存中
- InputStream in = Demo.class.getClassLoader().getResourceAsStream("db.properties");
- Properties prop = new Properties();
- prop.load(in);
- prop.getProperty("username");
这种方法的
弊端
:类装载器加载内容时,会先查找内存中是否已经存在相应的内容:
如果有就不再加载直接使用内存中的,所以此方法,第一次加载后,如果文件内容有变动,第二次加载后还是原来的内容,无法加载修改后的内容
下边的第二种方法取文件的绝对路径来加载不会有这种问题
2.得到文件的绝对路径再进行操作
- String path = Demo.class.getClassLoader().getResource("db.properties").getPath();
- FileInputStream in = new FileInputStream(path);
- Properties prop = new Properties();
- prop.load(in);
- prop.getProperty("username");
最后给一个例子:
- //根据key读取value
- public static String readValue(String filePath,String key) {
- Properties props = new Properties();
- try {
- InputStream in = new BufferedInputStream (new FileInputStream(filePath));
- props.load(in);
- String value = props.getProperty (key);
- System.out.println(key+value);
- return value;
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }
- }
- //读取properties的全部信息
- public static void readProperties(String filePath) {
- Properties props = new Properties();
- try {
- InputStream in = new BufferedInputStream (new FileInputStream(filePath));
- props.load(in);
- Enumeration en = props.propertyNames();
- while (en.hasMoreElements()) {
- String key = (String) en.nextElement();
- String Property = props.getProperty (key);
- System.out.println(key+Property);
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- //写入properties信息
- public static void writeProperties(String filePath,String parameterName,String parameterValue) {
- Properties prop = new Properties();
- try {
- InputStream fis = new FileInputStream(filePath);
- //从输入流中读取属性列表(键和元素对)
- prop.load(fis);
- //调用 Hashtable 的方法 put。使用 getProperty 方法提供并行性。
- //强制要求为属性的键和值使用字符串。返回值是 Hashtable 调用 put 的结果。
- OutputStream fos = new FileOutputStream(filePath);
- prop.setProperty(parameterName, parameterValue);
- //以适合使用 load 方法加载到 Properties 表中的格式,
- //将此 Properties 表中的属性列表(键和元素对)写入输出流
- prop.store(fos, "Update '" + parameterName + "' value");
- } catch (IOException e) {
- System.err.println("Visit "+filePath+" for updating "+parameterName+" value error");
- }
- }
- public static void main(String[] args) {
- readValue("info.properties","url");
- writeProperties("info.properties","age","21");
- readProperties("info.properties" );
- System.out.println("OK");
- }