当前位置:编程学习 > JSP >>

jsp中利用jfreechart实现统计效果图

1 preparation
1.1 sample introduction
JFreeChart是JAVA平台上的一个开放的图表绘制类库。I完全使用JAVA语言编写,是为applications, applets, servlets 以及JSP等使用所设计。JFreeChart可生成饼图(pie charts)、柱状图(bar charts)、散点图(scatter plots)、时序图(time series)、甘特图(Gantt charts)等等多种图表,并且可以产生PNG和JPEG格式的输出,还可以与PDF和EXCEL关联。JFreeChart截止2011年2月22日为止的相当不错的java图形解决方案,基本能够解决目前的图形方面的需求。

1.2 download
下载地址:http://sourceforge.net/project/showfiles.php?group_id=15494

说明:下载后介绍请注意以下三个:1)source目录-存放jfreechart的源码 2) lib目录-存放jar文件 3)jfreechart-1.0.14-demo.jar 例子程序

1.3 notes
开发环境:

System:xp    JDK:1.5  Tomcat:5.X  Myeclipse:6.5

项目环境:

web工程的目录结构如下:

image

没有去细分每个jar是做什么的,就把所有的jfreechart的lib目录下所有的jar都导入了工程中,本次用的jfreechart的版本是jfreechart-1.0.14)

    bar是条形图存放的文件夹,line是曲线文件存放的文件夹,pie是饼图文件存放的文件夹,曲线和饼图文件在后面介绍,在本项目中,条形图我用java代码来书写,曲线图我用GUI来书写,饼图我用jsp来书写。

2 start
web.xml
这里只贴出公共部分,后面的介绍需要修改此配置文件的,我会另外的说明给出

 代码如下 复制代码
<?xml version="1.0" encoding="UTF-8"?>
 <web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
     xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
 http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
 
     <!-- 增加的jfreechart的处理类 -->
     <servlet>
         <servlet-name>DisplayChart</servlet-name>
         <servlet-class>
             org.jfree.chart.servlet.DisplayChart
         </servlet-class>
         <init-param>
             <!-- 解决可能存在的servlet中文乱码问题 -->
             <param-name>encoding</param-name>
             <param-value>GBK</param-value>
         </init-param>
     </servlet>
     <servlet-mapping>
         <servlet-name>DisplayChart</servlet-name>
         <url-pattern>/DisplayChart</url-pattern>
     </servlet-mapping>
 
 </web-app>

下面来介绍用java文件书写的条形图(我用的是servlet实现,为了演示效果,我条形图放了四个实现不同效果条形图的方法)

2.1 在配置文件中加servlet文件的配置
<!-- 为了把新建图的代码写在java代码中增加的一个servlet -->

 代码如下 复制代码

<servlet>
    <servlet-name>BarServlet</servlet-name>
    <servlet-class>bar.BarServlet</servlet-class>
    <init-param>
        <!-- 解决可能存在的servlet中文乱码问题 -->
        <param-name>encoding</param-name>
        <param-value>GBK</param-value>
    </init-param>
</servlet>
<servlet-mapping>
    <servlet-name>BarServlet</servlet-name>
    <url-pattern>/BarServlet</url-pattern>
</servlet-mapping>

2.2 BarServlet.java

 代码如下 复制代码
