项目源代码可访问我的 github:https://github.com/Spacider/Gather-and-store
如果觉得好的话请给个 star 哦~
开发 IDE: IDEA 2018.03 JDK 1.8
开发环境: macOS 10.13.6 (如 windows 请对项目中部分路径进行改写)
数据库: Oracle 11g
上节完毕项目主体已经基本开发完成,下列进行一系列优化操作:
1.备份模块:
先编写一个接口:
/** * 备份模块 * 1. 客户端发送集合没有发送出去或连不上服务器需要备份 * 2. 服务端接收了集合对象写入数据库出错,对集合进行备份 */ public interface BackUp extends WossModel{ /** * 将集合保存到文件中 * @param coll 需要备份的集合 */ void storeEnvs(Collection<Environment> coll ); /** * 将集合从文件中读取 * @return */ Collection<Environment> loadEnvs(); }
具体实现:
通过对象流套文件流的形式把对象存入备份文件中:
@Override public void storeEnvs(Collection<Environment> coll) { File file = new File(path); FileOutputStream fos = null; ObjectOutputStream oos = null; try { if (!file.exists()) { file.createNewFile(); } fos = new FileOutputStream(file,true); oos = new ObjectOutputStream(fos); oos.writeObject(coll); oos.flush(); System.out.println("已经存入备份文件中 ,path:" +path); } catch (IOException e) { e.printStackTrace(); } finally { IOUtil.close(fos,oos); } }
从备份文件中取出相应对象!
@Override public Collection <Environment> loadEnvs() { FileInputStream fis = null; ObjectInputStream ois = null; Object backupObject = null; try { fis = new FileInputStream(path); ois = new ObjectInputStream(fis); backupObject = ois.readObject(); // System.out.println("已经从 path:" + path + "取出备份文件"); log.info("从 path:" + path + "取出备份文件中..."); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } return (Collection <Environment>) backupObject; }
2.日志模块
先编写一个接口叫 Log
,里边封装了日志的四种级别:
public interface Log extends WossModel{ /** * 输出 debug 级别的日志 * @param msg */ void debug(String msg); /** * 输出 info 级别的日志 * @param msg */ void info(String msg); /** * 输出 warn 级别的日志 * @param msg */ void warn(String msg); /** * 输出 error 级别的日志 * @param msg */ void error(String msg); }
日志模块的实现就是调用 log4j 的日志配置,然后做对应的输出:
private Logger LOGGER = Logger.getLogger(LogImpl.class); public LogImpl() { PropertyConfigurator.configure("/Users/wjh/Desktop/FirstProject/src/main/resources/log4j.properties"); } @Override public void debug(String msg) { LOGGER.debug(msg); } @Override public void info(String msg) { LOGGER.info(msg); } @Override public void warn(String msg) { LOGGER.warn(msg); } @Override public void error(String msg) { LOGGER.error(msg); } @Override public void init(Properties properties) { }
3.配置模块
首先配置一个接口为 WossModel
,里面有一个方法,有了这个方法之后当一个类如果实现了这个接口,我们就可以把 Properties
文件传给它,在 Properties
文件中我们又可以传入相应的参数,这样做可以灵活替换类中的可变参,让整个程序变得灵活起来!
/** * 初始化配置参数 */ public interface WossModel { void init(Properties properties); }
再定义一个接口为 Configuration
,在这个接口的实现中会通过反射去获取各个类(又可以说是一个模块)的对象,保证了对象的唯一性!这一部分又可以看做是一个注入过程,让对象的创建变得灵活,变得安全!
/** * 获取各个配置模块对象 * 对所有对象进行管理 */ public interface Configuration { /** * 获取采集模块对象 */ Gather getGather(); /** * 获取客户端模块对象 */ EnvClient getClient(); /** * 获取服务器端对象 */ EnvServer getServer(); /** * 获取入库模块对象 */ DBStore getDBStore(); /** * 获取日志模块对象 */ Log getLog(); /** * 获取备份模块对象 */ BackUp getBackUp(); }
最后一个接口是 ConfigurationAware
,这个接口的主要用途就是当遇到一个模块对象调用到另一个模块对象时间,就可以把刚才写的 Configuration
整个注入进去,有了这个 Configuration
对象就相当于可以获取到了其他所有配置模块的对象,这样你就可以灵活去获取到你需要的对象并且调用其内的方法了!
public interface ConfigurationAware { void SetConfiguration(Configuration conf); }
具体实施
讲了这么多,具体咋实施呢:
给你需要的类上接口!
例如:
客户端的类:
让接口实现 WossModel
接口,那么子实现就默认也实现了这个接口!
由于在客户端类中
public interface EnvClient extends WossModel { } public class EnvClientImpl implements EnvClient , ConfigurationAware { }
当继承了 WossModel
接口之后我们就可以实现其方法来接入 Properties
文件:
public void init(Properties properties) { port = Integer.parseInt(properties.getProperty("port")); }
这样我们就可以替换掉 port
属性,等会有写怎样去通过写入 XML 文件的形式读取到这个 port 属性!
最后是重头戏,实现 Configuration
类,编写 ConfigurationImpl
类:
按照如下形式去
<?xml version="1.0" encoding="UTF-8" ?> <EMS> <Log class="com.briup.util.Impl.LogImpl"> </Log> <gather class="com.briup.Client.Impl.GatherImpl"> <logFile>/Users/wjh/Desktop/FirstProject/src/radwtmp</logFile> <positionFile>/Users/wjh/Desktop/FirstProject/src/main/resources/FilePostion.properties</positionFile> </gather> <EnvClient class = "com.briup.Client.Impl.EnvClientImpl"> <host>127.0.0.1</host> <port>9999</port> <path>/Users/wjh/Desktop/FirstProject/src/BackUptmp</path> </EnvClient> <EnvServer class="com.briup.Server.Impl.EnvServerImpl"> <port>9999</port> </EnvServer> <BackUp class="com.briup.util.Impl.BackUpImpl"> <BackUppath>/Users/wjh/Desktop/FirstProject/src/BackUptmp</BackUppath> </BackUp> <DBStore class="com.briup.Server.Impl.DBStoreImpl"> </DBStore> </EMS>
以
<对象名 class="对象的全限定名"> <需要替换的属性名>属性值</需要替换的属性名> </对象名>
来进行编写
在 ConfigurationImpl
的构造器中,编写对 XML 文件进行解析的代码!
这里还是用之前用到的 Dom4j,先获取到根节点,然后获取最大的子节点,也就是对象名
fis = new FileInputStream("/Users/wjh/Desktop/FirstProject/src/main/resources/EMS.xml"); document = saxReader.read(fis); EMS = document.getRootElement(); List<Element> EMSlist = EMS.elements();
使用一个 map 集合来存储我们得到的对象:
// 使用 Map 集合来存放 模块名-对象 private Map<String,WossModel> ObjectMap = new HashMap <>();
再遍历对象名的时候,获取对象名的 class 属性,也就是全限定名,通过 Class.forName()
来获取到的对应的对象,然后存入 map 集合中,然后继续遍历下一层,碰到有需要可变参数的模块的时候把可变参数提取出来,交给 Properties
,等待对象成功创建以后通过其中的 init
方法把 Properties
对象传入,最后如果遇到了类继承了 ConfigurationAware
接口(也就是说这个类会调用其他类的对象),就调用 SetConfiguration
方法把 Configuration
对象传入!这样就可以在一个类中去调用其他类的对象了:
for (Element element : EMSlist){ //gather -- EnvClient -- EnvServer -- BackUp -- DBStore -- String elementName = element.getName(); String elementClass = element.attribute("class").getText(); WossModel obj = (WossModel) Class.forName(elementClass).newInstance(); if (obj instanceof ConfigurationAware){ ((ConfigurationAware) obj).SetConfiguration(this); } // 遍历子节点,为应该赋值的变量赋值 List<Element> ChildEMSList = element.elements(); Properties properties =new Properties(); for (Element element1 : ChildEMSList){ properties.setProperty(element1.getName(),element1.getText()); } // 调用其 init 方法,对变量进行赋值 obj.init(properties); ObjectMap.put(elementName,obj); }
当做完了这一切,对象的创建变得异常简单:
@Override public Gather getGather() { return (Gather) ObjectMap.get("gather"); } @Override public EnvClient getClient() { return (EnvClient) ObjectMap.get("EnvClient"); } @Override public EnvServer getServer() { return (EnvServer) ObjectMap.get("EnvServer"); } @Override public DBStore getDBStore() { return (DBStore) ObjectMap.get("DBStore"); } @Override public Log getLog() { return (Log) ObjectMap.get("Log"); } @Override public BackUp getBackUp() { return (BackUp) ObjectMap.get("BackUp"); }
在配置模块之后,就可以通过这样来创造一个新的对象:
ConfigurationImpl configuration = new ConfigurationImpl(); Gather gather = configuration.getGather();
而在继承了 ConfigurationAware
接口的类中,可以通过
private Configuration configuration; private Log logger; @Override public void SetConfiguration(Configuration conf) { this.configuration = conf; logger = configuration.getLog(); }
日志也变得如此简单:
logger.info("插入数据库成功:" + count + "数据"); logger.error("插入数据库失败");
项目的说明就此结束,项目的说明中的代码不一定完整,完整代码在 https://github.com/Spacider/Gather-and-store
这是一个练手项目,通过这个项目你可以对 java 基础有更深层次的了解,其中运用了注入,模块分割等常用的方法,使你之后对 spring 等框架的理解更深一个层次!
本人语言沟通能力尚缺,可能讲的地方会出问题,请指教!希望能和大家一起学习,一起进步!
个人网站:http://www.spacider.com/
CSDN:https://blog.csdn.net/qq_37163479
联系 QQ:729215049
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于