Java中根據模板導出數據到word的解決方案

火星人 @ 2014-03-09 , reply:0


  Java中根據模板導出數據到word的解決方案我們需求如下:給你一個模板,裡面有一個表格,標題已經給好,程序主要就是完成把數據填寫到word中並提供給用戶下載.

  網上找了很久,發現主要兩種開源包:POI和Jacob,Jacob首先被否決掉了,運行在windows平台上.Excel導入導出我就是用的POI,但是POI中的word操作實在不行,讀取還可以,寫入數據遠不能滿足項目的需要.後面嘗試諸如生成PDF然後轉Word,生成XML轉Word,生成Html轉word,生成rtf轉word.這些都是可行的辦法,但是網上開源包的功能有限,iText對PDF操作是很強,但是對PDF內容的解析不行.

  這個事情拖了四天,我看文檔的時候想到mht文件,於是我把Word模板導出成mht文件,然後查看其源碼,其實就是html代碼,我想這樣可以用velocity填寫數據,然後直接導出成doc格式的文檔,後來這樣基本滿足我們項目的需要.

  #foreach($bean in $beanList)

  <tr style=3D'mso-yfti-irow:1;mso-yfti-lastrow:yes;page-break-inside:avoid;

  height:22.7pt'>

  <td width=3D"6%" style=3D'width:6.52%;border:solid windowtext 1.0pt;borde=

  r-top:

  none;mso-border-top-alt:solid windowtext .5pt;mso-border-alt:solid window=

  text .5pt;

  padding:0cm 5.4pt 0cm 5.4pt;height:22.7pt'>

  <p class=3DMsoNormal align=3Dcenter style=3D'text-align:center'><span style=3D'mso-bidi-font-size:10.5pt;font-family:SimSun'>${bean.seqNo}<o:p></o:p></span=

  ></p>

  </td>

  <td width=3D"10%" style=3D'width:10.12%;border-top:none;border-left:none;

  border-bottom:solid windowtext 1.0pt;border-right:solid windowtext 1.0pt;

  mso-border-top-alt:solid windowtext .5pt;mso-border-left-alt:solid window=

  text .5pt;

  mso-border-alt:solid windowtext .5pt;padding:0cm 5.4pt 0cm 5.4pt;height:2=

  2.7pt'>

  <p class=3DMsoNormal align=3Dcenter style=3D'text-align:center'><span

  style=3D'mso-bidi-font-size:10.5pt;font-family:SimSun'>${bean.name}<span

  lang=3DEN-US><o:p></o:p></span></span></p>

  </td>

  </tr>

  #end

  上面是用mht源碼改寫成vm文件的部分代碼,就和生成一般的文本文件一摸一樣.這兒如果直接把漢字、特殊字元什麼的傳給模板(當然英文不會),會出現亂碼,導致根據模板導出的word文件這個都是亂碼,關於亂碼的詳細解釋可以參考:http://blog.csdn.net/myyate/archive/2008/04/08/2260234.aspx ; 也就是說導出時需要把帶漢字的字元串都進行轉碼,這兒給出一些方法:

  /**

  * 把給定的str轉換為10進位表示的unicode,格式為:姨

  * 目前只是用於mht模板的轉碼

  * @param str

  * @return

  */

  public static String encode2HtmlUnicode(String str) {

  if(str == null) return "";

  StringBuilder sb = new StringBuilder(str.length() * 2);

  for (int i = 0; i < str.length(); i ) {

  sb.append(encode2HtmlUnicode(str.charAt(i)));

  }

  return sb.toString();

  }

  public static String encode2HtmlUnicode(char character) {

  if (character > 255) {

  return "&#" (character & 0xffff) ";";

  } else {

  return String.valueOf(character);

  }

  }

  public static String encode2HtmlUnicode(Character character) {

  if(character == null) return null;

  return encode2HtmlUnicode(character.charValue());

  }

  public static void encode2HtmlUnicode(String[] value) {

  if(value == null || value.length < 1) return;

  for(int i = 0; i < value.length; i ) {

  value[i] = encode2HtmlUnicode(value[i]);

  }

  }

  上面的這些方法在項目中可以通過遞歸來對bean進行轉碼,當然我在項目中只是針對String和Character兩種數據類型進行轉碼.

  static Object encodeStringAndCharacter(Object value) {

  if(value instanceof String) {

  return StringUtils.replaceNullString(EncodeUtils.encode2HtmlUnicode((String) value));

  } else if(value instanceof Character) {

  return StringUtils.replaceNullString(EncodeUtils.encode2HtmlUnicode((Character) value));

  }

  return null;

  }

  static void encodeExceptStringAndCharacter(Object value) throws Exception {

  if(value instanceof String[]) {

  EncodeUtils.encode2HtmlUnicode((String[]) value);

  } else if(value instanceof List) {

  encode2HtmlUnicode((List) value);

  } else if(value instanceof Form) {

  encode2HtmlUnicode((Form) value);

  } else if(value instanceof Form[]) {

  encode2HtmlUnicode((Form[]) value);

  }

  }

  @SuppressWarnings("unchecked")

  static void encode2HtmlUnicode(List value) throws Exception {

  if(value == null || value.size() < 1) return;

  for(int i = 0; i < value.size(); i ) {

  Object ele = value.get(i);

  Object result = encodeStringAndCharacter(ele);

  if(result != null) {

  value.set(i, result);

  } else {

  encodeExceptStringAndCharacter(ele);

  }

  }

  }

  static void encode2HtmlUnicode(Form[] value) throws Exception {

  if(value == null) return;

  for(int i = 0; i < value.length; i ) {

  encode2HtmlUnicode(value[i]);

  }

  }

  static void encode2HtmlUnicode(Form value) throws Exception {

  if(value == null) return;

  PropertyDescriptor[] pds = PropertyUtils.getPropertyDescriptors(value.getClass());

  if(pds == null) return;

  for(int i = 0; i < pds.length; i ) {

  PropertyDescriptor pd = pds[i];

  Object fieldValue = PropertyUtils.getProperty(value, pd.getName());

  Class fieldType = pd.getPropertyType();

  if(String.class.isAssignableFrom(fieldType)) {

  PropertyUtils.setProperty(value,

  pd.getName(),

  StringUtils.replaceNullString(EncodeUtils.encode2HtmlUnicode((String) fieldValue)));

  } else if(List.class.isAssignableFrom(fieldType)) {

  encode2HtmlUnicode((List) fieldValue);

  } else if(Form.class.isAssignableFrom(fieldType)) {

  encode2HtmlUnicode((Form) fieldValue);

  } else if(Form[].class.isAssignableFrom(fieldType)) {

  encode2HtmlUnicode((Form[]) fieldValue);

  }

  }

  }

  上面的Form類型不需要關心,就是一個Bean介面.這樣在傳到模板之前進行上述轉碼,就基本上可以避免亂碼問題.

  導出結束后,只需要把OutputStream導出到*.doc文件中即可,下載的話只要設置:

  response.setHeader("Content-disposition", "attachment; filename=" URLEncoder.encode("***.doc", DEFAULT_WEB_ENCODING));

  項目還有一個需求,對於一些飛機零件或者領導名字要求是圖片顯示.這個不難,但模板做起比較繁瑣,mht模板和難控制,一般都是通過word另存的,裡面漢字有時ascii格式,需要顯示圖片的地方都事先放一個圖片,然後再在mht代碼中替換一下:

  #if($manager)

  <!--[if gte vml =

  1]><v:shape

  id=3D"_x0000_i1026" type=3D"#_x0000_t75" style=3D'width:96pt;height:96pt'>

  <v:imagedata src=3D"file6086.files/image003.gif" o:title=3D"5080"/>

  </v:shape><![endif]--><![if !vml]><img width=3D128 height=3D128

  src=3D"file6086.files/image003.gif" v:shapes=3D"_x0000_i1026"><![endif]>

  #end

  ......

  #if($manager)

  ------=_NextPart_01C8E27B.AE1C3C80

  Content-Location: file:///C:/C071D706/file6086.files/image003.gif

  Content-Transfer-Encoding: base64

  Content-Type: image/gif

  ${manager}

  #end

  在mht文件中,圖片都是以base64碼來存儲的,在存取之前,把二進位的圖片進行base64編碼:

  static BASE64Encoder base64Encoder = new BASE64Encoder();

  /**

  * 把給定的二進位流變成base64碼

  * @param in

  * @return

  * @throws IOException

  */

  public static synchronized String base64Encode(InputStream in) throws IOException {

  byte[] img = new byte[in.available()];

  in.read(img);

  return base64Encode(img);

  }

  public static synchronized String base64Encode(String in) throws IOException {

  return base64Encode(in.getBytes());

  }

  public static synchronized String base64Encode(byte[] in) throws IOException {

  return base64Encoder.encode(in);

  }

  轉換成String以後在傳到模板文件中顯示則可.關於任何格式、布局等等等的調整你可以事先在word里先全部調整好再另存.導出效果如下圖:

  上面的「編號」、表格數據、橫線上的都是填寫的.

  這種導出到word的方法的一個缺點是模板複雜繁瑣,尤其修改的時候更是如此;是officexp及其以上版本.

  本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/myyate/archive/2008/01/17/2049701.aspx





[火星人 via ] Java中根據模板導出數據到word的解決方案已經有177次圍觀

http://www.coctec.com/docs/java/show-post-60575.html