数字水印有可见不可见之分,可见的比如课件上印有学校校徽,微博发图片会水印上上传者的信息及微博logo等。
用java实现可见的数字水印,草人主要是用到了java.awt包中的AlphaComposite类,当然在实现之前先介绍一下AlphaComposite类:
AlphaComposite类是关于两个目标重叠的混合处理类,此类实现的特定规则是 T. Porter 和 T. Duff 合著的 “Compositing Digital Images”, SIGGRAPH 84, 253-259 中描述的 12 条基本规则集。该类提供的一个getInstance的方法,其中的两个参数为rule和alpha,第二个参数将由调用者设置一个alpha值,即是透明度的设置,而第一个参数则是混合方式。此类扩展了 Porter 和 Duff 定义的方程,包含一个额外的因子。AlphaComposite 类的实例可以包含一个 alpha 值,在将该值用于混合方程之前,可以用它来修改不透明度和每个源像素的覆盖率。
Porter和 Duff 的论文在混合方程的描述中使用了以下因子:
以规则SRC_OVER为例,使用这些因子,Porter 和 Duff 定义了 12 种选择混合因子 Fs 和 Fd 的方法,从而产生了 12 种令人满意的可视效果。在对 12 个指定可视效果的静态字段的描述中,给出了具有确定 Fs 和 Fd 值的方程。SRC_OVER在目标色之上合成源色(Porter-Duff Source Over Destination 规则)。指定Fs = 1 和 Fd = (1-As),因此:
Graphics2D g2d=image.createGraphics();//用源图像填充背景g2d.drawImage(image, 0, 0, image.getWidth(), image.getHeight(), null, null);
然后为 Graphics2D 上下文设置 Composite后就可以将想要写入的文字或者图片写入源图片上
AlphaComposite ac = AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha); //为 Graphics2D 上下文设置 Composite。 Composite 用于所有绘制方法中,如 drawImage、 //drawString、draw 和 fill。 它指定新的像素如何在呈现过程中与图形设备上的现有像素组合。 g2d.setComposite(ac);
完整代码(代码中注释足够了,就不多啰嗦):
package cumt.zry.two;import java.awt.*;import java.awt.image.BufferedImage;import java.io.*;import javax.imageio.*;public class Watermark2{ public Watermark2(){super();};/** * 在源图片上设置水印文字 */ public void WordsToImage(String srcImagePath,float alpha, String font,int fontStyle,int fontSize,Color color, String inputWords,int x,int y,String imageFormat,String toPath) throws IOException{ FileOutputStream fos=null; try { //读取图片 BufferedImage image = ImageIO.read(new File(srcImagePath)); //创建java2D对象 Graphics2D g2d=image.createGraphics(); //用源图像填充背景 g2d.drawImage(image, 0, 0, image.getWidth(), image.getHeight(), null, null); //!!!! AlphaComposite ac = AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha); //为 Graphics2D 上下文设置 Composite。 Composite 用于所有绘制方法中,如 drawImage、 //drawString、draw 和 fill。 它指定新的像素如何在呈现过程中与图形设备上的现有像素组合。 g2d.setComposite(ac); //设置文字字体名称、样式、大小 g2d.setFont(new Font(font, fontStyle, fontSize)); g2d.setColor(color);//设置字体颜色 g2d.drawString(inputWords, x, y); //输入水印文字及其起始x、y坐标 g2d.dispose(); //将水印后的图片写入toPath路径中 fos=new FileOutputStream(toPath); ImageIO.write(image, imageFormat, fos); } //文件操作错误抛出 catch (Exception e) { e.printStackTrace(); }finally{ if(fos!=null){ fos.close(); } } } /** * 在源图像上设置图片水印 */ public void ImageToImage(String srcImagePath,String appendImagePath, float alpha,int x,int y,int width,int height, String imageFormat,String toPath) throws IOException{ FileOutputStream fos = null; try { //读图 BufferedImage image = ImageIO.read(new File(srcImagePath)); //创建java2D对象 Graphics2D g2d=image.createGraphics(); //用源图像填充背景 g2d.drawImage(image, 0, 0, image.getWidth(), image.getHeight(), null, null); //关键地方 AlphaComposite ac = AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha); g2d.setComposite(ac); BufferedImage appendImage = ImageIO.read(new File(appendImagePath)); g2d.drawImage(appendImage, x, y, width, height, null, null); g2d.dispose(); fos=new FileOutputStream(toPath); ImageIO.write(image, imageFormat, fos); } catch (Exception e) { e.printStackTrace(); }finally{ if(fos!=null){ fos.close(); } } } public static void main(String[] args) throws Exception { Watermark2 imageObj = new Watermark2(); //源图片路径 String srcImagePath = "F:/27.jpg"; //水印图片路径 String appendImagePath = "F:/logo.jpg"; // ---- 宋体 普通字体 77号字 红色 透明度0.4" float alpha = 0.4F; String font = "宋体"; int fontStyle = Font.PLAIN; int fontSize = 77; Color color = Color.RED; String inputWords = "图片上设置水印文字"; int x = 1700; int y = 77; String imageFormat = "jpg"; //水印文字后的存储路径 String wToPath = "F:/31.png"; //水印图片后的存储路径 String IToPath = "F:/7.png" ; imageObj.WordsToImage(srcImagePath, alpha, font, fontStyle, fontSize, color, inputWords, x, y, imageFormat, wToPath); imageObj.ImageToImage(srcImagePath, appendImagePath, alpha, x, y, 300, 200, imageFormat, IToPath); }}
实现预览:
更多参考文档,Java2d文档:http://www.apihome.cn/api/java/Graphics2D.html
- 1楼Hyman_Lee
- 学习了,但是g2d.drawImage(image, 0, 0, image.getWidth(), image.getHeight(), null, null);这句有点多余。
- Re: 半透明的稻草人
- @Hyman_Lee,谢谢,这个问题太好了。草人写的原版是界面版的,为了在面板中全图显示所以我用到了drawImage()。但是你这么一提醒,我发现在上面代码中这句不仅是多余,而且应该还算是错误吧,因为这样写重复了(我调整了4,5参数后)——,这样看来就是图片被写了两次。但是却发现了新问题,用ImageIO.read()读取图片然后不做任何操作,用ImageIO.write()将图片写入后发现图片大了不小,为什么会大呢?