欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

详解五种方式让你在java中读取properties文件内容不再是难题

程序员文章站 2024-03-08 20:26:46
一、背景 最近,在项目开发的过程中,遇到需要在properties文件中定义一些自定义的变量,以供java程序动态的读取,修改变量,不再需要修改代码的问题。就借此机会把s...

一、背景

最近,在项目开发的过程中,遇到需要在properties文件中定义一些自定义的变量,以供java程序动态的读取,修改变量,不再需要修改代码的问题。就借此机会把spring+springmvc+mybatis整合开发的项目中通过java程序读取properties文件内容的方式进行了梳理和分析,先和大家共享。

二、项目环境介绍

  • spring 4.2.6.release
  • springmvc 4.2.6.release
  • mybatis 3.2.8
  • maven 3.3.9
  • jdk 1.7
  • idea 15.04

三、五种实现方式

方式1.通过context:property-placeholder加载配置文件jdbc.properties中的内容

<context:property-placeholder location="classpath:jdbc.properties" ignore-unresolvable="true"/>

上面的配置和下面配置等价,是对下面配置的简化

<bean id="propertyconfigurer" class="org.springframework.beans.factory.config.propertyplaceholderconfigurer">
  <property name="ignoreunresolvableplaceholders" value="true"/>
  <property name="locations">
   <list>
     <value>classpath:jdbc.properties</value>
   </list>
  </property>
</bean>

注意:这种方式下,如果你在spring-mvc.xml文件中有如下配置,则一定不能缺少下面的红色部分,关于它的作用以及原理.

<!-- 配置组件扫描,springmvc容器中只扫描controller注解 -->
<context:component-scan base-package="com.hafiz.www" use-default-filters="false">
  <context:include-filter type="annotation" expression="org.springframework.stereotype.controller"/>
</context:component-scan>

方式2.使用注解的方式注入,主要用在java代码中使用注解注入properties文件中相应的value值

<bean id="prop" class="org.springframework.beans.factory.config.propertiesfactorybean">
  <!-- 这里是propertiesfactorybean类,它也有个locations属性,也是接收一个数组,跟上面一样 -->
  <property name="locations">
    <array>
     <value>classpath:jdbc.properties</value>
    </array>
  </property>
</bean>

方式3.使用util:properties标签进行暴露properties文件中的内容

<util:properties id="propertiesreader" location="classpath:jdbc.properties"/>

注意:使用上面这行配置,需要在spring-dao.xml文件的头部声明以下红色的部分

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/xmlschema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:util="http://www.springframework.org/schema/util"
    xsi:schemalocation="http://www.springframework.org/schema/beans
    http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
    http://www.springframework.org/schema/context 
    http://www.springframework.org/schema/context/spring-context-3.2.xsd
    http://www.springframework.org/schema/util 
     http://www.springframework.org/schema/util/spring-util.xsd">

方式4.通过propertyplaceholderconfigurer在加载上下文的时候暴露properties到自定义子类的属性中以供程序中使用

<bean id="propertyconfigurer" class="com.hafiz.www.util.propertyconfigurer">
  <property name="ignoreunresolvableplaceholders" value="true"/>
  <property name="ignoreresourcenotfound" value="true"/>
  <property name="locations">
    <list>
     <value>classpath:jdbc.properties</value>
    </list>
  </property>
</bean>

自定义类propertyconfigurer的声明如下:

package com.hafiz.www.util;

import org.springframework.beans.beansexception;
import org.springframework.beans.factory.config.configurablelistablebeanfactory;
import org.springframework.beans.factory.config.propertyplaceholderconfigurer;

import java.util.properties;

/**
 * desc:properties配置文件读取类
 * created by hafiz.zhang on 2016/9/14.
 */
public class propertyconfigurer extends propertyplaceholderconfigurer {

  private properties props;    // 存取properties配置文件key-value结果

  @override
  protected void processproperties(configurablelistablebeanfactory beanfactorytoprocess, properties props)
              throws beansexception {
    super.processproperties(beanfactorytoprocess, props);
    this.props = props;
  }

  public string getproperty(string key){
    return this.props.getproperty(key);
  }

  public string getproperty(string key, string defaultvalue) {
    return this.props.getproperty(key, defaultvalue);
  }

  public object setproperty(string key, string value) {
    return this.props.setproperty(key, value);
  }
}

使用方式:在需要使用的类中使用@autowired注解注入即可。

