JFreeChart笔记 (七) 时间序列图:值标记线

在一些比如说产量、销售额等的图表中常常会涉及到一个预期值或者目标值,这种值标记线如何在图表中画出来呢?这一节将教给大家值标记线的画法。

 

先看效果图:

 

TimeSeries06

 

实现代码如下,关键部分已标出:

package lw.release.s1TimeSeries;

import java.awt.Color;
import java.awt.Dimension;
import java.awt.Font;
import java.text.SimpleDateFormat;

import javax.swing.BorderFactory;
import javax.swing.JPanel;
import javax.swing.border.CompoundBorder;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.StandardChartTheme;
import org.jfree.chart.axis.DateAxis;
import org.jfree.chart.axis.DateTickUnit;
import org.jfree.chart.axis.DateTickUnitType;
import org.jfree.chart.plot.ValueMarker;
import org.jfree.chart.plot.XYPlot;
import org.jfree.chart.renderer.xy.XYItemRenderer;
import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer;
import org.jfree.data.time.Month;
import org.jfree.data.time.RegularTimePeriod;
import org.jfree.data.time.TimeSeries;
import org.jfree.data.time.TimeSeriesCollection;
import org.jfree.data.xy.XYDataset;
import org.jfree.ui.ApplicationFrame;
import org.jfree.ui.RectangleAnchor;
import org.jfree.ui.RefineryUtilities;
import org.jfree.ui.TextAnchor;

import lw.release.ChartBasePanel;

/**
 * 时间序列图:值标记线
 * 
 * 新增功能点:
 * 	 ① 图表中增加一条值标记线,可以作为销售额、产量等的目标值等
 * 
 * @author 刘伟  2012-10-26
 * 
 * 楼主辛勤整理,无私免费提供给大家观看,体惜楼主辛苦,转载时请注明出处:http://lw2078.iteye.com/ 
 * */
@SuppressWarnings("serial")
public class TimeSeries06 extends ApplicationFrame {

	public TimeSeries06(String title) {
		super(title);
		setContentPane(new TimeSeriesPanel());
	}

	public JPanel createDemoPanel() {
		return new TimeSeriesPanel();
	}

	public static void main(String[] arg) {
		TimeSeries06 timeSeries = new TimeSeries06("值标记线示例图");
		timeSeries.pack();
		RefineryUtilities.centerFrameOnScreen(timeSeries);
		timeSeries.setVisible(true);
	}

	/**
	 * 显示该Demo图表的容器
	 * 
	 * ChartBasePanel类是Swing框架下所有例子共同使用的,这里不重复贴出
	 * 这个类源码在:http://lw2078.iteye.com/blog/1705637  
	 * */
	private class TimeSeriesPanel extends ChartBasePanel {
		private TimeSeries series;	// 间隔定长时间(如年、月、日、时、分、秒等)的数据序列
		private ChartPanel chartPanel;	
		private JFreeChart chart = createChart();	// 创建一个JFreeChart时间序列图表

		public TimeSeriesPanel() {
			super();
			
			addChart(this.chart);	// 将此JFreeChart加入JFreeChart列表中
			
			// 将JFreeChart放在专用的图表容器ChartPanel中
			this.chartPanel = new ChartPanel(this.chart);
			this.chartPanel.setPreferredSize(new Dimension(600, 250));
			
			// 设置chartPanel容器边框
			CompoundBorder compoundBorder = BorderFactory.createCompoundBorder(
					BorderFactory.createEmptyBorder(4, 4,4, 4),
					BorderFactory.createEtchedBorder());
			this.chartPanel.setBorder(compoundBorder);
			
			// 将chartPanel加入到本容器中
			add(this.chartPanel);
		}

