转自   http://blog.csdn.net/cping1982/article/details/5353049

今天看到某网友关于“如何以Java实现网页截图技术”的咨询帖,由于出现该咨询的地点非常不适合较长回复,故以博文形式回答。

事实上,如果您想以Java实现网页截图,也就是“输入一段网址,几秒钟过后就能截取一张网页缩略图”的效果。那么,您至少有3种方式可以选择。

1、最直接的方式——使用Robot

方法详解:该方法利用Robat提供的强大桌面操作能力,硬性调用浏览器打开指定网页,并将网页信息保存到本地。

优势:简单易用,不需要任何第三方插件。

缺点:不能同时处理大量数据,技术含量过低,属于应急型技巧。

实现方法:使用如下代码即可。

  1. public static void main(String[] args) throws MalformedURLException,
  2. IOException, URISyntaxException, AWTException {
  3. //此方法仅适用于JdK1.6及以上版本
  4. Desktop.getDesktop().browse(
  5. new URL("http://google.com/intl/en/").toURI());
  6. Robot robot = new Robot();
  7. robot.delay(10000);
  8. Dimension d = new Dimension(Toolkit.getDefaultToolkit().getScreenSize());
  9. int width = (int) d.getWidth();
  10. int height = (int) d.getHeight();
  11. //最大化浏览器
  12. robot.keyRelease(KeyEvent.VK_F11);
  13. robot.delay(2000);
  14. Image image = robot.createScreenCapture(new Rectangle(0, 0, width,
  15. height));
  16. BufferedImage bi = new BufferedImage(width, height,
  17. BufferedImage.TYPE_INT_RGB);
  18. Graphics g = bi.createGraphics();
  19. g.drawImage(image, 0, 0, width, height, null);
  20. //保存图片
  21. ImageIO.write(bi, "jpg", new File("google.jpg"));
  22. }

2、最常规的方式——利用JNI,调用第三方C/C++组件

方法详解:目前来讲,Java领域对于网页截图组件的开发明显不足(商机?),当您需要完成此种操作时,算得上碰到了Java的软肋。但是,众所周知Java也拥有强大的JNI能力,可以轻易将C/C++开发的同类组件引为己用。

优势:实现简单,只需要封装对应的DLL文件,就可以让Java实现同类功能。

劣势:同其他JNI实现一样,在跨平台时存在隐患,而且您的程序将不再属于纯Java应用。

实现方法:可参见此用例,具体封装何种C/C++组件请自行选择。