方式5.自定义工具类propertyutil,并在该类的static静态代码块中读取properties文件内容保存在static属性中以供别的程序使用

package com.hafiz.www.util;

import org.slf4j.logger;
import org.slf4j.loggerfactory;

import java.io.*;
import java.util.properties;

/**
 * desc:properties文件获取工具类
 * created by hafiz.zhang on 2016/9/15.
 */
public class propertyutil {
  private static final logger logger = loggerfactory.getlogger(propertyutil.class);
  private static properties props;
  static{
    loadprops();
  }

  synchronized static private void loadprops(){
    logger.info("开始加载properties文件内容.......");
    props = new properties();
    inputstream in = null;
    try {
       <!--第一种,通过类加载器进行获取properties文件流-->
      in = propertyutil.class.getclassloader().getresourceasstream("jdbc.properties");
       <!--第二种,通过类进行获取properties文件流-->
      //in = propertyutil.class.getresourceasstream("/jdbc.properties");
      props.load(in);
    } catch (filenotfoundexception e) {
      logger.error("jdbc.properties文件未找到");
    } catch (ioexception e) {
      logger.error("出现ioexception");
    } finally {
      try {
        if(null != in) {
          in.close();
        }
      } catch (ioexception e) {
        logger.error("jdbc.properties文件流关闭出现异常");
      }
    }
    logger.info("加载properties文件内容完成...........");
    logger.info("properties文件内容:" + props);
  }

  public static string getproperty(string key){
    if(null == props) {
      loadprops();
    }
    return props.getproperty(key);
  }

  public static string getproperty(string key, string defaultvalue) {
    if(null == props) {
      loadprops();
    }
    return props.getproperty(key, defaultvalue);
  }
}

说明:这样的话,在该类被加载的时候,它就会自动读取指定位置的配置文件内容并保存到静态属性中,高效且方便,一次加载,可多次使用。

四、注意事项及建议

以上五种方式,前三种方式比较死板,而且如果你想在带有@controller注解的bean中使用,你需要在springmvc的配置文件spring-mvc.xml中进行声明,如果你想在带有@service、@respository等非@controller注解的bean中进行使用,你需要在spring的配置文件中spring.xml中进行声明。

我个人比较建议第四种和第五种配置方式,第五种为最好,它连工具类对象都不需要注入,直接调用静态方法进行获取,而且只一次加载,效率也高。而且前三种方式都不是很灵活,需要修改@value的键值。

五、测试验证是否可用

1.首先我们创建propertiesservice

package com.hafiz.www.service;

/**
 * desc:java程序获取properties文件内容的service
 * created by hafiz.zhang on 2016/9/16.
 */
public interface propertiesservice {

  /**
   * 第一种实现方式获取properties文件中指定key的value
   *
   * @return
   */
  string getproperybyfirstway();

  /**
   * 第二种实现方式获取properties文件中指定key的value
   *
   * @return
   */
  string getproperybysecondway();

  /**
   * 第三种实现方式获取properties文件中指定key的value
   *
   * @return
   */
  string getproperybythirdway();

  /**
   * 第四种实现方式获取properties文件中指定key的value
   *
   * @param key
   *
   * @return
   */
  string getproperybyfourthway(string key);

  /**
   * 第四种实现方式获取properties文件中指定key的value
   *
   * @param key
   *
   * @param defaultvalue
   *
   * @return
   */
  string getproperybyfourthway(string key, string defaultvalue);

  /**
   * 第五种实现方式获取properties文件中指定key的value
   *
   * @param key
   *
   * @return
   */
  string getproperybyfifthway(string key);

  /**
   * 第五种实现方式获取properties文件中指定key的value
   *
   * @param key
   *
   * @param defaultvalue
   *
   * @return
   */
  string getproperybyfifthway(string key, string defaultvalue);
}

2.创建实现类propertiesserviceimpl

package com.hafiz.www.service.impl;

import com.hafiz.www.service.propertiesservice;
import com.hafiz.www.util.propertyconfigurer;
import com.hafiz.www.util.propertyutil;
import org.springframework.beans.factory.annotation.autowired;
import org.springframework.beans.factory.annotation.value;
import org.springframework.stereotype.service;

/**
 * desc:java程序获取properties文件内容的service的实现类
 * created by hafiz.zhang on 2016/9/16.
 */
