选择正确的打印机 还记得我们在开始关于讨论DocFlavor之前关于打印机的那个精确支持你想要打印的数据类型的假设吗?这似乎看起来没有必要。实际上,你会对给你的打印机所支持的文档类型感到吃惊。例如,刚提到文本类型看起来似乎是最容易支持的,所以,如果你的程序要打印一个普通文本或者HTML文本,你可以随便选择一个打印服务并把它送到打印机那去。然而大部分打印机不支持基于文本的表现类,如果你试图向打印机发送它不支持的DocFlavor,会产生下面的异常: Exception in thread "main" sun.print.PrintJobFlavorException: invalid flavor at sun.print.Win32PrintJob.print(Win32PrintJob.java:290) at PrintTest.main(PrintTest.java:11) 现在你已经知道了如何得到一个DocFlavor的引用而且我们也讨论了选择支持这个flavor的打印机重要性,接下来我来告诉你如何确定你使用的打印机支持它。我先前说过lookupPrintServices()允许你指定一个DocFlavor作为第一个参数,如果你指定的参数非空,那么方法会返回相应支持这个的打印机的实例。例如以下代码将返回可以通过URL来打印gif文件的打印机的列表: DocFlavor flavor = DocFlavor.URL.GIF; PrintService[] services = PrintServiceLookup.lookupPrintServices(flavor, null); 另外,如果你的程序已经获得了打印服务的实例,而你想知道它是否支持另一种特定的flavor,你可以调用isDocFlavorSupported()方法。在下面的代码里,将得到一个默认打印机的引用,如果不能打印gif就会出现错误信息: PrintService service = PrintServiceLookup.lookupDefaultPrintService(); DocFlavor flavor = DocFlavor.URL.GIF; if (!service.isDocFlavorSupported(flavor)) { System.err.println("The printer does not support the appropriate DocFlavor"); } AttributeSet 正如你看到的,DocFlavor描述打印数据而且可以用来确定打印服务是否支持这种数据。然而,你的程序需要选择一个基于那些支持的元素的打印机。例如,你要打印图片用不同的颜色来描述不同的信息,你想知道提供的服务是否支持彩色打印,如果不,那么要么禁止它使用或者要求提供一个黑白图片。 类似彩色打印,两边打印或者使用不同的定位取决于打印机本身的属性,而javax.print.attribute包包含了许多你可以用于描述这些属性的包和接口。其中一个接口是前面提到的lookupPrintServices()中第二个参数AttributeSet。正如你愿,它返回属性的集合,在调用lookupPrintServices()指定一个不为空的值将返回支持这些属性的打印服务。换句话说,如果DocFlavor和 AttributeSet都不为空,那么方法将返回那些这两种属性都支持的打印机 Attribute AttributeSet 是属性的集合,一个显而易见的问题是如何指定属性的值呢? javax.print.attribute包里同时含有一个叫Attribute的接口,你马上可以看到通过调用add方法来给AttributeSet创建一个Attribute实例来获得这个集合。在javax.print.attribute.standard包里定义了大量你将要用到的接口。在之前,你可以查看javax.print.attribute这个包里的其他接口。 属性模块 目前为止,我们把属性描述成打印服务的功能,而实际上在java支持的属性中算很简单的。对应每个属性,java都有相应的模块。只有遵循这些模块属性才有效。在不同的java打印服务位置使用不同的属性,而不是所有的属性在任何地方都适用。 为了更好的理解这个,来看一下javax.print.attribute.standard 包里定义的 OrientationRequested和 ColorSupported接口。创建一个新的打印文档时可以指定OrientationRequested属性和用于打印的定位。ColorSupported在你调用PrintService接口的getAttributes方法时返回。OrientationRequested是一个你用来传给打印机的属性,而ColorSupported是打印服务用来提供给你关于打印机能力信息的工具。你可以在创建打印文档时把ColorSupported作为属性指定,因为打印机是否支持彩色打印是你的程序不能控制的。 接口和继承 你第一次查看javax.print.attribute包里的接口和类时你也许会感到选择那些列表里的接口和类很麻烦。除了Attribute 和AttributeSet和继承AttributeSet的HashAttributeSet,javax.print.attribute包里有4个子接口和类,列出在表4和图1中。 Table 4. javax.print.attribute 里定义的接口和类 Figure 1. javax.print.attribute 包的一部分类的层次结构. 那么有了Attribute, AttributeSet, 和 HashAttributeSet为什么需要使用这些不同的接口和继承类呢?是因为这些特殊的类是为那些特殊的属性量身定做的。比方说,我提到过当你创建打印文档的时候有个地方可以使用的属性例如ColorSupported在那里不能使用。当创建这样的文档,你可以使用DocAttributeSet接口(或者更专业一点,HashDocAttributeSet这个继承的类),这个继承类只允许你添加继承DocAttribute这个接口的属性。这四种不同的模块如下: ·Doc: 在创建文档时指定如何打印文档 ·PrintJob: 打印任务的属性描述任务的状态 ·PrintRequest: 初始化打印时传给任务的请求 ·PrintService:由打印服务返回来描述打印机的功能 要知道如何工作,我们来创建一个DocAttributeSet的实例然后为AttributeSet设置DocAttributeSet和OrientationRequested属性。HashDocAttributeSet定义了很好的结构,所有你可以很简便的如下创建实例: DocAttributeSet attrs = new HashDocAttributeSet(); 现在你已经创建了AttributeSet,你可以调用add方法并把它传给Attribute的继承实例去。如果你看了OrientationRequested这个类的文档,你会发现它包含了一系列静态的OrientationRequest实例,每一个对应一种文档定位方式。要指定你想要的类型,你所要做的只是按下面的方法传给add方法一个静态的实例的引用: DocAttributeSet attrs = new HashDocAttributeSet(); attrs.add(OrientationRequested.PORTRAIT); ColorSupported类有一点不同但一样很简单,它定义了两种静态实例:一个表示支持彩色打印另一个不是。你可以试着增加一个ColorSupported属性到DocAttributeSet去,代码如下: DocAttributeSet attrs = new HashDocAttributeSet(); attrs.add(OrientationRequested.PORTRAIT); attrs.add(ColorSupported.SUPPORTED); 早先提过,去指定是否支持彩色打印不恰当因为这不是程序所能控制的内容。换句话说,ColorSupported这个属性放到一系列文档属性中并不合适,所以,运行先前的代码当添加ColorSupported属性时会抛出一个ClassCastException异常。 要学习怎么运行,记住每一个AttributeSet子接口都有一个相应Attribute子接口和继承子类。当添加一个属性时,继承的子类试图把Attribute作为参数给相应的子接口,这样来确保只有当前适当的属性会成功添加。 这样的话,HashDocAttributeSet 的add方法第一次和OrientationRequested的一个实例一起调用,并成功的把它作为一个object传给DocAttribute。因为如图2所示,OrientationRequested继承了那个接口。与之相对应,传ColorSupported实例的时候因为没有继承DocAttribute所以失败了。 Figure 2. javax.print.attribute 包的一部分类的层次结构 这个例子举例说明,表4里的四个接口和类组来保证使用正确的属性。注意模块和不同的属性之间有大量的交互,所以很多属性与不止一个模块关联。例如,许多属性继承了PrintJobAttribute 和 PrintRequestAttribute因为大部分是通过一个相关的打印任务获得提供给你的。你可以在初始化时指定它们。举个例子,你可以把它加到PrintRequestAttributeSet中去来指定任务名,并且在打印的时候通过PrintJobAttributeSet来返回它。因此,JobName属性类同时继承PrintRequestAttribute 和 PrintJobAttribute。 AttributeSet and HashAttributeSet 你已经知道了为什么会有四个子类,但是AttributeSet接口和HashAttributeSet父类又是什么呢?AttributeSet/HashAttributeSet在你不能确定要存储在这个集合中的那些仅仅和一个模块相关的属性时使用。记得我以前提到的lookupPrintServices()方法允许你指定AttributeSet参数来限制返回的打印服务。表面上看来最好指定PrintServiceAttributeSet的实例,但是很多你可能用到的属性并不继承PrintServiceAttribute。 我们假设你想要让lookupPrintServices()方法返回支持彩色打印和风景画打印的打印机。这些属性与ColorSupported和OrientationRequested属性关联,但是请注意这些类并不共享模块,前者是一个PrintServiceAttribute而OrientationRequested与另外三个模块(Doc, PrintRequest,和 PrintJob)关联。这意味着不存在单个的AttributeSet接口或类来同时包含ColorSupported和Sides属性。 创建AttributeSet的方法使用一个HashAttributeSet实例同时包含一个OrientationRequested 和 ColorSupported太简单了。不像 |