总共步骤为:1、提供一个需要替换参数传入类;2、解析xml文件,替换xml文件中的特定参数(写死格式);3、解析xml文件中的pdf文件布局(包括表、段落、文字、换行等);
4、生成pdf文件。
第一步:参数可以用map存放,根据相应的key可以取到对应的值,如果值为List,则可用来替换表格中的参数,根据List大小扩展表格的行数。
第二步:解析xml文件,可以直接用dom4j进行解析标准格式的xml文件。此次先用文件流进行解析吧,方便将拿到的参数进行字符串替换操作。解析代码如下:
/** * 逐行读取传入的文件内容 * @param filepath 文件绝对路径和文件名 * @return 存放文件全部内容的String * @throws Exception */ public static String readFile(String filepath) throws Exception { InputStreamReader read = null; BufferedReader bufferedReader = null; FileInputStream fileInput = null; try { fileInput = new FileInputStream(filepath); read = new InputStreamReader(fileInput, "GBK");//读取文件流内容 bufferedReader = new BufferedReader(read); String lineTxt = null; StringBuffer buffer = new StringBuffer(); while ((lineTxt = bufferedReader.readLine()) != null) { buffer.append(lineTxt + "\r\n"); } String str1 = buffer.toString(); return str1; }
<span style="white-space:pre"> </span>finally { if (bufferedReader != null) bufferedReader.close(); if (read != null) read.close(); if(fileInput != null) fileInput.close(); } }将文件内容按节点进行拆分为文件头与文件体,文件头存放字体配置文件头、文件页的信息,文件体存放具体需要展示的pdf内容。
示例一种根据特定的xml格式设定的表达式替换方法,代码如下:
/** * 找到始终的表达式,并进行替换。 表达式的格式为: ${status.index -----} * @param str * @param statusName * @param index * @return * @throws Exception */ private String findExpressAndReplace(String str, String statusName, int index) throws Exception { if ((statusName == null) || (statusName.trim().equals(""))) return str; StringBuffer sb = new StringBuffer(); // Pattern pattern = Pattern.compile("\\$\\{[\\w\\W]*?" + statusName + "\\.index[\\w\\W]*?\\}"); Pattern pattern = Pattern.compile("\\$\\{" + statusName + "\\.index[\\w\\W]*?\\}"); Matcher m = pattern.matcher(str); while (m.find()) { String sValue = m.group(0); sValue = sValue.substring(2, sValue.length() - 1); sValue = sValue.replace(statusName + ".index", index+""); sValue = runJsExpression(sValue); m.appendReplacement(sb, sValue); } m.appendTail(sb); return sb.toString(); } private String runJsExpression(String expression) throws Exception { ScriptEngineManager manager = new ScriptEngineManager(); ScriptEngine engine = manager.getEngineByName("JavaScript"); System.out.println("表达式计算结果为:"+engine.eval(expression).toString()); return engine.eval(expression).toString(); }2016/05/05 就写到这了。
2016/05/06 开写:
熟练运用正则表达式能节省很多代码。正则表达式一般包括两种引擎,一种传统NFA,一种DFA。java语言使用的是NFA引擎,使用表达式去匹配内容,当遇到第一个符合条件的值时,接收第一个匹配项。而DFA引擎则是用内容去匹配表达式。若需要匹配所有内容则需要用到Pattern与Matcher,用Matcher.find()读取匹配的下一个字符序列。
第三步:使用jdom解析xml文件,用模型对象存储各个pdf对象的值。
此步骤需要有面向对象编程思想,首先需要一个DocModel存放包含全局参数的全局对象GlobalModel,以及包含内容参数的内容体对象BodyModel。
其次BodyModel向下可以包含pdf的各个组建对象,包括单行(p)、换页(np)、图片(img)、表格(table)、直线(line)、换行(br)。各个元组的内容append到BodyModel对象中。
第四步:创建各个对象元组的具体内容,加入到Document,生成pdf文件。
主要调用iText包里的方法set配置参数,生成pdf内容。代码如下:
/** * 根据model对象的内容,开始写入pdf文件 * @param model 存放文件内容文件对象 * @param os 文件输出流 * @throws Exception */ public void create(DocModel model, OutputStream os) throws Exception { this.docModel = model; PageModel pageModel = this.docModel.getGlobalModel().getPage(); Document document = null; try { if ((pageModel.getMarginLeft() < 0) && (pageModel.getMarginRight() < 0) && (pageModel.getMarginTop() < 0) && (pageModel.getMarginBottom() < 0)) document = new Document(PageSize.A4); //如果page参数设置不完整,则默认使用A4大小 else { document = new Document(PageSize.A4, pageModel.getMarginLeft(), pageModel.getMarginRight(), pageModel.getMarginTop(), pageModel.getMarginBottom()); } this.pdfWriter = PdfWriter.getInstance(document, os); document.setFooter(createHeaderFooter()); document.open(); createContent(document, this.docModel.getBodyModel().getCells()); } catch (Exception e) { e.printStackTrace(); throw e; } finally { if (document != null) document.close(); if (this.pdfWriter != null) this.pdfWriter.close(); } } /** * 添加各元组实例对象到Document中 * @param document * @param cells * @throws Exception */ private void createContent(Document document, List<CellModel> cells) throws Exception { for (CellModel cell : cells) if ((cell instanceof PModel)) //CellModel为所有模型均须实现的公共接口,可以用 instanceof 判断具体内容具体是被哪个接口实现类实例化的 document.add((Element) new PCreator(this.docModel.getGlobalModel(), this.pdfWriter).create(cell)); else if ((cell instanceof LineModel)) { new LineCreator(this.docModel.getGlobalModel(), this.pdfWriter,document, 10.0F).create(cell); } else if ((cell instanceof ImgModel)) document.add((Element) new ImgCreator(this.docModel.getGlobalModel(), this.pdfWriter).create(cell)); else if ((cell instanceof TableModel)) document.add((Element) new TableCreator(this.docModel.getGlobalModel(), this.pdfWriter, document).create(cell)); else if ((cell instanceof BrModel)) document.add((Element) new BrCreator(this.docModel.getGlobalModel(), this.pdfWriter).create(cell)); else if ((cell instanceof NpModel)) document.newPage(); }
后续需要好好研究一下正则表达式的使用,已经拓展面向对象编程思想。