java通过xml配置属性用iText包生成pdf文件

来源:互联网 发布:thinkphp视频网站源码 编辑:程序博客网 时间:2024/06/01 11:00

总共步骤为: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内容。
用第一步set进的对象中的map内容,替换返回的Str中的内容。

示例一种根据特定的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();}

到此处为止,pdf文件已经生成完成。

后续需要好好研究一下正则表达式的使用,已经拓展面向对象编程思想。


0 0