All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.base4j.util.setting.dialect.BasicSettingAbstract Maven / Gradle / Ivy

The newest version!
package com.base4j.util.setting.dialect;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Writer;
import java.net.URL;
import java.nio.charset.Charset;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Properties;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.base4j.util.BeanUtil;
import com.base4j.util.FileUtil;
import com.base4j.util.IoUtil;
import com.base4j.util.RegexUtil;
import com.base4j.util.StrUtil;
import com.base4j.util.UrlUtil;
import com.base4j.util.BeanUtil.ValueProvider;
import com.base4j.util.setting.AbstractAbsSetting;

/**
 * 分组设置工具类。 用于支持设置文件
* 1、支持变量,默认变量命名为 ${变量名},变量只能识别读入行的变量,例如第6行的变量在第三行无法读取 * 2、支持分组,分组为中括号括起来的内容,中括号以下的行都为此分组的内容,无分组相当于空字符分组
* 若某个key是name,加上分组后的键相当于group.name * 3、注释以#开头,但是空行和不带“=”的行也会被跳过,但是建议加# * 4、store方法不会保存注释内容,慎重使用 * * @author xxx */ public class BasicSettingAbstract extends AbstractAbsSetting { // private final static Log log = StaticLog.get(); private final static Logger log = LoggerFactory.getLogger(BasicSettingAbstract.class); final Map map = new ConcurrentHashMap(); /** * 默认字符集 */ public final static String DEFAULT_CHARSET = "utf8"; /** * 数组类型值默认分隔符 */ public final static String DEFAULT_DELIMITER = ","; /** * 注释符号(当有此符号在行首,表示此行为注释) */ private final static String COMMENT_FLAG_PRE = "#"; /** * 赋值分隔符(用于分隔键值对) */ private final static String ASSIGN_FLAG = "="; /** * 分组行识别的环绕标记 */ private final static char[] GROUP_SURROUND = {'[', ']'}; /** * 变量名称的正则 */ private String reg_var = "\\$\\{(.*?)\\}"; /** * 本设置对象的字符集 */ private Charset charset; /** * 是否使用变量 */ private boolean isUseVariable; /** * 设定文件的URL */ private URL settingUrl; private LinkedList groups = new LinkedList(); /** * 基本构造
* 需自定义初始化配置文件
* * @param charset 字符集 * @param isUseVariable 是否使用变量 */ public BasicSettingAbstract(Charset charset, boolean isUseVariable) { this.charset = charset; this.isUseVariable = isUseVariable; } /** * 构造,使用相对于Class文件根目录的相对路径 * * @param pathBaseClassLoader 相对路径(相对于当前项目的classes路径) * @param charset 字符集 * @param isUseVariable 是否使用变量 */ public BasicSettingAbstract(String pathBaseClassLoader, String charset, boolean isUseVariable) { if (null == pathBaseClassLoader) { pathBaseClassLoader = StrUtil.EMPTY; } final URL url = UrlUtil.getURL(pathBaseClassLoader); if (url == null) { throw new RuntimeException(StrUtil.format("Can not find Setting file: [{}]", pathBaseClassLoader)); } this.init(url, charset, isUseVariable); } /** * 构造 * * @param configFile 配置文件对象 * @param charset 字符集 * @param isUseVariable 是否使用变量 */ public BasicSettingAbstract(File configFile, String charset, boolean isUseVariable) { if (configFile == null) { throw new RuntimeException("Null Setting file!"); } final URL url = UrlUtil.getURL(configFile); if (url == null) { throw new RuntimeException(StrUtil.format("Can not find Setting file: [{}]", configFile.getAbsolutePath())); } this.init(url, charset, isUseVariable); } /** * 构造,相对于classes读取文件 * * @param path 相对路径 * @param clazz 基准类 * @param charset 字符集 * @param isUseVariable 是否使用变量 */ public BasicSettingAbstract(String path, Class clazz, String charset, boolean isUseVariable) { final URL url = UrlUtil.getURL(path, clazz); if (url == null) { throw new RuntimeException(StrUtil.format("Can not find Setting file: [{}]", path)); } this.init(url, charset, isUseVariable); } /** * 构造 * * @param url 设定文件的URL * @param charset 字符集 * @param isUseVariable 是否使用变量 */ public BasicSettingAbstract(URL url, String charset, boolean isUseVariable) { if (url == null) { throw new RuntimeException("Null url define!"); } this.init(url, charset, isUseVariable); } /** * 构造 * * @param pathBaseClassLoader 相对路径(相对于当前项目的classes路径) */ public BasicSettingAbstract(String pathBaseClassLoader) { this(pathBaseClassLoader, DEFAULT_CHARSET, false); } /*--------------------------公有方法 start-------------------------------*/ /** * 初始化设定文件 * * @param settingUrl 设定文件的URL * @param charset 字符集 * @param isUseVariable 是否使用变量 * @return 成功初始化与否 */ public boolean init(URL settingUrl, String charset, boolean isUseVariable) { if (settingUrl == null) { throw new RuntimeException("Null setting url or charset define!"); } try { this.charset = Charset.forName(charset); } catch (Exception e) { log.warn("User custom charset [{}] parse error, use default charset: [{}]", charset, DEFAULT_CHARSET); this.charset = Charset.forName(DEFAULT_CHARSET); } this.isUseVariable = isUseVariable; this.settingUrl = settingUrl; return this.load(settingUrl); } /** * 加载设置文件 * * @param settingUrl 配置文件URL * @return 加载是否成功 */ synchronized public boolean load(URL settingUrl) { if (settingUrl == null) { throw new RuntimeException("Null setting url define!"); } log.debug("Load setting file [{}]", settingUrl.getPath()); InputStream settingStream = null; try { settingStream = settingUrl.openStream(); load(settingStream, isUseVariable); } catch (IOException e) { log.error("Load setting error!", e); return false; } finally { IoUtil.close(settingStream); } return true; } /** * 重新加载配置文件 */ public void reload() { this.load(settingUrl); } /** * 加载设置文件。 此方法不会关闭流对象 * * @param settingStream 文件流 * @param isUseVariable 是否使用变量(替换配置文件值中含有的变量) * @return 加载成功与否 * @throws IOException */ public boolean load(InputStream settingStream, boolean isUseVariable) throws IOException { map.clear(); BufferedReader reader = null; try { reader = new BufferedReader(new InputStreamReader(settingStream, charset)); // 分组 String group = null; while (true) { String line = reader.readLine(); if (line == null) { break; } line = line.trim(); // 跳过注释行和空行 if (StrUtil.isBlank(line) || line.startsWith(COMMENT_FLAG_PRE)) { continue; } // 记录分组名 if (line.charAt(0) == GROUP_SURROUND[0] && line.charAt(line.length() - 1) == GROUP_SURROUND[1]) { group = line.substring(1, line.length() - 1).trim(); this.groups.add(group); continue; } String[] keyValue = line.split(ASSIGN_FLAG, 2); // 跳过不符合简直规范的行 if (keyValue.length < 2) { continue; } String key = keyValue[0].trim(); if (false == StrUtil.isBlank(group)) { key = group + StrUtil.DOT + key; } String value = keyValue[1].trim(); // 替换值中的所有变量变量(变量必须是此行之前定义的变量,否则无法找到) if (isUseVariable) { value = replaceVar(value); } map.put(key, value); } } finally { IoUtil.close(reader); } return true; } /** * 设置变量的正则
* 正则只能有一个group表示变量本身,剩余为字符 例如 \$\{(name)\}表示${name}变量名为name的一个变量表示 * * @param regex 正则 */ public void setVarRegex(String regex) { this.reg_var = regex; } /** * @return 获得设定文件的路径 */ public String getSettingPath() { return settingUrl.getPath(); } @Override public int size() { return map.size(); } @Override public String getStr(String key, String defaultValue) { final String value = map.get(key); if (StrUtil.isBlank(value)) { return defaultValue; } return value; } /** * 获得指定分组的键对应值 * * @param key 键 * @param group 分组 * @return 值 */ public String getByGroup(String key, String group) { return getStr(keyWithGroup(key, group)); } /** * 获得所有键值对 * * @return map */ public Map getMap() { return this.map; } /** * 获得指定分组的所有键值对 * * @param group 分组 * @return map */ public Map getMap(String group) { Map map2 = new HashMap(); for (String key : map.keySet()) { if (StrUtil.isNotBlank(key) && key.startsWith(group)) { map2.put(key, map.get(key)); } } return map2; } //--------------------------------------------------------------- Set /** * 设置值,无给定键创建之。设置后未持久化 * * @param key 键 * @param value 值 */ public void setSetting(String key, Object value) { map.put(key, value.toString()); } //--------------------------------------------------------------------------------- Functions /** * 持久化当前设置,会覆盖掉之前的设置
* 持久化会不会保留之前的分组 * * @param absolutePath 设置文件的绝对路径 */ public void store(String absolutePath) { Writer writer = null; try { writer = FileUtil.getWriter(absolutePath, charset, false); Set> entrySet = map.entrySet(); for (Entry entry : entrySet) { writer.write(entry.getKey() + ASSIGN_FLAG + entry.getValue()); } writer.close(); } catch (FileNotFoundException e) { throw new RuntimeException(StrUtil.format("Can not find file [{}]!", absolutePath), e); } catch (IOException e) { throw new RuntimeException("Store Setting error!", e); } finally { IoUtil.close(writer); } } /** * 存储当前设置,会覆盖掉以前的设置 * * @param path 相对路径 * @param clazz 相对的类 */ public void store(String path, Class clazz) { this.store(FileUtil.getAbsolutePath(path, clazz)); } /** * 将setting中的键值关系映射到对象中,原理是调用对象对应的set方法
* 只支持基本类型的转换 * * @param bean Bean * @return Bean */ public Object toBean(final String group, Object bean) { return BeanUtil.fill(bean, new ValueProvider() { @Override public Object value(String name) { final String value = getByGroup(name, group); if (null != value) { log.debug("Parse setting to object field [{}={}]", name, value); } return value; } }); } /** * 将setting中的键值关系映射到对象中,原理是调用对象对应的set方法
* 只支持基本类型的转换 * * @param bean Bean * @return Bean */ public Object toBean(Object bean) { return toBean(null, bean); } /** * 转换为Properties对象,原分组变为前缀 * * @return Properties对象 */ public Properties toProperties() { Properties properties = new Properties(); properties.putAll(map); return properties; } /** * @return 获得所有分组名 */ public LinkedList getGroups() { return this.groups; } public Set> entrySet() { return map.entrySet(); } @Override public String toString() { return map.toString(); } /*--------------------------Private Method start-------------------------------*/ /** * 替换给定值中的变量标识 * * @param value 值 * @return 替换后的字符串 */ private String replaceVar(String value) { // 找到所有变量标识 final Set vars = RegexUtil.findAll(reg_var, value, 0, new HashSet()); for (String var : vars) { // 查找变量名对应的值 String varValue = map.get(RegexUtil.get(reg_var, var, 1)); if (null != varValue) { // 替换标识 value = value.replace(var, varValue); } } return value; } /** * 组合Key和Group,组合后为group.key * * @param key * @param group * @return 组合后的KEY */ private static String keyWithGroup(String key, String group) { String keyWithGroup = key; if (!StrUtil.isBlank(group)) { keyWithGroup = group + "." + keyWithGroup; } return keyWithGroup; } /*--------------------------Private Method end-------------------------------*/ }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy