在我們的實(shí)際工作中,經(jīng)常需要實(shí)現(xiàn)打印功能。但由于歷史原因,Java 提供的打印功能一直都比較弱。實(shí)際上最初的 jdk 根本不支持打印,直到 jdk1.1 才引入了很輕量的打印支持。所以,在以前Java/Applet/JSP/Servlet 設(shè)計(jì)的程序中,較復(fù)雜的打印都是通過(guò)調(diào)用 ActiveX/OCX 控件或者 VB/VC 程序來(lái)實(shí)現(xiàn)的,非常麻煩。
實(shí)際上,SUN 公司也一直致力于 Java 打印功能的完善,而 Java2 平臺(tái)則終于有了一個(gè)健壯的打印模式的開(kāi)端,該打印模式與 Java2D 圖形包充分結(jié)合成一體。更令人鼓舞的是,新發(fā)布的 jdk1.4 則提供了一套完整的“Java 打印服務(wù) API” (Java Print Service API),它對(duì)已有的打印功能是積極的補(bǔ)充。利用它,我們可以實(shí)現(xiàn)大部分實(shí)際應(yīng)用需求,包括打印文字、圖形、文件及打印預(yù)覽等等。本文將通過(guò)一個(gè)具體的程序?qū)嵗齺?lái)說(shuō)明如何設(shè)計(jì) Java 打印程序以實(shí)現(xiàn)這些功能,并對(duì)不同版本的實(shí)現(xiàn)方法進(jìn)行分析比較。希望大家能從中獲取一些有益的提示。
Java 中的打印
2.1 Java 的打印 API
Java 的打印 API 主要存在于 java.awt.print 包中。而 jdk1.4 新增的類(lèi)則主要存在于 javax.print 包及其相應(yīng)的子包 javax.print.event 和 javax.print.attribute 中。其中 javax.print 包中主要包含打印服務(wù)的相關(guān)類(lèi),而 javax.print.event 則包含打印事件的相關(guān)定義,javax.print.attribute 則包括打印服務(wù)的可用屬性列表等。
2.2 如何實(shí)現(xiàn)打印
要產(chǎn)生一個(gè)打印,至少需要考慮兩條:
需要一個(gè)打印服務(wù)對(duì)象。這可通過(guò)三種方式實(shí)現(xiàn):在 jdk1.4 之前的版本,必須要實(shí)現(xiàn) java.awt.print.Printable 接口或通過(guò) Toolkit.getDefaultToolkit().getPrintJob 來(lái)獲取打印服務(wù)對(duì)象;在 jdk1.4 中則還可以通過(guò) javax.print.PrintSerivceLookup 來(lái)查找定位一個(gè)打印服務(wù)對(duì)象。
需要開(kāi)始一個(gè)打印工作。這也有幾種實(shí)現(xiàn)方法:在 jdk1.4 之前可以通過(guò) java.awt.print.PrintJob(jdk1.1 提供的,現(xiàn)在已經(jīng)很少用了)調(diào)用 print 或 printAll 方法開(kāi)始打印工作;也可以通過(guò) java.awt.print.PrinterJob 的 printDialog 顯示打印對(duì)話(huà)框,然后通過(guò) print 方法開(kāi)始打?。辉?jdk1.4 中則可以通過(guò) javax.print.ServiceUI 的 printDialog 顯示打印對(duì)話(huà)框,然后調(diào)用 print 方法開(kāi)始一個(gè)打印工作。
2.3 打印機(jī)對(duì)話(huà)框
2.3.1 Printable 的打印對(duì)話(huà)框
開(kāi)始打印工作之前,可以通過(guò) PrinterJob.printDialog 來(lái)顯示一個(gè)打印對(duì)話(huà)框。它給用戶(hù)一個(gè)機(jī)會(huì)以選擇應(yīng)該打印的頁(yè)碼范圍,并可供用戶(hù)改變打印設(shè)置。它是一個(gè)本地對(duì)話(huà)框。
事實(shí)上,當(dāng)從一個(gè) Printable 對(duì)象進(jìn)行一個(gè)打印工作時(shí),打印對(duì)象并不知道需要打印多少頁(yè)。它只是不停地調(diào)用 print 方法。只要 print 方法返回 Printable.PAGE_EXISTS 值,打印工作就不停地產(chǎn)生打印頁(yè),直到 print 方法返回 Printable.NO_SUCH_PAGE 時(shí),打印工作才停止。
由于打印工作只有在打印完成后才進(jìn)行準(zhǔn)確的頁(yè)數(shù)計(jì)算,所以在對(duì)話(huà)框上的頁(yè)碼范圍是尚未初始化的 [1,9999]。我們可以通過(guò)構(gòu)建一個(gè) java.awt.print.Book 對(duì)象傳遞給打印對(duì)象;也可以通過(guò)指定的格式計(jì)算需要打印的頁(yè)數(shù)并傳遞給打印對(duì)象,使其準(zhǔn)確地知道要打印多少頁(yè)。
2.3.2 ServiceUI 的打印對(duì)話(huà)框
與 Printable 的對(duì)話(huà)框不同的是,在 jdk1.4 提供 ServiceUI 的打印機(jī)對(duì)話(huà)框的缺省行為已經(jīng)用新的 API 更改了:缺省情況下對(duì)話(huà)框不顯示。我們必須使用 ServiceUI 類(lèi)調(diào)用 printDialog 方法創(chuàng)建如下所示的打印對(duì)話(huà)框。
Java 打印程序設(shè)計(jì)實(shí)例
3.1 打印文本
3.1.1 應(yīng)用場(chǎng)景
假設(shè)我們需要打印一個(gè)窗體的某個(gè)文本編輯域(可能只有幾行,也可能包含多頁(yè))的內(nèi)容,并且每頁(yè)最多打印 54 行,如何實(shí)現(xiàn)呢?
3.1.2 解決方法
基本思路如下:首先我們需要實(shí)現(xiàn) Printable 接口,然后按照每頁(yè)最多 54 行的格式計(jì)算共需要打印多少頁(yè),當(dāng)打印文本的按鈕被點(diǎn)擊時(shí),執(zhí)行相應(yīng)的打印動(dòng)作。打印文本的具體操作可通過(guò) Graphics2D 的 drawString 方法來(lái)實(shí)現(xiàn)。
1、實(shí)現(xiàn) Printable 接口
/*Graphic 指明打印的圖形環(huán)境;PageFormat 指明打印頁(yè)格式(頁(yè)面大小以點(diǎn)為計(jì)量單位,
1 點(diǎn)為 1 英才的 1/72,1 英寸為 25.4 毫米。A4 紙大致為 595 × 842 點(diǎn));page 指明頁(yè)號(hào) */
public int print(Graphics g, PageFormat pf, int page) throws PrinterException
{
Graphics2D g2 = (Graphics2D)g;
g2.setPaint(Color.black); // 設(shè)置打印顏色為黑色
if (page 》= PAGES) // 當(dāng)打印頁(yè)號(hào)大于需要打印的總頁(yè)數(shù)時(shí),打印工作結(jié)束
return Printable.NO_SUCH_PAGE;
g2.translate(pf.getImageableX(), pf.getImageableY());// 轉(zhuǎn)換坐標(biāo),確定打印邊界
drawCurrentPageText(g2, pf, page); // 打印當(dāng)前頁(yè)文本
return Printable.PAGE_EXISTS; // 存在打印頁(yè)時(shí),繼續(xù)打印工作
}
/* 打印指定頁(yè)號(hào)的具體文本內(nèi)容 */
private void drawCurrentPageText(Graphics2D g2, PageFormat pf, int page)
{
String s = getDrawText(printStr)[page];// 獲取當(dāng)前頁(yè)的待打印文本內(nèi)容
// 獲取默認(rèn)字體及相應(yīng)的尺寸
FontRenderContext c
Font f = area.getFont();
String drawText;
float ascent = 16; // 給定字符點(diǎn)陣
int k, i = f.getSize(), lines = 0;
while(s.length() 》 0 && lines 《 54) // 每頁(yè)限定在 54 行以?xún)?nèi)
{
下面的例子實(shí)現(xiàn)了打印字符串,線(xiàn)(包括虛線(xiàn))和打印圖片。而且通過(guò)Paper的setImageableArea可以設(shè)置打印的區(qū)域和邊距,讓開(kāi)發(fā)者隨意的設(shè)置打印的位置。
打印字符串
package com.cn.gao;
import java.awt.Graphics;
import java.awt.print.Book;
import java.awt.print.PageFormat;
import java.awt.print.Paper;
import java.awt.print.Printable;
import java.awt.print.PrinterException;
import java.awt.print.PrinterJob;
import java.awt.*;
public class PrintTest implements Printable{
/**
* @param Graphic指明打印的圖形環(huán)境
* @param PageFormat指明打印頁(yè)格式(頁(yè)面大小以點(diǎn)為計(jì)量單位,1點(diǎn)為1英才的1/72,1英寸為25.4毫米。A4紙大致為595×842點(diǎn))
* @param pageIndex指明頁(yè)號(hào)
**/
public int print(Graphics gra, PageFormat pf, int pageIndex) throws PrinterException {
System.out.println(“pageIndex=”+pageIndex);
Component c = null;
//print string
String str = “中華民族是勤勞、勇敢和富有智慧的偉大民族?!?
//轉(zhuǎn)換成Graphics2D
Graphics2D g2 = (Graphics2D) gra;
//設(shè)置打印顏色為黑色
g2.setColor(Color.black);
//打印起點(diǎn)坐標(biāo)
double x = pf.getImageableX();
double y = pf.getImageableY();
switch(pageIndex){
case 0:
//設(shè)置打印字體(字體名稱(chēng)、樣式和點(diǎn)大?。ㄗ煮w名稱(chēng)可以是物理或者邏輯名稱(chēng))
//Java平臺(tái)所定義的五種字體系列:Serif、SansSerif、Monospaced、Dialog 和 DialogInput
Font font = new Font(“新宋體”, Font.PLAIN, 9);
g2.setFont(font);//設(shè)置字體
//BasicStroke bs_3=new BasicStroke(0.5f);
float[] dash1 = {2.0f};
//設(shè)置打印線(xiàn)的屬性。
//1.線(xiàn)寬 2、3、不知道,4、空白的寬度,5、虛線(xiàn)的寬度,6、偏移量
g2.setStroke(new BasicStroke(0.5f, BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER, 2.0f, dash1, 0.0f));
//g2.setStroke(bs_3);//設(shè)置線(xiàn)寬
float heigth = font.getSize2D();//字體高度
System.out.println(“x=”+x);
// -1- 用Graphics2D直接輸出
//首字符的基線(xiàn)(右下部)位于用戶(hù)空間中的 (x, y) 位置處
//g2.drawLine(10,10,200,300);
Image src = Toolkit.getDefaultToolkit().getImage(“F:\\workspace\\QQ.png”);
g2.drawImage(src,(int)x,(int)y,c);
int img_Height=src.getHeight(c);
int img_width=src.getWidth(c);
//System.out.println(“img_Height=”+img_Height+“img_width=”+img_width) ;
g2.drawString(str, (float)x, (float)y+1*heigth+img_Height);
g2.drawLine((int)x,(int)(y+1*heigth+img_Height+10),(int)x+200,(int)(y+1*heigth+img_Height+10));
g2.drawImage(src,(int)x,(int)(y+1*heigth+img_Height+11),c);
return PAGE_EXISTS;
default:
return NO_SUCH_PAGE;
}
}
public static void main(String[] args) {
// 通俗理解就是書(shū)、文檔
Book book = new Book();
// 設(shè)置成豎打
PageFormat pf = new PageFormat();
pf.setOrientation(PageFormat.PORTRAIT);
// 通過(guò)Paper設(shè)置頁(yè)面的空白邊距和可打印區(qū)域。必須與實(shí)際打印紙張大小相符。
Paper p = new Paper();
p.setSize(590,840);//紙張大小
p.setImageableArea(10,10, 590,840);//A4(595 X 842)設(shè)置打印區(qū)域,其實(shí)0,0應(yīng)該是72,72,因?yàn)锳4紙的默認(rèn)X,Y邊距是72
pf.setPaper(p);
// 把 PageFormat 和 Printable 添加到書(shū)中,組成一個(gè)頁(yè)面
book.append(new PrintTest(), pf);
//獲取打印服務(wù)對(duì)象
PrinterJob job = PrinterJob.getPrinterJob();
// 設(shè)置打印類(lèi)
job.setPageable(book);
try {
//可以用printDialog顯示打印對(duì)話(huà)框,在用戶(hù)確認(rèn)后打印;也可以直接打印
boolean a=job.printDialog();
if(a)
{
job.print();
}else{
job.cancel();
}
} catch (PrinterException e) {
e.printStackTrace();
}
}
}
代碼
下面的打印代碼沒(méi)有設(shè)置打印區(qū)域,默認(rèn)為打印紙張的區(qū)域和邊距,比如我們一般用的A4紙,打印的起點(diǎn)X和Y坐標(biāo)則是72,72。無(wú)區(qū)域設(shè)置的代碼:
package com.cn.gao;
import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Component;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.Toolkit;
import java.awt.print.*;
public class PrintTest1 implements Printable{
/**
* @param Graphic指明打印的圖形環(huán)境
* @param PageFormat指明打印頁(yè)格式(頁(yè)面大小以點(diǎn)為計(jì)量單位,1點(diǎn)為1英才的1/72,1英寸為25.4毫米。A4紙大致為595×842點(diǎn))
* @param pageIndex指明頁(yè)號(hào)
**/
public int print(Graphics gra, PageFormat pf, int pageIndex) throws PrinterException {
System.out.println(“pageIndex=”+pageIndex);
Component c = null;
//print string
String str = “中華民族是勤勞、勇敢和富有智慧的偉大民族?!?
//轉(zhuǎn)換成Graphics2D
Graphics2D g2 = (Graphics2D) gra;
//設(shè)置打印顏色為黑色
g2.setColor(Color.black);
/*Paper paper = pf.getPaper();//得到頁(yè)面格式的紙張
paper.setSize(500,500);//紙張大小
paper.setImageableArea(0,0,500,500); //設(shè)置打印區(qū)域的大小
System.out.println(paper.getWidth());
System.out.println(paper.getHeight());
pf.setPaper(paper);//將該紙張作為格式 */
//打印起點(diǎn)坐標(biāo)
double x = pf.getImageableX();
double y = pf.getImageableY();
switch(pageIndex){
case 0:
//設(shè)置打印字體(字體名稱(chēng)、樣式和點(diǎn)大?。ㄗ煮w名稱(chēng)可以是物理或者邏輯名稱(chēng))
//Java平臺(tái)所定義的五種字體系列:Serif、SansSerif、Monospaced、Dialog 和 DialogInput
Font font = new Font(“新宋體”, Font.PLAIN, 9);
g2.setFont(font);//設(shè)置字體
//BasicStroke bs_3=new BasicStroke(0.5f);
float[] dash1 = {4.0f};
g2.setStroke(new BasicStroke(0.5f, BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER, 4.0f, dash1, 0.0f));
float heigth = font.getSize2D();//字體高度
System.out.println(“x=”+x);
//使用抗鋸齒模式完成文本呈現(xiàn)
/*g2.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,
RenderingHints.VALUE_TEXT_ANTIALIAS_ON);*/
// -1- 用Graphics2D直接輸出
//首字符的基線(xiàn)(右下部)位于用戶(hù)空間中的 (x, y) 位置處
//g2.drawLine(10,10,200,10);
Image src = Toolkit.getDefaultToolkit().getImage(“d://logo.gif”);
g2.drawImage(src,(int)x,(int)y,c);
int img_Height=src.getHeight(c);
int img_width=src.getWidth(c);
//System.out.println(“img_Height=”+img_Height+“img_width=”+img_width) ;
g2.drawString(str, (float)x, (float)y+1*heigth+img_Height);
g2.drawLine((int)x,(int)(y+1*heigth+img_Height+10),(int)x+200,(int)(y+1*heigth+img_Height+10));
g2.drawImage(src,(int)x,(int)(y+1*heigth+img_Height+11),c);
// -2- 直接構(gòu)造TextLayout打印
/*FontRenderContext frc = g2.getFontRenderContext();
TextLayout layout = new TextLayout(str, font, frc);
layout.draw(g2, (float)x, (float)y+2*heigth);*/
// -3- 用LineBreakMeasurer進(jìn)行打印
/*AttributedString text = new AttributedString(str);
text.addAttribute(TextAttribute.FONT, font);
LineBreakMeasurer lineBreaker = new LineBreakMeasurer(text.getIterator(), frc);
//每行字符顯示長(zhǎng)度(點(diǎn))
double width = pf.getImageableWidth();
//首字符的基線(xiàn)位于用戶(hù)空間中的 (x, y) 位置處
Point2D.Double pen = new Point2D.Double (100, y+3*heigth);
while ( (layout = lineBreaker.nextLayout( (float) width)) != null){
layout.draw(g2, (float)x, (float) pen.y);
pen.y += layout.getAscent();
}*/
return PAGE_EXISTS;
default:
return NO_SUCH_PAGE;
}
}
public static void main(String[] args) {
//獲取打印服務(wù)對(duì)象
PrinterJob job = PrinterJob.getPrinterJob();
PageFormat pageFormat = job.defaultPage();//得到默認(rèn)頁(yè)格式
job.setPrintable(new PrintTest1());//設(shè)置打印類(lèi)
try {
//可以用printDialog顯示打印對(duì)話(huà)框,在用戶(hù)確認(rèn)后打??;也可以直接打印
boolean a=job.printDialog();
if(a)
{
job.print();
} else{
job.cancel();
}
} catch (PrinterException e) {
e.printStackTrace();
}
}
}
評(píng)論
查看更多