@service
public class propertiesserviceimpl implements propertiesservice {

  @value("${test}")
  private string testdatabyfirst;

  @value("#{prop.test}")
  private string testdatabysecond;

  @value("#{propertiesreader[test]}")
  private string testdatabythird;

  @autowired
  private propertyconfigurer pc;

  @override
  public string getproperybyfirstway() {
    return testdatabyfirst;
  }

  @override
  public string getproperybysecondway() {
    return testdatabysecond;
  }

  @override
  public string getproperybythirdway() {
    return testdatabythird;
  }

  @override
  public string getproperybyfourthway(string key) {
    return pc.getproperty(key);
  }

  @override
  public string getproperybyfourthway(string key, string defaultvalue) {
    return pc.getproperty(key, defaultvalue);
  }

  @override
  public string getproperybyfifthway(string key) {
    return propertyutil.getpropery(key);
  }

  @override
  public string getproperybyfifthway(string key, string defaultvalue) {
    return propertyutil.getproperty(key, defaultvalue);
  }
}

3.控制器类propertycontroller

package com.hafiz.www.controller;

import com.hafiz.www.service.propertiesservice;
import com.hafiz.www.util.propertyutil;
import org.springframework.beans.factory.annotation.autowired;
import org.springframework.stereotype.controller;
import org.springframework.web.bind.annotation.pathvariable;
import org.springframework.web.bind.annotation.requestmapping;
import org.springframework.web.bind.annotation.requestmethod;
import org.springframework.web.bind.annotation.responsebody;

/**
 * desc:properties测试控制器
 * created by hafiz.zhang on 2016/9/16.
 */
@controller
@requestmapping("/prop")
public class propertycontroller {
  @autowired
  private propertiesservice ps;

  @requestmapping(value = "/way/first", method = requestmethod.get)
  @responsebody
  public string getpropertybyfirstway(){
    return ps.getproperybyfirstway();
  }

  @requestmapping(value = "/way/second", method = requestmethod.get)
  @responsebody
  public string getpropertybysecondway(){
    return ps.getproperybysecondway();
  }

  @requestmapping(value = "/way/third", method = requestmethod.get)
  @responsebody
  public string getpropertybythirdway(){
    return ps.getproperybythirdway();
  }

  @requestmapping(value = "/way/fourth/{key}", method = requestmethod.get)
  @responsebody
  public string getpropertybyfourthway(@pathvariable("key") string key){
    return ps.getproperybyfourthway(key, "defaultvalue");
  }

  @requestmapping(value = "/way/fifth/{key}", method = requestmethod.get)
  @responsebody
  public string getpropertybyfifthway(@pathvariable("key") string key){
    return propertyutil.getproperty(key, "defaultvalue");
  }
}

4.jdbc.properties文件

jdbc.driver=com.mysql.jdbc.driver
jdbc.url=jdbc:mysql://192.168.1.196:3306/dev?useunicode=true&characterencoding=utf-8
jdbc.username=root
jdbc.password=123456
jdbc.maxactive=200
jdbc.minidle=5
jdbc.initialsize=1
jdbc.maxwait=60000
jdbc.timebetweenevictionrunsmillis=60000
jdbc.minevictableidletimemillis=300000
jdbc.validationquery=select 1 from t_user
jdbc.testwhileidle=true
jdbc.testonreturn=false
jdbc.poolpreparedstatements=true
jdbc.maxpoolpreparedstatementperconnectionsize=20
jdbc.filters=stat
#test data
test=com.hafiz.www

5.项目结果图

详解五种方式让你在java中读取properties文件内容不再是难题  

6.项目下载:demo http://xiazai.jb51.net/201612/yuanma/propertiesconfigurer_jb51.zip

7.测试结果

第一种方式

详解五种方式让你在java中读取properties文件内容不再是难题  

第二种方式

详解五种方式让你在java中读取properties文件内容不再是难题 

第三种方式

详解五种方式让你在java中读取properties文件内容不再是难题  

第四种方式

详解五种方式让你在java中读取properties文件内容不再是难题  

第五种方式

详解五种方式让你在java中读取properties文件内容不再是难题  

六、总结

通过本次的梳理和测试,我们理解了spring和springmvc的父子容器关系以及context:component-scan标签包扫描时最容易忽略的use-default-filters属性的作用以及原理。能够更好地定位和快速解决再遇到的问题。总之,棒棒哒~~~

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。