PS:示例来源于ACA HTML to Image Converter项目(http://www.acasystems.com/en/web-thumb-activex/faq-convert-html-to-image-in-java.htm ),这是一个收费的HTML转Image第三方组件,但封装方式在Java中大同小异。

引用JNI封装:

  1. import sun.awt.*;
  2. import java.awt.*;
  3. import javax.swing.*;
  4. import java.awt.event.*;
  5. import java.awt.*;
  6. import java.awt.peer.*;
  7. public class Snap
  8. {
  9. static
  10. {
  11. System.loadLibrary("Snap");
  12. }
  13. public static void main( String[] argv )
  14. {
  15. Snap t_xSnap = new Snap();
  16. t_xSnap.Start("http://www.google.com", "snapshot-google.png");
  17. }
  18. public native void Start(String pi_strURL, String pi_strImageName);
  19. }

CPP部分的实现:

  1. #include <windows.h>
  2. #include <atlbase.h>
  3. #include "snap.h"
  4. #pragma comment(lib,"atl.lib")
  5. #import "./../../acawebthumb.dll" no_namespace
  6. JNIEXPORT void JNICALL Java_Snap_Start(JNIEnv *pEnv, jobject, jstring pi_strUrl, jstring pi_strFileName)
  7. {
  8. CoInitialize(0);
  9. _bstr_t t_strUrl = pEnv->GetStringUTFChars(pi_strUrl, 0);
  10. _bstr_t t_strFileName = pEnv->GetStringUTFChars(pi_strFileName, 0);
  11. IThumbMakerPtr HTML_Converter = NULL;
  12. HRESULT hr = HTML_Converter.CreateInstance(L"ACAWebThumb.ThumbMaker");
  13. if (SUCCEEDED(hr))
  14. {
  15. HTML_Converter->SetURL(t_strUrl);
  16. if ( 0 == HTML_Converter->StartSnap() )
  17. HTML_Converter->SaveImage(t_strFileName);
  18. }
  19. if (HTML_Converter)
  20. HTML_Converter.Release();
  21. CoUninitialize();
  22. }

以该组件图像化yahoo界面的效果图:

3、最扎实的方法——自行解析HTML标记,并将其图像化

方法详解:众所周知,HTML之所以在浏览器中以具体的网页格式出现,并非服务器端传了一整个应用到客户端,而是源自于浏览器对于客户端自行解析的结果。因此,只要我们将对应的解析一一实现,那么将网页图形化,就将不是什么难事。

 

优势:纯Java实现,一劳永逸,一旦开发完成则永远通用,而且有一定的商用价值。

劣势:开发费时,且需要针对不同语法做精确分析,才能保证输出的基本正确。尤其在涉及到JavaScript解析时,难度将尤其增大。

实现方法:目前尚无具体案例可供参考。但是,由于Java有jdic之类的浏览器项目存在(https://jdic.dev.java.net/),而Java图形界面又属绘制生成。从理论上说,我们可以将所有具备Graphics的组件图形化保存。

而如果自行解析,那么您需要建立HTML解析器(或使用第三方的,万幸Java在这方面的组件很多),了解Java2D机制,了解何时该使用drawString绘制文字,何时又该使用drawImage插入图片等等。

补充:

这是一个利用内置浏览器截图的示例,使用了DJNativeSwing组件。

示例工程下载地址(Eclipse工程,含lib):http://greenvm.googlecode.com/files/Screenshot.7z

  1. import java.awt.BorderLayout;
  2. import java.awt.Dimension;
  3. import java.awt.FlowLayout;
  4. import java.awt.image.BufferedImage;
  5. import java.io.File;
  6. import java.io.IOException;
  7. import javax.imageio.ImageIO;
  8. import javax.swing.JFrame;
  9. import javax.swing.JPanel;
  10. import javax.swing.SwingUtilities;
  11. import chrriis.dj.nativeswing.swtimpl.NativeComponent;
  12. import chrriis.dj.nativeswing.swtimpl.NativeInterface;
  13. import chrriis.dj.nativeswing.swtimpl.components.JWebBrowser;
  14. import chrriis.dj.nativeswing.swtimpl.components.WebBrowserAdapter;
  15. import chrriis.dj.nativeswing.swtimpl.components.WebBrowserEvent;
  16. public class Main extends JPanel {
  17. /**
  18. *
  19. */
  20. private static final long serialVersionUID = 1L;
  21. // 行分隔符
  22. final static public String LS = System.getProperty("line.separator", "/n");
  23. // 文件分割符
  24. final static public String FS = System.getProperty("file.separator", "//");
  25. //以javascript脚本获得网页全屏后大小
  26. final static StringBuffer jsDimension;
  27. static {
  28. jsDimension = new StringBuffer();
  29. jsDimension.append("var width = 0;").append(LS);
  30. jsDimension.append("var height = 0;").append(LS);
  31. jsDimension.append("if(document.documentElement) {").append(LS);
  32. jsDimension.append(
  33. "  width = Math.max(width, document.documentElement.scrollWidth);")
  34. .append(LS);
  35. jsDimension.append(
  36. "  height = Math.max(height, document.documentElement.scrollHeight);")
  37. .append(LS);
  38. jsDimension.append("}").append(LS);
  39. jsDimension.append("if(self.innerWidth) {").append(LS);
  40. jsDimension.append("  width = Math.max(width, self.innerWidth);")
  41. .append(LS);
  42. jsDimension.append("  height = Math.max(height, self.innerHeight);")
  43. .append(LS);
  44. jsDimension.append("}").append(LS);
  45. jsDimension.append("if(document.body.scrollWidth) {").append(LS);
  46. jsDimension.append(
  47. "  width = Math.max(width, document.body.scrollWidth);")
  48. .append(LS);
  49. jsDimension.append(
  50. "  height = Math.max(height, document.body.scrollHeight);")
  51. .append(LS);
  52. jsDimension.append("}").append(LS);
  53. jsDimension.append("return width + ':' + height;");
  54. }
  55. //DJNativeSwing组件请于http://djproject.sourceforge.net/main/index.html下载
  56. public Main(final String url, final int maxWidth, final int maxHeight) {
  57. super(new BorderLayout());
  58. JPanel webBrowserPanel = new JPanel(new BorderLayout());
  59. final String fileName = System.currentTimeMillis() + ".jpg";
  60. final JWebBrowser webBrowser = new JWebBrowser(null);
  61. webBrowser.setBarsVisible(false);
  62. webBrowser.navigate(url);
  63. webBrowserPanel.add(webBrowser, BorderLayout.CENTER);
  64. add(webBrowserPanel, BorderLayout.CENTER);
  65. JPanel panel = new JPanel(new FlowLayout(FlowLayout.CENTER, 4, 4));
  66. webBrowser.addWebBrowserListener(new WebBrowserAdapter() {
  67. // 监听加载进度
  68. public void loadingProgressChanged(WebBrowserEvent e) {
  69. // 当加载完毕时
  70. if (e.getWebBrowser().getLoadingProgress() == 100) {
  71. String result = (String) webBrowser
  72. .executeJavascriptWithResult(jsDimension.toString());
  73. int index = result == null ? -1 : result.indexOf(":");
  74. NativeComponent nativeComponent = webBrowser
  75. .getNativeComponent();
  76. Dimension originalSize = nativeComponent.getSize();
  77. Dimension imageSize = new Dimension(Integer.parseInt(result
  78. .substring(0, index)), Integer.parseInt(result
  79. .substring(index + 1)));
  80. imageSize.width = Math.max(originalSize.width,
  81. imageSize.width + 50);
  82. imageSize.height = Math.max(originalSize.height,
  83. imageSize.height + 50);
  84. nativeComponent.setSize(imageSize);
  85. BufferedImage image = new BufferedImage(imageSize.width,
  86. imageSize.height, BufferedImage.TYPE_INT_RGB);
  87. nativeComponent.paintComponent(image);
  88. nativeComponent.setSize(originalSize);
  89. // 当网页超出目标大小时
  90. if (imageSize.width > maxWidth
  91. || imageSize.height > maxHeight) {
  92. //截图部分图形
  93. image = image.getSubimage(0, 0, maxWidth, maxHeight);
  94. /*此部分为使用缩略图
  95. int width = image.getWidth(), height = image
  96. .getHeight();
  97. AffineTransform tx = new AffineTransform();
  98. tx.scale((double) maxWidth / width, (double) maxHeight
  99. / height);
  100. AffineTransformOp op = new AffineTransformOp(tx,
  101. AffineTransformOp.TYPE_NEAREST_NEIGHBOR);
  102. //缩小
  103. image = op.filter(image, null);*/
  104. }
  105. try {
  106. // 输出图像
  107. ImageIO.write(image, "jpg", new File(fileName));
  108. } catch (IOException ex) {
  109. ex.printStackTrace();
  110. }
  111. // 退出操作
  112. System.exit(0);
  113. }
  114. }
  115. }
  116. );
  117. add(panel, BorderLayout.SOUTH);
  118. }
  119. public static void main(String[] args) {
  120. NativeInterface.open();
  121. SwingUtilities.invokeLater(new Runnable() {
  122. public void run() {
  123. // SWT组件转Swing组件,不初始化父窗体将无法启动webBrowser
  124. JFrame frame = new JFrame("以DJ组件保存指定网页截图");
  125. // 加载指定页面,最大保存为640x480的截图
  126. frame.getContentPane().add(
  127. new Main("http://blog.csdn.net/cping1982", 640, 480),
  128. BorderLayout.CENTER);
  129. frame.setSize(800, 600);
  130. // 仅初始化,但不显示
  131. frame.invalidate();
  132. frame.pack();
  133. frame.setVisible(false);
  134. }
  135. });
  136. NativeInterface.runEventPump();
  137. }
  138. }

如何以Java实现网页截图技术的更多相关文章

  1. java实现网页截图

    使用工具 java+selenium+phantomjs /chromedriver /firefox 1.分别是 phantomjs插件 google截图插件 和 firefox火狐浏览器截图插件2 ...

  2. 使用PhantomJS实现网页截图服务

    这是上半年遇到的一个小需求,想实现网页的抓取,并保存为图片.研究了不少工具,效果都不理想,不是显示太差了(Canvas.Html2Image.Cobra),就是性能不怎么样(如SWT的Brower). ...

  3. 用java编网页的学习流程,我的一些小心得(初学java到高深运用)

    (1)java基础:首先得会写int,String,for循环,数组,**等等(熟练各种基础的关键字,各种java自带的排序,随即等等算法)什么是封装,继承,多态,然后private,public,p ...

  4. Java基础96 ajax技术的使用

    本文知识点(目录): 1.ajax的概念   2.使用ajax技术获取服务端的数据_实例   3.使用ajax技术检查用户名是否已存在_实例   4.使用ajax技术验证登录页面的用户名和密码_实例 ...

  5. Java精选笔记_JSP技术

    JSP技术 JSP概述 什么是JSP 在JSP全名是Java Server Page,它是建立在Servlet规范之上的动态网页开发技术. 在JSP文件中,HTML代码与Java代码共同存在,其中,H ...

  6. java学习笔记—Servlet技术(11)

    如果大家要开发一个动态的网站,那么就必须要学习一种动态的网页开发技术.那么在SUN提供的JavaEE中主要包含两种开发动态网页的技术:Servlet和JSP技术. Servlet技术简介 Servle ...

  7. Atitit.java的浏览器插件技术 Applet japplet attilax总结

    Atitit.java的浏览器插件技术  Applet  japplet attilax总结 1. Applet类及各个方法说明 1 2. JApplet类示例 2 3. / 用main方法运行JAp ...

  8. 网页截图工具CutyCapt

    网页截图工具CutyCapt   CuteCapt是Kali Linux提供的一款网页截图工具.该工具运行在命令行中,可以将WebKit引擎解析的网页保存为图片.它保存的文件支持矢量图和位图两大类型, ...

  9. Java web应用开发技术

    Java web应用程序供用户通过浏览器发送请求,程序通过执行产生web页面,并将页面传递给客户机器上的浏览器,将得到的web页面呈现给用户. 一个完整的Java web应用程序通常由许多组件构成的, ...

随机推荐

  1. javascript 关于局部变量和全局变量

    js中函数运行过程不仅仅是单纯的局部变量覆盖全局变量.和函数里面的声明情况有关. 比方: <script> var a =1; function test(){ alert(a); //a ...

  2. 菜鸟调错(三)——Jboss与jdk版本不兼容导致WebService调用出错

    环境: jdk1.6 Jboss 5.1.0.GA 问题描述: EJB发布webserivce已经成功,并且能够成功访问wsdl文件: 使用axis1自带的sample/client下的类Dynami ...

  3. 菜鸟学Java(十五)——Java反射机制(二)

    上一篇博文<菜鸟学编程(九)——Java反射机制(一)>里面,向大家介绍了什么是Java的反射机制,以及Java的反射机制有什么用.上一篇比较偏重理论,理论的东西给人讲出来总感觉虚无缥缈, ...

  4. lua--clone

    clone 深度克隆一个值. 格式: value = clone(值) 用法示例: -- 下面的代码,t2 是 t1 的引用,修改 t2 的属性时,t1 的内容也会发生变化 , b = } local ...

  5. 【Linux】Linux根目录下各文件夹的意义

    [root@localhost /]# ll / total 102 dr-xr-xr-x. 2 root root 4096 Dec 1 07:37 bin # binary file,二进制执行文 ...

  6. 扩展music-list.vue让列表前三名显示🏆奖杯

    1.在music-list.vue中写DOM <li @click="seletItem(song,index)" class="song-item" v ...

  7. hive一行变多行及多行变一行

    hive一行变多行及多行变一行 场景 name alias zhaoqiansun abc def ghi 处理数据时需要将上表处理成为下面的形式: name alias zhaoqiansun ab ...

  8. Cisco交换机堆叠与HSRP之间的区别

    随着Internet的日益普及,人们对网络的依赖性也越来越强.这同时对网络的稳定性提出了更高的要求,人们自然想到了基于设备的备份结构,就像在服务器中为提高数据的安全性而采用双硬盘结构一样.核心交换机是 ...

  9. html页面去掉滚动条

    有时候特别需要,个别网页要去掉横向滚动条和竖向滚动条,那该怎么去掉呢,很简单,看代码: 让竖条没有: <body style=`overflow:-Scroll;overflow-y:hidde ...

  10. java基础篇---XML解析(一)

    XML是可扩展标记语言 在XML文件中由于更多的是描述信息的内容,所以在得到一个xml文档后应该利用程序安装其中元素的定义名称去除对应的内容,这样的操作称为XML解析. 在XML解析中W3C定义SAX ...