package bar;
 
 import org.jfree.chart.ChartFactory;
 import org.jfree.chart.ChartRenderingInfo;
 import org.jfree.chart.JFreeChart;
 import org.jfree.chart.labels.ItemLabelAnchor;
 import org.jfree.chart.labels.ItemLabelPosition;
 import org.jfree.chart.labels.StandardCategoryItemLabelGenerator;
 import org.jfree.chart.plot.PlotOrientation;
 import org.jfree.chart.renderer.category.BarRenderer3D;
 import org.jfree.chart.servlet.ServletUtilities;
 import org.jfree.data.category.CategoryDataset;
 import org.jfree.data.category.DefaultCategoryDataset;
 import org.jfree.data.general.DatasetUtilities;
 import org.jfree.ui.TextAnchor;
 import org.jfree.chart.ChartUtilities;
 import java.awt.BasicStroke;
 import java.awt.Color;
 import java.awt.Font;
 import java.io.IOException;
 import java.io.PrintWriter;
 import org.jfree.chart.title.TextTitle;
 import org.jfree.chart.axis.AxisLocation;
 import org.jfree.chart.axis.CategoryLabelPositions;
 import org.jfree.chart.axis.NumberAxis;
 import org.jfree.chart.axis.CategoryAxis;
 import org.jfree.chart.axis.ValueAxis;
 import org.jfree.chart.entity.StandardEntityCollection;
 import org.jfree.chart.plot.CategoryPlot;
 import javax.servlet.ServletOutputStream;
 import javax.servlet.http.HttpServlet;
 import javax.servlet.http.HttpServletRequest;
 import javax.servlet.http.HttpServletResponse;
 import javax.servlet.http.HttpSession;
 
 /**
 *
 * Module: BarServlet.java
 * Description: 条形图用java代码来书写
 * Company:
 * Asiainfo Author: pan易做图eng
 * Date: Dec 15, 2011
 */
 
 public class BarServlet extends HttpServlet {
 
     private static final long serialVersionUID = 1L;
     // 访问此servlet的url为:http://localhost:8080/JFreeChart/BarServlet?num=2
     public void doGet(HttpServletRequest request, HttpServletResponse response)
             throws IOException {
         //获得回话的session对象,ServletUtilities类的saveChartAsPNG方法需要用时把session放进去
 //HttpSession session = request.getSession();
         /**解决servlet输出的中文乱码问题,这里的乱码和条形图中的乱码可是两种不同的乱码
          * 如果使用response.getWriter()方法获得对象,则setContentType方法必须放在获得out对象之前
          * 如果使用response.getOutputStream()方法获得对象,则setContentType方法放在获得out对象之前或之后都可以
          * */
         response.setContentType("text/html; charset=GBK");
         PrintWriter out = response.getWriter();
         //ServletOutputStream out = response.getOutputStream();
 
 //response.setCharacterEncoding("GBK");
 
         String num = request.getParameter("num");//获得请求url的参数,用来判断是显示第几个条形图示例
 
         /*
          * 为了少新建文件,也为了少配置web.xml文件,把条形图的介绍都放在一个servlet中了,
          * 每个if分支都是独立的,调用四个不同的方法,实现四个不同效果的条形图,从1到4,功能依次增强
 */
          String filename ="";//定义一个公共变量,保存生成的名称
         if (num.equals("1")) {
             out.print("<b>第一种情况,一个简单的条形图</b><br><h1></h1>");
             filename = bar1();
         } else if (num.equals("2")) {
             out.print("<b>第二种情况,各种不同颜色的条形图</b><br><h1></h1>");
             filename = bar2();
         } else if (num.equals("3")) {
             out.print("<b>第三种情况,多个条形图对比</b><br><h1></h1>");
             filename = bar3();
         } else if (num.equals("4")) {
             out.print("<b>第四种情况,多个条形图对比并且每个条形图上面加上数字</b><br><h1></h1>");
             filename = bar4();
         } else {
             out.println("<b>输入的url不对,请输入1、2、3、4 来查看不同的条形图</b><br><h1></h1>");
         }
         String graphURL = request.getContextPath() + "/DisplayChart?filename="
                 + filename;// 调用jfreechart的处理类
 // 条形图的输出
 //---------------start---------------
         out.println("<html>");
         out.println("<body>");
         out
                 .println("<img src=""
                         + graphURL
                         + "" width=600 height=400 border=0 usemap="#"
                         + filename + ""/>");
         out.println("</body>");
         out.println("</html>");
         //---------------end---------------
         out.flush();//刷新该流的缓冲
         out.close();//关闭该流并释放与之关联的所有系统资源
     }
     //加上接受post请求的处理方法,实质也还是调用doGet方法来处理
     public void doPost(HttpServletRequest request, HttpServletResponse response)
             throws IOException {
         this.doGet(request, response);
     }
     //条形图示例1 一个简单的条形图
     public String bar1() throws IOException{
         DefaultCategoryDataset dataset = new DefaultCategoryDataset();
         // 设置填充数据
         dataset.addValue(25, "襄阳", "襄城");
         dataset.addValue(20, "襄阳", "樊城");
         dataset.addValue(15, "襄阳", "襄州");
         dataset.addValue(10, "襄阳", "东津");
         dataset.addValue(5, "襄阳", "鱼梁州");
 
         /**
          * ChartFactory类的createBarChart3D方法参数介绍:
          * createBarChart3D
          * (
          * String title, 图表标题
          * String categoryAxisLabel, 统计种类轴标题,可以理解为X轴标题
          * String valueAxisLabel,统计值轴标题,可以理解为y轴标题
          * CategoryDataset dataset, 绘图数据集
          * PlotOrientation orientation, 用于设置柱形图的绘制方向,PlotOrientation.VERTICAL(垂直),PlotOrientation.HORIZONTAL(水平)
          * boolean legend, 用于设定是否显示图例
          * boolean tooltips, 用于设定是否采用标准生成器
          * boolean urls 用于设置定否包生成链接
          * )
 */
         JFreeChart chart = ChartFactory.createBarChart3D("襄阳城区人口统计", "地区",
                 "人口数(单位:万)", dataset, PlotOrientation.VERTICAL, true,
                 false, false);
 
         // 设置主标题指定字体,解决中文乱码
         Font font = new Font("宋体", Font.BOLD, 20);
         TextTitle title = new TextTitle("襄阳城区人口统计(主标题)", font);
         chart.setTitle(title);
         /* 以上三句和下面的一句效果是一样的 */
         // chart.setTitle(new TextTitle("襄阳城区人口统计(主标题)",new
 // Font("宋体",Font.BOLD,20)));
         CategoryPlot plot = chart.getCategoryPlot();
         //获得横轴对象,并设置相关的绘图属性
         CategoryAxis domainAxis = plot.getDomainAxis();
         domainAxis.setAxisLineStroke(new BasicStroke(1.6f));  // 设置轴线粗细
         domainAxis.setAxisLinePaint(Color.BLACK);             //设置轴线颜色
         domainAxis.setCategoryLabelPositionOffset(5);         //设置统计种类与轴线的颜色
         domainAxis.setLabelPaint(Color.BLACK);                //设置坐标轴标题颜色
         domainAxis.setCategoryLabelPositions(CategoryLabelPositions.STANDARD);// 设置坐标轴标题旋转角度
         /*------设置X轴坐标上的文字,解决中文乱码-----------*/
         /* 本例指的是襄城、樊城、襄州、东津、鱼梁州这些字 */
         domainAxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN, 15));
         /*------设置X轴的标题文字,解决中文乱码------------*/
         /* 本例指的是"地区"两个字 */
         domainAxis.setLabelFont(new Font("宋体", Font.PLAIN, 15));
 
         //获得纵轴对象,并设置相关的绘图属性
         ValueAxis rangeAxis = plot.getRangeAxis();
         rangeAxis.setAxisLineStroke(new BasicStroke(1.6f));       //设置轴线粗细
         rangeAxis.setAxisLinePaint(Color.RED);                    // 设置轴线颜色
         rangeAxis.setUpperBound(30.0f);                           // 设置坐标最大值
         rangeAxis.setTickMarkStroke(new BasicStroke(1.6f));       //设置坐标标记大小
         rangeAxis.setTickMarkPaint(Color.BLACK);                  // 设置坐标标记颜色
         rangeAxis.setLabelPaint(Color.BLACK);                     //设置坐标轴标题颜色
         rangeAxis.setLabelAngle(Math.PI / 2);                     //设置坐标轴标题旋转角度   
         rangeAxis.setLabelFont(new Font("黑体", Font.PLAIN, 15)); //设置Y轴的标题文字,解决中文乱码
         rangeAxis.setUpperMargin(0.15);                           //设置最高一个柱与图片顶端的距离
         rangeAxis.setLowerMargin(0.15);                           //设置最低的一个柱与图片底端的距离
 
 //也可以通过NumberAxis的方法解决y轴中文乱码问题       
         /*------设置Y轴的标题文字,解决中文乱码------------*/
         // 本例中指的是"销量"两个字
 //NumberAxis numberaxis = (NumberAxis) plot.getRangeAxis();
 //numberaxis.setLabelFont(new Font("黑体", Font.PLAIN, 15));
         /*------设置Y轴坐标上的文字,解决中文乱码------------*/
         /* 本例中指的是"0,25,50..."这些值,因为本来这些就是数值,所以有没有此句都可以 */
         // numberaxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN,
 // 15));
 
         /*------解决底部汉字乱码的问题,如过程序没有底部,下面的一行代码如果释放的话会报空指针的错误-----------*/
         chart.getLegend().setItemFont(new Font("宋体", Font.PLAIN, 15));//显示的是dataset.addValue中的值"襄阳"
         /**
          * ServletUtilities类的saveChartAsPNG方法参数介绍:
          * saveChartAsPNG(
          * JFreeChart chart, chart对象
          * int width, 图片的宽度
          * int height, 图片的高度
          * ChartRenderingInfo info,info对象,可以为null
          * HttpSession session session对象,可以为null
          * )
 */
 
         String filename = ServletUtilities
                 .saveChartAsPNG(chart, 600, 400, null);
         /**
          * 使用writeImageMap方法输出图片 <?暂时还有问题,后期会修改?>
         ChartRenderingInfo info=new ChartRenderingInfo(new StandardEntityCollection());
         ChartUtilities.writeImageMap(out, filename, info, true);
 */
         return filename;
     }
     //条形图示例2 一个不同颜色的条形图
     public String bar2() throws IOException {
         double[][] data = new double[][] { {25},{ 20 }, { 15 }, { 10 },
                 { 5 } };
         String[] rowKeys = { "襄城", "樊城", "襄州", "东津","鱼梁州" };
         String[] columnKeys = { "" };
         CategoryDataset dataset = DatasetUtilities.createCategoryDataset(
                 rowKeys, columnKeys, data);
         JFreeChart chart = ChartFactory
                 .createBarChart3D("襄阳城区人口统计", "地区", "人口", dataset,
                         PlotOrientation.VERTICAL, true, false, false);
         //设置主标题指定字体,解决中文乱码
         Font font = new Font("宋体", Font.BOLD, 16);
         TextTitle title = new TextTitle("襄阳城区人口统计(主标题)", font);
         chart.setTitle(title);
 
         CategoryPlot plot = chart.getCategoryPlot();
         NumberAxis numberaxis = (NumberAxis) plot.getRangeAxis();
         CategoryAxis domainAxis = plot.getDomainAxis();
         /*------设置X轴坐标上的文字-----------*/
         //domainAxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN, 11));
         /*------设置X轴的标题文字------------*/
         domainAxis.setLabelFont(new Font("宋体", Font.PLAIN, 12));
         /*------设置Y轴坐标上的文字-----------*/
         //numberaxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN, 12));
         /*------设置Y轴的标题文字------------*/
         numberaxis.setLabelFont(new Font("黑体", Font.PLAIN, 12));
 
         /*------解决底部汉字乱码的问题-----------*/
         chart.getLegend().setItemFont(new Font("宋体", Font.PLAIN, 12));
         String filename = ServletUtilities.saveChartAsPNG(chart, 500, 300,
                 null);
         return filename;
     }
 
     //条形图示例3 多个条形图对比
     public String bar3() throws IOException {
         double[][] data = new double[][] { { 1310, 1220, 1110, 1000 ,666},
                 { 720, 700, 680, 640 ,777}, { 1130, 1020, 980, 800 ,888},
                 { 440, 400, 360, 300 ,999} ,{400,400,400,400,555}};
 
         String[] rowKeys = { "猪肉", "牛肉", "鸡肉", "鱼肉","羊肉" };
         String[] columnKeys = { "襄城", "樊城", "襄州", "东津","鱼梁州" };
         CategoryDataset dataset = DatasetUtilities.createCategoryDataset(
                 rowKeys, columnKeys, data);
         JFreeChart chart = ChartFactory
                 .createBarChart3D("襄阳各区肉类销量统计图", "地区", "销量", dataset,
                         PlotOrientation.VERTICAL, true, false, false);
         //设置主标题指定字体,解决中文乱码
         Font font = new Font("宋体", Font.BOLD, 16);
         TextTitle title = new TextTitle("肉类销量统计图(主标题)", font);
         chart.setTitle(title);
 
         CategoryPlot plot = chart.getCategoryPlot();
         NumberAxis numberaxis = (NumberAxis) plot.getRangeAxis();
         CategoryAxis domainAxis = plot.getDomainAxis();
         /*------设置X轴坐标上的文字-----------*/
         domainAxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN, 11));
         /*------设置X轴的标题文字------------*/
         domainAxis.setLabelFont(new Font("宋体", Font.PLAIN, 12));
         /*------设置Y轴坐标上的文字-----------*/
         //numberaxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN, 12));
         /*------设置Y轴的标题文字------------*/
         numberaxis.setLabelFont(new Font("黑体", Font.PLAIN, 12));
 
         /*------解决底部汉字乱码的问题-----------*/
         chart.getLegend().setItemFont(new Font("宋体", Font.PLAIN, 12));
         String filename = ServletUtilities
                 .saveChartAsPNG(chart, 500, 300, null);
         return filename;
     }
 
     //条形图示例4 多个条形图对比并且每个条形图上面加上数字
     public String bar4() throws IOException {
         double[][] data = new double[][] { { 1310, 1220, 1110, 1000 ,666},
                 { 720, 700, 680, 640 ,777}, { 1130, 1020, 980, 800,888 },
                 { 440, 400, 360, 300 ,999} ,{400,400,400,400,555}};
 
         String[] rowKeys = { "猪肉", "牛肉", "鸡肉", "鱼肉" ,"羊肉"};
         String[] columnKeys = { "襄城", "樊城", "襄州", "东津","鱼梁州" };
         CategoryDataset dataset = DatasetUtilities.createCategoryDataset(
                 rowKeys, columnKeys, data);
         JFreeChart chart = ChartFactory.createBarChart3D("襄阳各区肉类销量统计图", "地区",
                 "销量", dataset, PlotOrientation.VERTICAL, true, true, false);
 
         CategoryPlot plot = chart.getCategoryPlot();
 
         //设置网格背景颜色
         plot.setBackgroundPaint(Color.white);
         //设置网络竖线颜色
         plot.setDomainGridlinePaint(Color.pink);
         //显示每个柱的数值,并修改该数字的字体属性
         BarRenderer3D renderer = new BarRenderer3D();
         renderer
                 .setBaseItemLabelGenerator(new StandardCategoryItemLabelGenerator());
         renderer.setBaseItemLabelsVisible(true);
         //默认的数字显示在柱子中,通过以下两句调整数字的显示
 //注意,此句很关键,若无此句,那数字的显示会覆盖,给人数字没有显示出来的问题
         renderer.setBasePositiveItemLabelPosition(new ItemLabelPosition(
                 ItemLabelAnchor.OUTSIDE12, TextAnchor.BASELINE_LEFT));
         renderer.setItemLabelAnchorOffset(10D);
         //设置每个地区所包含的平行柱的之间的距离
 //renderer.setItemMargin(0.3);
         plot.setRenderer(renderer);
 
         //设置地区、销量的显示位置
 //将下方的“肉类”放到上方
         plot.setDomainAxisLocation(AxisLocation.TOP_OR_RIGHT);
         //将默认的左边的“销量”放到右方
         plot.setRangeAxisLocation(AxisLocation.BOTTOM_OR_RIGHT);
         NumberAxis numberaxis = (NumberAxis) plot.getRangeAxis();
         CategoryAxis domainAxis = plot.getDomainAxis();
         //图表标题以及副标题乱码
         Font font = new Font("宋体", Font.BOLD, 16);
         TextTitle title = new TextTitle("襄阳", font);//副标题
         TextTitle subtitle = new TextTitle("肉类销量统计图", new Font("黑体", Font.BOLD,
                 12));
         chart.addSubtitle(subtitle);//子标题
         chart.setTitle(title); //标题
 
 //X轴乱码
 //X轴坐标上的文字:
         domainAxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN, 11));
         //X轴坐标标题(肉类)
         domainAxis.setLabelFont(new Font("宋体", Font.PLAIN, 12));
         //Y轴坐标上的文字
         numberaxis.setTickLabelFont(new Font("sans-serif", Font.PLAIN, 12));
         //Y轴坐标标题(销量):
         numberaxis.setLabelFont(new Font("黑体", Font.PLAIN, 12));
         //图表底部乱码(猪肉等文字)
         chart.getLegend().setItemFont(new Font("黑体", Font.PLAIN, 12));
         String filename = ServletUtilities
                 .saveChartAsPNG(chart, 700, 400, null);
         return filename;
     }
 }

bar1的运行截图:

访问的URL为:http://localhost:8080/JFreeChart/BarServlet?num=1

image

bar2的运行截图:

访问的URL为:http://localhost:8080/JFreeChart/BarServlet?num=2

image

bar3的运行截图:

访问的URL为:http://localhost:8080/JFreeChart/BarServlet?num=3

image

bar4的运行截图:

访问的URL为:http://localhost:8080/JFreeChart/BarServlet?num=4

image

补充:Jsp教程,Java技巧及代码
CopyRight © 2022 站长资源库 编程知识问答 zzzyk.com All Rights Reserved
部分文章来自网络,