我正在使用JFreeChart绘制图表。我有XYSeries,其点为(0,0),(1、2),(2、5),并且我想读取Y值,例如x = 1.5。

是否可以读取不在XYSeries中的点的值?我找不到类似的话题。

最佳答案

不直接支持此功能。在许多情况下,这没有任何意义:x = 1.5根本没有可用数据。那里的值可能是1000.0或-3.141。你不知道

但是,您很可能正在寻找线性插值。因此,实用的方法是找到包含各个x值的区间,并对y值进行线性插值。

有一些技术上的警告。例如。 XYSeries可能未排序,或者可能包含重复的x值,在这种情况下,给定x值没有唯一的y值。但是现在,我们可以假定数据集不具有这些属性。

以下是如何实现此示例。请注意,这不是很有效。如果必须计算许多中间值(也就是说,如果您打算经常调用interpolate方法),则创建一个基于树的数据结构以允许在O(logn)中查找间隔将是有益的。

但是,如果这不是时间紧迫的情况(例如,如果您仅打算在工具提示中显示该值),则可以插值如下:

import java.util.List;

import org.jfree.data.xy.XYDataItem;
import org.jfree.data.xy.XYSeries;

public class XYInterpolation
{
    public static void main(String[] args)
    {
        XYSeries s = new XYSeries("Series");

        s.add(0,0);
        s.add(1,2);
        s.add(2,5);

        double minX = -0.5;
        double maxX = 3.0;
        int steps = 35;
        for (int i=0; i<=steps; i++)
        {
            double a = (double)i / steps;
            double x = minX + a * (maxX - minX);
            double y = interpolate(s, x);
            System.out.printf("%8.3f : %8.3f\n", x, y);
        }
    }

    private static double interpolate(XYSeries s, double x)
    {
        if (x <= s.getMinX())
        {
            return s.getY(0).doubleValue();
        }
        if (x >= s.getMaxX())
        {
            return s.getY(s.getItemCount()-1).doubleValue();
        }
        List<?> items = s.getItems();
        for (int i=0; i<items.size()-1; i++)
        {
            XYDataItem i0 = (XYDataItem) items.get(i);
            XYDataItem i1 = (XYDataItem) items.get(i+1);
            double x0 = i0.getXValue();
            double y0 = i0.getYValue();
            double x1 = i1.getXValue();
            double y1 = i1.getYValue();

            if (x >= x0 && x <= x1)
            {
                double d = x - x0;
                double a = d / (x1-x0);
                double y = y0 + a * (y1 - y0);
                return y;
            }
        }
        // Should never happen
        return 0;
    }

}


(此实现限制了限制。这意味着,对于小于最小x值或大于最大x值的x值,将分别返回最小/最大x值的y值。 )

09-05 10:32