		/**
		 * 创建jfreechart图表
		 * */
		private JFreeChart createChart() {
			// 生成图表数据集合
			XYDataset xyDataset = createDataset(); 
			
			// 增加汉字支持
			StandardChartTheme standardChartTheme=new StandardChartTheme("CN");		//创建主题样式          
			standardChartTheme.setExtraLargeFont(new Font("隶书",Font.BOLD,20));    //设置标题字体       
			standardChartTheme.setRegularFont(new Font("SimSun",Font.PLAIN,15));    //设置图例的字体    
			standardChartTheme.setLargeFont(new Font("宋体",Font.PLAIN,15));      //设置轴向的字体   
			ChartFactory.setChartTheme(standardChartTheme); //应用主题样式 	
			
			// 创建一个时间序列图表的JFreeChart
			JFreeChart jFreeChart = ChartFactory.createTimeSeriesChart(
					"值标记线示例图", 	// 图表名
					"时间", 				// 横轴标签文字
					"数值", 				// 纵轴标签文字
					xyDataset,			// 图表的数据集合
					true, 				// 是否显示图表中每条数据序列的说明
					false, 				// 是否显示工具提示
					false);				// 是否显示图表中设置的url网络连接
		
			// XYPlot图表区域的设置对象,用来设置图表的一些显示属性
			XYPlot xyPlot = (XYPlot) jFreeChart.getPlot();	
	        
			// 设置数据点和序列线的显示格式
	        XYItemRenderer r = xyPlot.getRenderer();
	        if (r instanceof XYLineAndShapeRenderer) {
	            XYLineAndShapeRenderer renderer = (XYLineAndShapeRenderer) r;
	            renderer.setBaseShapesVisible(true);	// 数据点显示外框
	            renderer.setBaseShapesFilled(true);		// 数据点外框内填充
	        }
	        
//  新功能点  
	        // 设置值标记线
		    ValueMarker valueMarker = new ValueMarker(100.0D);
		  
		    valueMarker.setPaint(Color.blue);	// 值标记线颜色
		    valueMarker.setAlpha(0.9F);			// 值标记线透明度
		    valueMarker.setLabel("目标值");		// 值标记线显示的文字
		    valueMarker.setLabelPaint(Color.BLUE);	// 值标记线显示的文字的颜色
		    valueMarker.setLabelFont(new Font("宋体",Font.PLAIN,12));// 值标记线显示的文字的字体
		    valueMarker.setLabelAnchor(RectangleAnchor.LEFT);	// 值标记线显示的文字定位到最左端的数据点处
		    valueMarker.setLabelTextAnchor(TextAnchor.BOTTOM_LEFT); // 值标记线在显示的文字的下方左端

		    xyPlot.addRangeMarker(valueMarker);	// 在图表中使用自定义的值标记线
//
			
			// 设置X时间轴按月显示,时间间隔为1个月
			DateAxis dateAxis=(DateAxis)xyPlot.getDomainAxis();		// DateAxis是X时间轴线的显示样式设置对象
			SimpleDateFormat frm = new SimpleDateFormat("MM月");	// 设置时间显示样式
			dateAxis.setTickUnit(new DateTickUnit(DateTickUnitType.MONTH, 1, frm));  // 设置显示时间间隔为1年
						
			return jFreeChart;
		}

		/**
		 * 创建jfreechart图表所用的数据集合
		 * 
		 * @return
		 */
		private XYDataset createDataset() {

			// 生成数据序列
			this.series = new TimeSeries("产量");	
			setSeriesData(series, 100, new Month(1,2011), 12); // 以月为时间单位,从2011年1月开始,随机产生12个月的模拟数据

			// 将两条数据序列都放在一个数据集合中
			TimeSeriesCollection dataset = new TimeSeriesCollection();
			dataset.addSeries(this.series);	
			
			return dataset;
		}

		/**
		 * 随机生成数据,自动定位到时间序列上的下一个时间点,将新数据点加入到数据序列中
		 *
		 * @param series	数据序列对象
		 * @param baseData	生成的随机数据的基准值
		 * @param regularTime	定长的时间间隔(年、月、日、时、分、秒等)
		 * @param sampleNum  生成的数据点个数
		 */
		private void setSeriesData(TimeSeries series, double baseData, RegularTimePeriod regularTime, int sampleNum) {

			// 生成随机模拟数据
			double value = baseData;
			for (int i = 0; i < sampleNum; i++) {
				series.add(regularTime, value);		
				regularTime = regularTime.next();	//自动定位到下一个时间点
				value *= (1.0D + (Math.random() - 0.495D) / 4.0D);
			}
		}
	}
	
	
}
 
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值