3

我试图在相同的数据上有两个轴。

数据是几个DefaultTableXYDatasets。情节是一个XYPlot,我有两个XYLineAndShapeRenderers和一个StackedXYAreaRenderer2

y 值的所有数据都以米为单位,我希望一个轴以米为单位显示它,一个轴以英尺为单位显示它。现在这感觉像是一件很常见的事情,但我无法决定最明显的方法。一种可行的方法是复制数据并以英尺为单位的 y 值,然后添加另一个NumberAxis并完成它。

但我认为子类化NumberAxis或注入一些功能NumberAxis来缩放值会更明智。还是我应该采用第一种方法?

你怎么看?

4

2 回答 2

3

为避免重复数据,您可以使用该XYPlot方法mapDatasetToRangeAxes()将数据集索引映射到轴索引列表。在下面的示例中,meters是主轴,相应feet轴的范围也相应缩放,如下所示。请注意,invokeLater()需要确保在轴发生任何变化feet对轴进行缩放。meters

图片

import java.awt.EventQueue;
import java.util.Arrays;
import java.util.List;
import javax.swing.JFrame;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.axis.AxisLocation;
import org.jfree.chart.axis.NumberAxis;
import org.jfree.chart.axis.ValueAxis;
import org.jfree.chart.event.AxisChangeEvent;
import org.jfree.chart.plot.XYPlot;
import org.jfree.chart.renderer.xy.XYItemRenderer;
import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer;
import org.jfree.data.xy.XYDataset;
import org.jfree.data.xy.XYSeries;
import org.jfree.data.xy.XYSeriesCollection;

/**
 * @see https://stackoverflow.com/q/13358758/230513
 */
public class AxisTest {

    private static final int N = 5;
    private static final double FEET_PER_METER = 3.28084;

    private static XYDataset createDataset() {
        XYSeriesCollection data = new XYSeriesCollection();
        final XYSeries series = new XYSeries("Data");
        for (int i = -N; i < N * N; i++) {
            series.add(i, i);
        }
        data.addSeries(series);
        return data;
    }

    private JFreeChart createChart(XYDataset dataset) {
        NumberAxis meters = new NumberAxis("Meters");
        NumberAxis feet = new NumberAxis("Feet");
        ValueAxis domain = new NumberAxis();
        XYItemRenderer renderer = new XYLineAndShapeRenderer();
        XYPlot plot = new XYPlot(dataset, domain, meters, renderer);
        plot.setRangeAxis(1, feet);
        plot.setRangeAxisLocation(1, AxisLocation.BOTTOM_OR_LEFT);
        List<Integer> axes = Arrays.asList(0, 1);
        plot.mapDatasetToRangeAxes(0, axes);
        scaleRange(feet, meters);
        meters.addChangeListener((AxisChangeEvent event) -> {
            EventQueue.invokeLater(() -> {
                scaleRange(feet, meters);
            });
        });
        JFreeChart chart = new JFreeChart("Axis Test",
            JFreeChart.DEFAULT_TITLE_FONT, plot, true);
        return chart;
    }

    private void scaleRange(NumberAxis feet, NumberAxis meters) {
        feet.setRange(meters.getLowerBound() * FEET_PER_METER,
            meters.getUpperBound() * FEET_PER_METER);
    }

    private void display() {
        JFrame f = new JFrame("AxisTest");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(new ChartPanel(createChart(createDataset())));
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(() -> {
            new AxisTest().display();
        });
    }
}

或者,您可以使用 aJCheckBox在米和英尺这两个系列之间切换,如相关示例所示。使用 可用的方法XYLineAndShapeRenderer,您可以隐藏第二个系列的线条、形状和图例。系列本身必须可见以建立轴范围。

于 2012-11-13T14:46:32.487 回答
3

最终我选择了这个解决方案,它可能不是最优雅的,但它确实有效。我有第二个轴feetAxis,并AxisChangeListener在第一个轴上添加了一个名为meterAxis. 当meterAxis更改将范围设置为feetAxis.

我使用SwingUtilities.invokeLater了 ,否则缩小图表时范围会不正确,那么feetAxis只会从 0 变为 1。虽然没有检查原因。

feetAxis = new NumberAxis("Height [ft]");
metersAxis = new NumberAxis("Height [m]");
pathPlot.setRangeAxis(0, metersAxis);
pathPlot.setRangeAxis(1, feetAxis);

metersAxis.addChangeListener(new AxisChangeListener() {

   @Override
   public void axisChanged(AxisChangeEvent event) {

       SwingUtilities.invokeLater(new Runnable() {

           @Override
           public void run() {
               feetAxis.setRange(metersAxis.getLowerBound() * MetersToFeet, metersAxis.getUpperBound() * MetersToFeet);
                }
           });
       }
   });
于 2012-11-13T15:19:03.973 回答