`
zhaohaolin
  • 浏览: 1017102 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

使用 JFreeChart 来创建基于 web 的图表

    博客分类:
  • JAVA
 
阅读更多
刘冬 (winter.lau@163.com ), 软件工程师, 珠海市创我科技发展有限公司

简介:  WWW 的发展使得基于因特网的应用程序不再局限于静态或者简单的动态内容提供。传统的一些以软件包形式发布应用程序例如报表系统等都在逐渐搬到因特网上。但是这 两者之间有着天壤之别,虽然对于数据获取、业务处理等方面基本类似,但是最大的差别在于用户界面。为了能在 web 浏览器上显示要求用户界面使用 HTML 以及图片的方式来展现数据,而传统的一些利用操作系统本身的控件来开发的用户界面无法适应琳琅满目的客户端,因此在这里也变得无能为力。回到本文的题目上 来,为了创建一个可以在 web 浏览器上查看到图表一般有两种做法:第一种就是使用 applet 利用 java 本身对图形的支持来显示一个图表;第二种就是直接在 web 服务器端生成好图表图片文件后发送给浏览器。第一种方式显然对于客户端要求太高,随着现在主流浏览器放弃对 JAVA 的支持后,这种方式只适合一些局域网的应用,而对于因特网的环境就显得不太适合。因此我们下面将介绍一个 JAVA 的图表引擎 JFreeChart 用来产生基于 WEB 的图表。

本文的标签:   jfreechart

发布日期:  2003 年 9 月 08 日
级别:  初级
访问情况  7183 次浏览
建议:  0 (添加评论 )

1 star 2 stars 3 stars 4 stars 5 stars 平均分 (共 12 个评分 )

JFreeChart 项目简介

JFreeChart 是开放源代码站点 SourceForge.net 上的一个 JAVA 项目,它主要用来各种各样的图表,这些图表包括:饼图、柱状图 ( 普通柱状图以及堆栈柱状图 )、线图、区域图、分布图、混合图、甘特图以及一些仪表盘等等。这些不同式样的图表基本上可以满足目前的要求。为了减少篇幅本文主要介绍前面三种类型的图 表,读者可以触类旁通去开发其他样式的图表。下面几个是 JFreeChart 产生的这三种类型图表的结果:


图 1


图 2


图 3

上面的三个图都是表示四个季度的某个产品的销量信息。在继续下面小节之前必须先准备好开发环境,因为是基于 WEB 浏览器的图表展现,因此需要一个 Servlet 引擎或者是 J2EE 应用服务器(例如 WebSphere,Tomcat 等)。WEB 环境的搭建就不累赘了,读者根据喜好自行安装。JFreeChart 引擎本身需要到 SourceForge.net 上下载,地址如下:

 JFreeChart 主页:
        http://www.jfree.org/jfreechart/index.html

 JFreeChart 下载页面:
        http://sourceforge.net/projects/jfreechart/

			

 

下载的时候需要注意的是必须下载两个文件:JFreeChart 以及 Jcommon。目前最新配套版本是:JFreeChart 0.9.11 Jcommon 0.8.6

这里有点笔者在开发中遇见的问题需要注意的是:在使用 Eclipse 开发的时候会报一个莫名其妙的错误,错误可能指向某个类文件的第一行。遇到这样的问题一般是因为没有把 Jcommon 的 jar 包设置到项目的类路径中的缘故。具体的原因不祥。

解读 JFreeChart 的源码结构

在开始使用 JFreeChart 之前我们有必要先大概了解一下 JFreeChart 本身的结构以及它所带一些例子程序,这样有助于我们下一步自行开发。下载 JFreeChart 包后已经带有非常丰富的例子,因为 JFreeChart 这个项目本身的使用文档非常少,因此学习它最好的办法就是学习它所带的例子源码。在包 org.jfree.chart.demo 中有几十个文件用于展示 JFreeChart 所能支持的所有图表的结果。如果你的 JDK 是比较新的情况下可能在运行这些例子时会有问题,现象如下:

java.lang.UnsatisfiedLinkError: initDDraw 
at sun.awt.windows.Win32OffScreenSurfaceData.initDDraw(Native Method) 
at sun.awt.windows.Win32OffScreenSurfaceData.<clinit>
    (Win32OffScreenSurfaceData.java:141)
at sun.awt.Win32GraphicsDevice.<clinit>(Win32GraphicsDevice.java:58) 
at sun.awt.Win32GraphicsEnvironment.makeScreenDevice(Win32GraphicsEnvironment.java:168)
at sun.java2d.SunGraphicsEnvironment.getScreenDevices(SunGraphicsEnvironment.java:240) 
at sun.awt.Win32GraphicsEnvironment.getDefaultScreenDevice
    (Win32GraphicsEnvironment.java:61)
at java.awt.Window.init(Window.java:224) 
at java.awt.Window.<init>(Window.java:268) 
at java.awt.Frame.<init>(Frame.java:398) 
at javax.swing.JFrame.<init>(JFrame.java:198) 
at org.jfree.chart.demo.JFreeChartDemo.<init>(JFreeChartDemo.java:148) 
at org.jfree.chart.demo.JFreeChartDemo.main(JFreeChartDemo.java:285) 
Exception in thread "main"

 

这个错误是由于新版的 Swing 大量的使用了微软的 DirectDraw 的技术来提高画图的性能,而可能你的显卡在这时候会跟你闹点情绪或者显卡本身并不支持这样的一个技术。难道就没有办法了嘛?要解决这个问题也非常简单,我 们可以屏蔽掉 DirectDraw,不让 Swing 使用该技术就可以了。在运行这些代码时给虚拟机指定参数 -Dsun.java2d.noddraw 即可。

这时可能你又该纳闷了,不说是基于 Web 的图表嘛,怎么又扯到 Swing 上了?这是因为为了使开发者容易上手,无需配置任何运行环境,所以这些例子都是基于 GUI 方式的用于展现给开发者如果生成一个图表,我们要学习的也就是如何利用这个引擎生成图表而不是怎么来显示一个图表。当我们把生成的图表对象 Export 到一个图像文件即可在 Web 上发布。

下面我们来介绍 JFreeChart 中几个核心的对象类:

类名 类的作用以及简单描述
JFreeChart 图表对象,任何类型的图表的最终表现形式都是在该对象进行一些属性的定制。JFreeChart 引擎本身提供了一个工厂类用于创建不同类型的图表对象
XXXXXDataset 数据集对象,用于提供显示图表所用的数据。根据不同类型的图表对应着很多类型的数据集对象类
XXXXXPlot 图表区域对象,基本上这个对象决定着什么样式的图表,创建该对象的时候需要 Axis、Renderer 以及数据集对象的支持
XXXXXAxis 用于处理图表的两个轴:纵轴和横轴
XXXXXRenderer 负责如何显示一个图表对象
XXXXXURLGenerator 用于生成 Web 图表中每个项目的鼠标点击链接
XXXXXToolTipGenerator 用于生成图象的帮助提示,不同类型图表对应不同类型的工具提示类

基本上我认为 JFreeChart 项目本身的类结构的设计并不是很好,首先在创建图表的时候用到了大量的工厂方法,这样做虽然可以简化创建图表对象的代码,但是对项目本身或者开发人员来讲 自行扩展一种新的图表都仍然是一件很麻烦的事情;其次除图表对象本身外其余的类过于复杂,使用者必须去了解每个类型的图表对象应该对应哪些 Axis、Plot、Renderer 类,并且必须非常熟悉这些类的构造函数中每个参数的具体含义。这些问题都大大困扰很多初学者。不过,虽然存在很多问题,但是 JFreeChart 本身仍不失为一个非常优秀的图表引擎,况且项目本身也在逐渐的发展中。

在非常简略的介绍了 JFreeChart 本身的代码结构后,下面我们开始动手试验几个常用的图表并把他们放到 web 上。

使用 JFreeChart 生成各种样式的图表

限于篇幅的问题我们在这里只实现两种常用的图表,其他类型图表读者可以触类旁通。我们先给出柱状图的实现,饼图的实现再来跟柱状图进行比较。

1 柱状图

 package lius.chart.demo; 
 import java.io.*; 
 import org.jfree.data.*; 
 import org.jfree.chart.*; 
 import org.jfree.chart.plot.*; 
 /** 
 * 该类用于演示最简单的柱状图生成
 * @author Winter Lau 
 */ 
 public class BarChartDemo { 
     public static void main(String[] args) throws IOException{ 
         CategoryDataset dataset = getDataSet2(); 
         JFreeChart chart = ChartFactory.createBarChart3D( 
                            "水果产量图", // 图表标题
                            "水果", // 目录轴的显示标签
                            "产量", // 数值轴的显示标签
                             dataset, // 数据集
                             PlotOrientation.VERTICAL, // 图表方向:水平、垂直
                             true,  // 是否显示图例(对于简单的柱状图必须是 false)
                             false, // 是否生成工具
                             false  // 是否生成 URL 链接
                             ); 
                            
         FileOutputStream fos_jpg = null; 
         try { 
             fos_jpg = new FileOutputStream("D:\\fruit.jpg"); 
             ChartUtilities.writeChartAsJPEG(fos_jpg,100,chart,400,300,null); 
         } finally { 
             try { 
                 fos_jpg.close(); 
             } catch (Exception e) {} 
         } 
     } 
     /** 
     * 获取一个演示用的简单数据集对象
     * @return 
     */ 
     private static CategoryDataset getDataSet() { 
         DefaultCategoryDataset dataset = new DefaultCategoryDataset(); 
         dataset.addValue(100, null, "苹果"); 
         dataset.addValue(200, null, "梨子"); 
         dataset.addValue(300, null, "葡萄"); 
         dataset.addValue(400, null, "香蕉"); 
         dataset.addValue(500, null, "荔枝"); 
         return dataset; 
     } 
     /** 
     * 获取一个演示用的组合数据集对象
     * @return 
     */ 
     private static CategoryDataset getDataSet2() { 
         DefaultCategoryDataset dataset = new DefaultCategoryDataset(); 
         dataset.addValue(100, "北京", "苹果"); 
         dataset.addValue(100, "上海", "苹果"); 
         dataset.addValue(100, "广州", "苹果"); 
         dataset.addValue(200, "北京", "梨子"); 
         dataset.addValue(200, "上海", "梨子"); 
         dataset.addValue(200, "广州", "梨子"); 
         dataset.addValue(300, "北京", "葡萄"); 
         dataset.addValue(300, "上海", "葡萄"); 
         dataset.addValue(300, "广州", "葡萄"); 
         dataset.addValue(400, "北京", "香蕉"); 
         dataset.addValue(400, "上海", "香蕉"); 
         dataset.addValue(400, "广州", "香蕉"); 
         dataset.addValue(500, "北京", "荔枝"); 
         dataset.addValue(500, "上海", "荔枝"); 
         dataset.addValue(500, "广州", "荔枝"); 
         return dataset; 
     } 
 } 

 

程序运行结束后生成的图片文件效果如下图所示:


图 4

如果是使用简单的数据即使用 getDataSet 方法获取数据集时产生的图片文件如下:


图 5

2 饼图

对于饼图而言,数据集的获取用的不是同一个数据集类,另外饼图不支持同一个类别的项目中还有子项目这样的数据。我们只给出创建饼图的代码,至于写图表到一个文件则与柱状图一致,无需重复。

 package lius.chart.demo; 
 import java.io.*; 
 import org.jfree.data.*; 
 import org.jfree.chart.*; 
 /** 
 * 用于演示饼图的生成
 * @author Winter Lau 
 */ 
 public class PieChartDemo { 
	 public static void main(String[] args) throws IOException{ 
		 DefaultPieDataset data = getDataSet(); 
		 JFreeChart chart = ChartFactory.createPie3DChart("水果产量图",  // 图表标题
		 data, 
		 true, // 是否显示图例
		 false, 
		 false 
		 ); 
		 // 写图表对象到文件,参照柱状图生成源码
	 } 
	 /** 
	 * 获取一个演示用的简单数据集对象
	 * @return 
	 */ 
	 private static DefaultPieDataset getDataSet() { 
		 DefaultPieDataset dataset = new DefaultPieDataset(); 
		 dataset.setValue("苹果",100); 
		 dataset.setValue("梨子",200); 
		 dataset.setValue("葡萄",300); 
		 dataset.setValue("香蕉",400); 
		 dataset.setValue("荔枝",500); 
		 return dataset; 
	 } 
 } 

 

生成的饼图文件效果如下:


图 6

将生成的图表移到浏览器上

为了将生成的图表直接传给客户端浏览器,只需要将前面两个例子中的文件流换成是通过 HttpServletResponse 对象获取到的输出流,详细代码清单如下:

 package lius.chart.demo; 
 import java.io.IOException; 
 import javax.servlet.*; 
 import javax.servlet.http.HttpServlet; 
 import org.jfree.data.*; 
 import org.jfree.chart.*; 
 /** 
 * 演示通过 servlet 直接输出图表
 * @author Winter Lau 
 */ 
 public class ChartDemoServlet extends HttpServlet { 
	 public void service(ServletRequest req, ServletResponse res) 
		 throws ServletException, IOException 
	 { 
		 res.setContentType("image/jpeg"); 
		 DefaultPieDataset data = getDataSet(); 
		 JFreeChart chart = ChartFactory.createPie3DChart("水果产量图", 
		 data, 
		 true, 
		 false, 
		 false 
		 ); 
				
		 ChartUtilities.writeChartAsJPEG(res.getOutputStream(), 
			 100,chart,400,300,null); 
	 } 
	 /** 
	 * 获取一个演示用的简单数据集对象
	 * @return 
	 */ 
	 private static DefaultPieDataset getDataSet() { 
		 DefaultPieDataset dataset = new DefaultPieDataset(); 
		 dataset.setValue("苹果",100); 
		 dataset.setValue("梨子",200); 
		 dataset.setValue("葡萄",300); 
		 dataset.setValue("香蕉",400); 
		 dataset.setValue("荔枝",500); 
		 return dataset; 
	 } 
 } 

 

高级主题

很多情况我们不仅仅要求可以在浏览器上显示一个图表,我们更需要客户可以直接在图表上做一下交互的操作,例如获取信息提示,点击图表某个部分进行更详细信 息的展示等等。例如前面生成的简单柱状图,用户需要在看到柱状图后点击某种水果例如是苹果即可看到各个地区苹果产量的情况。为此就要求该图形具有交互操作 的功能。在 HTML 中为了让一个图像具有可交互的功能就必须给该图像定义一个 Map 对象。下表节选一段具有该功能的 HTML 代码:

<MAP NAME="chartMap"> 
<AREA SHAPE="RECT" COORDS="81,15,126,254" href="?series=0&category=100" 
    title="100 = 7,048" 
    onclick="javascript:clickChart('100');return false;"> 
<AREA SHAPE="RECT" COORDS="143,27,188,255" href="?series=0&category=200" 
    title="200 = 6,721" 
    onclick="javascript: clickChart ('200');return false;"> 
<AREA SHAPE="RECT" COORDS="205,54,250,255" href="?series=0&category=300" 
    title="300 = 5,929" 
    onclick="javascript: clickChart ('300');return false;"> 
<AREA SHAPE="RECT" COORDS="267,85,312,255" href="?series=0&category=400" 
    title="400 = 5,005" 
    onclick="javascript: clickChart ('400');return false;"> 
<AREA SHAPE="RECT" COORDS="329,17,374,255" href="?series=0&category=Diet" 
    title="Diet = 7,017" onclick="javascript: 
    clickChart ('Diet');return false;"> 
</MAP> 

 

由此就产生了一个问题:如果根据一个图像来生成对应的 MAP 对象。我们回头看看刚才的代码,在创建一个图表对象时候有两个参数,我们举柱状图的例子来讲这两个参数就是 ChartFactory. createBarChart3D 方法中的最后两个参数,这两个参数的类型都是布尔值。这两个参数意思分别是:是否创建工具提示 (tooltip) 以及是否生成 URL。这两个参数分别对应着 MAP 中一个 AREA 的 title 属性以及 href 属性。

可是我想知道的是怎么来产生这个 MAP 啊!哈哈,不要着急,JFreeChart 已经帮我们做好生成 MAP 对象的功能。为了生成 MAP 对象就要引入另外一个对象:ChartRenderingInfo。因为 JFreeChart 没有直接的方法利用一个图表对象直接生成 MAP 数据,它需要一个中间对象来过渡,这个对象就是 ChartRenderingInfo。下图是生成 MAP 数据的流程图:


图 7

如上图所示,ChartUtilities 类是整个流程的核心,它周围的对象都是一些例如数据对象或者是文件等。这个流程简单描述如下:首先创建一个 ChartRenderingInfo 对象并在调用 ChartUtilities 的 writeChartAsJPEG 时作为最后一个参数传递进去。调用该方法结束后将产生一个图像文件以及一个填充好 MAP 数据的 ChartRenderingInfo 对象,有了这个对象我们还是没有办法获取具体的 MAP 数据,我们还必须借助于 ChartUtilities 的 writeImageMap 方法来将 ChartRenderingInfo 对象读取出来,获取 MAP 数据的代码片断如下:

		 PrintWriter w = null; 
		 FileOutputStream fos_jpg = null; 
		 FileOutputStream fos_cri = null; 
		 try{ 
			 // 根据不同类型的图表使用不同类,以下是针对饼图的操作
			 PiePlot plot = (PiePlot) chart.getPlot(); 
			 plot.setURLGenerator(new StandardPieURLGenerator(url)); 
			 // 设置工具提示
		   plot.setToolTipGenerator(new StandardPieToolTipGenerator()); 
			 fos_jpg = new FileOutputStream(“d:\\fruit.jpg”); 
			 ChartUtilities.writeChartAsJPEG( 
				 fos_jpg, 
				 100, 
				 chart, 
				 400, 
				 300, 
				 info); 
			 fos_cri = new FileOutputStream(__d:\\fruit.map__); 
			 w = new PrintWriter(fos_cri); 
			 ChartUtilities.writeImageMap(w, __mapname__, info); 
			 w.flush(); 
		 }finally{ 
			 try{ 
				 w.close(); 
			 }catch(Exception e){} 	
			 try{ 
				 fos_cri.close(); 
			 }catch(Exception e){} 	
			 try{ 
				 fos_jpg.close(); 
			 }catch(Exception e){} 
		 } 

 

打开文件 D:\fruit.map,文件的内容就是要写到页面上的 MAP 数据。把生成的图像文件以及 MAP 数据文件写到页面上即可完成热点图表的功能。至于怎么结合两者之间的关系例如图像的 useMap 属性值必须与 MAP 对象的名称结合起来,必须根据实际的应用情况进行相应的处理。笔者建议把二者通过标签库封装起来,图像文件的名称以及 MAP 对象的名称由标签库统一进行控制,这样可以保证二者的一致性。

 

参考资料

关于作者

刘冬,他已经不再叫凤梨罐头了。珠海市创我科技发展有限公司软件工程师,主要从事在J2EE平台上实现商业智能(BI)方面的开发。
联系电话:0756-3377435-383
电子邮件: winter.lau@163.com

分享到:
评论

相关推荐

    使用 JFreeChart来创建基于web的图表

    下面我们将详细探讨如何使用JFreeChart创建基于Web的图表。 首先,集成JFreeChart到Web项目中。你需要将JFreeChart的jar文件添加到项目的类路径中,这可以通过在构建路径中包含它或者将其放入WEB-INF/lib目录下完成...

    使用JFreeChart来创建基于web的图表.docx

    使用JFreeChart来创建基于web的图表 JFreeChart是一个开放源代码站点SourceForge.net上的一个JAVA项目,主要用来生成各种各样的图表,例如饼图、柱状图、线图、区域图、分布图、混合图、甘特图等等。下面将详细介绍...

    使用 JFreeChart来创建基于web的图表 WORD版

    本文档主要讲述的是使用 JFreeChart来创建基于web的图表;JFreeChart是开放源代码站点SourceForge.net上的一个JAVA项目,它主要用来各种各样的图表,这些图表包括:饼图、柱状图(普通柱状图以及堆栈柱状图)、线图、...

    使用 JFreeChart来创建基于web的图表_files

    在这个名为"使用 JFreeChart来创建基于web的图表_files"的压缩包中,很显然包含了与在Web环境中利用JFreeChart生成图表相关的资源。 1. **JFreeChart简介** JFreeChart是一个开源项目,它提供了一套完整的API,...

    使用JFreeChart来创建基于web的图表

    在本篇文章中,我们将深入探讨如何使用JFreeChart这一强大的Java图表库来创建基于Web的图表。通过细致地解析标题、描述以及部分已给出的内容,本文将为读者提供全面而深入的理解,帮助开发者掌握利用JFreeChart进行...

    JFreeChart -- web图表生成源码

    传统的一些以软件包形式发布应用程序例如报表系统等都在逐渐搬到因特网上。但是这两者之间有着天壤之别,虽然对于数据获取、业务处理等方面...因此我们下面将介绍一个JAVA的图表引擎JFreeChart用来产生基于WEB的图表。

    基于jfreechart-1.0.6的WEB图表开发示例

    ### 基于jfreechart-1.0.6的WEB图表开发示例 #### 一、概述 本文档旨在介绍如何使用jfreechart-1.0.6版本进行Web图表开发,帮助开发者掌握基本及高级图表绘制技巧。JFreeChart是一款强大的开源Java图表库,适用于...

    一种基于JFreeChart的Web统计图表.pdf

    ##### JFreeChart创建Web图表的基本原理 当客户端通过浏览器向Web服务器发起请求时,JSP页面将处理请求并调用后端的JavaBean或Servlet组件。这些组件根据前端提供的数据,在Web服务器端生成图表文件(通常是JPEG或...

    jfreeChart java web 图表包

    JFreeChart 是一个强大的 Java 库,专为创建高质量的图表而设计,尤其适用于 Java Web 应用程序。它提供了丰富的图表类型,包括曲线图、柱状图、饼状图、甘特图、雷达图和蜘蛛图等,满足了开发者在数据分析和可视化...

    jfreechart的WEB图表开发_____个人测试

    这篇个人测试的博客文章(尽管描述中未提供具体详情)很可能探讨了如何使用JFreeChart进行WEB图表的开发和实现。 首先,JFreeChart的使用通常涉及到以下几个步骤: 1. **引入依赖**:在项目中添加JFreeChart的库...

    jfreechart中文文档和jar包

    **使用 JFreeChart来创建基于web的图表.htm** 和 **使用 JFreeChart来创建基于web的图表.files**: 这部分内容可能涉及到如何将JFreeChart集成到Web应用程序中,比如使用Servlet或JSP来生成图表,并在网页上显示。这...

    JSP创建基于WEB的动态图表之开源代码篇源代码.rar_jsp_jsp动态_动态WEB_基于Web

    总的来说,这个项目提供了JSP创建基于Web的动态图表的实践案例,涵盖了JFreeChart和Cewolf这两个开源库的使用,以及如何在JSP中实现动态图表的生成和交互。通过学习和研究这些源代码,开发者可以掌握在Web应用中生成...

    纯Java动态生成SVG饼图与JFreeChart超强功能生成SVG图表

    使用 JFreeChart 生成 SVG 图表非常简单,只需要创建一个 `JFreeSVG` 对象,并将其添加到图表中即可。 ```java import org.jfree.chart.JFreeChart; import org.jfree.chart.plot.PlotOrientation; import org....

    JFreeChart 开发文档汇总

    jfreechart-1.0.0-demo.jar jfreechart-1.0.0.zip JFreeChart-API.chm JFreeChart -- 柱状图.mht JfreeChart.doc JFreeChart0.9.16 ...使用 JFreeChart来创建基于web的图表_IBM.mht 使用JFreeChart创建图象http.mht

    JFreechart-web图表书第4章

    通过本章的学习,读者将掌握如何在Tomcat上部署和发布基于JSP/Servlet的Web图表应用程序,为后续的JFreeChart应用打下基础。JFreechart是一个Java库,用于创建各种图表,如饼图、柱状图、线图等,适用于Web应用程序...

    基于JAVA的动态Web图表技术实现.pdf

    以柱状图为例,可以使用JFreeChart创建动态的Web图表。具体步骤如下: 1. **准备数据集**:首先需要创建一个Dataset对象来存储柱状图所需的数据。 2. **创建Plot对象**:根据Dataset创建相应的Plot对象,指定坐标轴...

Global site tag (gtag.js) - Google Analytics