从 CSV 文件中读取数据到 ArrayList 并显示在 XY Chart 中
Read data from CSV file into ArrayList and display in XY Chart
我想显示随时间变化的温度曲线。我现在已经阅读了一个类似于 CSV 的文件,它让我知道了指示的时间和温度。现在我想使用 JFreeChart 显示值的 XY 图。
文件原始数据如下:
utc,local,celsius
2017-07-12T07:02:53+00:00,2017-07-12T09:02:53+02:00,26.25
2017-07-12T08:02:54+00:00,2017-07-12T10:02:54+02:00,26.08
2017-07-12T09:02:55+00:00,2017-07-12T11:02:55+02:00,25.78
2017-07-12T10:02:56+00:00,2017-07-12T12:02:56+02:00,25.96
2017-07-12T10:51:02+00:00,2017-07-12T12:51:02+02:00,26.14
2017-07-12T10:51:02+00:00,2017-07-12T12:51:02+02:00,26.14
时间和温度值的输出(我已经从原始文件中分离出来)看起来像:
09:02:53,26.25
10:02:54,26.08
11:02:55,25.78
12:02:56,25.96
12:51:02,26.14
12:51:02,26.14
编辑:
现在我在 Trashgod 的示例中插入了一个 DateTimeFormatter:
看起来像:
public class Test {
public static void main(String[] args) {
EventQueue.invokeLater(() -> {
ApplicationFrame frame = new ApplicationFrame("CSVTest");
Test test = new Test();
frame.add(test.createChart("Temperature profile"));
frame.pack();
frame.setLocationRelativeTo(null);;
frame.setVisible(true);
});
}
private ChartPanel createChart(String chartTitle) {
JFreeChart chart = ChartFactory.createTimeSeriesChart(chartTitle,
"Time", "Temperature", createDataset(), true, true, false);
ChartPanel chartPanel = new ChartPanel(chart);
XYPlot plot = chart.getXYPlot();
DateAxis domain = (DateAxis) plot.getDomainAxis();
domain.setDateFormatOverride(DateFormat.getDateInstance());
plot.setBackgroundPaint(Color.WHITE);
return chartPanel;
}
private XYDataset createDataset() {
TimeSeries series = new TimeSeries("Temperature");
TimeSeriesCollection dataset = new TimeSeriesCollection(series);
try (FileReader fr = new FileReader("TestCSV.csv");
BufferedReader br = new BufferedReader(fr)) {
String line;
br.readLine();
while ((line = br.readLine()) != null) {
String[] split = line.split(",");
System.out.println(ZonedDateTime.parse(split[1]).format(DateTimeFormatter.ISO_LOCAL_TIME) + "," +split[2]);
ZonedDateTime zdt = ZonedDateTime.of(LocalDate.now(),LocalTime.parse(split[0]), ZoneId.systemDefault());
String s = ZonedDateTime.parse(split[0]).format(DateTimeFormatter.ISO_LOCAL_TIME);
Second second = new Second(Date.from(zdt.toInstant()));
series.add(second, Double.valueOf(split[1]));
}
} catch (IOException | SeriesException e) {
System.err.println("Error: " + e);
}
return dataset;
}
仍然显示"CSV"类文件的第一行09:02:53,26.25
然后我得到一个 DateTimeParseException: Text '2017-07-12T09:02:53+02:00' could not be parsed at index 2
Exception in thread "AWT-EventQueue-0" java.time.format.DateTimeParseException: Text '2017-07-12T07:02:53+00:00' could not be parsed at index 2
at java.time.format.DateTimeFormatter.parseResolved0(Unknown Source)
at java.time.format.DateTimeFormatter.parse(Unknown Source)
at java.time.LocalTime.parse(Unknown Source)
at java.time.LocalTime.parse(Unknown Source)
at org.jfree.chart.demo.Test.createDataset(Test.java:63)
at org.jfree.chart.demo.Test.createChart(Test.java:43)
at org.jfree.chart.demo.Test.lambda[=13=](Test.java:34)
为什么文件的其余部分无法读取也无法显示?
("System.out.println()" 应该只作为最后的对照)。 DateTimeFomatter 是正确的,不是吗?
用你的方法,本地制作的时间我就没来得及程序无法翻译。我做错了什么?直接输出
怎么行
09:02:53,26.25
10:02:54,26.08
11:02:55,25.78
12:02:56,25.96
12:51:02,26.14
12:51:02,26.14
是否显示在图表中?我认为像我那样分裂和变形是可以的,不是吗?现在我在代码中设置了 setDateFormatOverride (),但错误消息以及输出保持不变。
有几个问题很明显:
你从不给lines
添加任何东西;至少,您需要这样的东西:
lines.add(line);
而不是 ChartFactory.createXYLineChart()
,考虑创建时间序列:
ChartFactory.createTimeSeriesChart(…)
createDataset()
返回的 XYDataset
应该是一个 TimeSeriesCollection
添加了一个 TimeSeries
.
在createDataset()
中,遍历lines
,解析数据字段,并将值添加到TimeSeries
.
给定的时间值最接近 LocalTime
,但 TimeSeries
期望 add()
坐标由 RegularTimePeriod
和 double
;请参阅 Legacy Date-Time Code 关于下面所示的转换。
请注意,对于重复的域值,TimeSeries
会抛出 SeriesException
;结果,四行中只有三行 int eh 样本输入空气被绘制出来。
而不是替换工厂提供的XYLineAndShapeRenderer
,获取参考以供以后修改。
使用显示的方法之一改变图表的大小here。
避免扩展顶级容器行 ApplicationFrame
.
上 仅 构造和操作 Swing GUI 对象
使用一个try-with-resources statement确保每个资源在语句结束时关闭。
作为您的 actual data contains ISO 8601 dates, ZonedDateTime.parse()
can be used directly; use setDateFormatOverride()
格式化日期轴标签;下面的示例指定了 ISO 8601 格式的 UTC 时区,以便于比较;注释掉对 setDateFormatOverride()
的调用以查看您当地时区的时间。
import java.awt.Color;
import java.awt.EventQueue;
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.time.ZonedDateTime;
import java.util.Date;
import java.util.TimeZone;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.axis.DateAxis;
import org.jfree.chart.plot.XYPlot;
import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer;
import org.jfree.data.general.SeriesException;
import org.jfree.data.time.Second;
import org.jfree.data.time.TimeSeries;
import org.jfree.data.time.TimeSeriesCollection;
import org.jfree.data.xy.XYDataset;
import org.jfree.ui.ApplicationFrame;
/** @see */
public class CSVTest {
public static void main(String[] args) {
EventQueue.invokeLater(() -> {
ApplicationFrame frame = new ApplicationFrame("CSVTest");
CSVTest test = new CSVTest();
frame.add(test.createChart("Temperature profile"));
frame.pack();
frame.setLocationRelativeTo(null);;
frame.setVisible(true);
});
}
private ChartPanel createChart(String chartTitle) {
JFreeChart chart = ChartFactory.createTimeSeriesChart(chartTitle,
"Time", "Temperature", createDataset(), true, true, false);
ChartPanel chartPanel = new ChartPanel(chart);
XYPlot plot = chart.getXYPlot();
plot.setBackgroundPaint(Color.WHITE);
XYLineAndShapeRenderer r = (XYLineAndShapeRenderer) plot.getRenderer();
r.setBaseShapesVisible(true);
DateAxis axis = (DateAxis) plot.getDomainAxis();
SimpleDateFormat df = new SimpleDateFormat("HH:mm:ssX");
df.setTimeZone(TimeZone.getTimeZone("UTC"));
axis.setDateFormatOverride(df);
return chartPanel;
}
private XYDataset createDataset() {
TimeSeries series = new TimeSeries("Temperature");
TimeSeriesCollection dataset = new TimeSeriesCollection(series);
try (FileReader fr = new FileReader("temp.csv");
BufferedReader br = new BufferedReader(fr)) {
String line;
while ((line = br.readLine()) != null) {
String[] s = line.split(",");
ZonedDateTime zdt = ZonedDateTime.parse(s[0]);
Second second = new Second(Date.from(zdt.toInstant()));
series.add(second, Double.valueOf(s[2]));
}
} catch (IOException | SeriesException e) {
System.err.println("Error: " + e);
}
return dataset;
}
}
我想显示随时间变化的温度曲线。我现在已经阅读了一个类似于 CSV 的文件,它让我知道了指示的时间和温度。现在我想使用 JFreeChart 显示值的 XY 图。 文件原始数据如下:
utc,local,celsius
2017-07-12T07:02:53+00:00,2017-07-12T09:02:53+02:00,26.25
2017-07-12T08:02:54+00:00,2017-07-12T10:02:54+02:00,26.08
2017-07-12T09:02:55+00:00,2017-07-12T11:02:55+02:00,25.78
2017-07-12T10:02:56+00:00,2017-07-12T12:02:56+02:00,25.96
2017-07-12T10:51:02+00:00,2017-07-12T12:51:02+02:00,26.14
2017-07-12T10:51:02+00:00,2017-07-12T12:51:02+02:00,26.14
时间和温度值的输出(我已经从原始文件中分离出来)看起来像:
09:02:53,26.25
10:02:54,26.08
11:02:55,25.78
12:02:56,25.96
12:51:02,26.14
12:51:02,26.14
编辑: 现在我在 Trashgod 的示例中插入了一个 DateTimeFormatter: 看起来像:
public class Test {
public static void main(String[] args) {
EventQueue.invokeLater(() -> {
ApplicationFrame frame = new ApplicationFrame("CSVTest");
Test test = new Test();
frame.add(test.createChart("Temperature profile"));
frame.pack();
frame.setLocationRelativeTo(null);;
frame.setVisible(true);
});
}
private ChartPanel createChart(String chartTitle) {
JFreeChart chart = ChartFactory.createTimeSeriesChart(chartTitle,
"Time", "Temperature", createDataset(), true, true, false);
ChartPanel chartPanel = new ChartPanel(chart);
XYPlot plot = chart.getXYPlot();
DateAxis domain = (DateAxis) plot.getDomainAxis();
domain.setDateFormatOverride(DateFormat.getDateInstance());
plot.setBackgroundPaint(Color.WHITE);
return chartPanel;
}
private XYDataset createDataset() {
TimeSeries series = new TimeSeries("Temperature");
TimeSeriesCollection dataset = new TimeSeriesCollection(series);
try (FileReader fr = new FileReader("TestCSV.csv");
BufferedReader br = new BufferedReader(fr)) {
String line;
br.readLine();
while ((line = br.readLine()) != null) {
String[] split = line.split(",");
System.out.println(ZonedDateTime.parse(split[1]).format(DateTimeFormatter.ISO_LOCAL_TIME) + "," +split[2]);
ZonedDateTime zdt = ZonedDateTime.of(LocalDate.now(),LocalTime.parse(split[0]), ZoneId.systemDefault());
String s = ZonedDateTime.parse(split[0]).format(DateTimeFormatter.ISO_LOCAL_TIME);
Second second = new Second(Date.from(zdt.toInstant()));
series.add(second, Double.valueOf(split[1]));
}
} catch (IOException | SeriesException e) {
System.err.println("Error: " + e);
}
return dataset;
}
仍然显示"CSV"类文件的第一行09:02:53,26.25 然后我得到一个 DateTimeParseException: Text '2017-07-12T09:02:53+02:00' could not be parsed at index 2
Exception in thread "AWT-EventQueue-0" java.time.format.DateTimeParseException: Text '2017-07-12T07:02:53+00:00' could not be parsed at index 2
at java.time.format.DateTimeFormatter.parseResolved0(Unknown Source)
at java.time.format.DateTimeFormatter.parse(Unknown Source)
at java.time.LocalTime.parse(Unknown Source)
at java.time.LocalTime.parse(Unknown Source)
at org.jfree.chart.demo.Test.createDataset(Test.java:63)
at org.jfree.chart.demo.Test.createChart(Test.java:43)
at org.jfree.chart.demo.Test.lambda[=13=](Test.java:34)
为什么文件的其余部分无法读取也无法显示? ("System.out.println()" 应该只作为最后的对照)。 DateTimeFomatter 是正确的,不是吗?
用你的方法,本地制作的时间我就没来得及程序无法翻译。我做错了什么?直接输出
怎么行09:02:53,26.25
10:02:54,26.08
11:02:55,25.78
12:02:56,25.96
12:51:02,26.14
12:51:02,26.14
是否显示在图表中?我认为像我那样分裂和变形是可以的,不是吗?现在我在代码中设置了 setDateFormatOverride (),但错误消息以及输出保持不变。
有几个问题很明显:
你从不给
lines
添加任何东西;至少,您需要这样的东西:lines.add(line);
而不是
ChartFactory.createXYLineChart()
,考虑创建时间序列:ChartFactory.createTimeSeriesChart(…)
createDataset()
返回的XYDataset
应该是一个TimeSeriesCollection
添加了一个TimeSeries
.在
createDataset()
中,遍历lines
,解析数据字段,并将值添加到TimeSeries
.给定的时间值最接近
LocalTime
,但TimeSeries
期望add()
坐标由RegularTimePeriod
和double
;请参阅 Legacy Date-Time Code 关于下面所示的转换。请注意,对于重复的域值,
TimeSeries
会抛出SeriesException
;结果,四行中只有三行 int eh 样本输入空气被绘制出来。而不是替换工厂提供的
XYLineAndShapeRenderer
,获取参考以供以后修改。使用显示的方法之一改变图表的大小here。
避免扩展顶级容器行
ApplicationFrame
.- 上 仅 构造和操作 Swing GUI 对象
使用一个try-with-resources statement确保每个资源在语句结束时关闭。
作为您的 actual data contains ISO 8601 dates,
ZonedDateTime.parse()
can be used directly; usesetDateFormatOverride()
格式化日期轴标签;下面的示例指定了 ISO 8601 格式的 UTC 时区,以便于比较;注释掉对setDateFormatOverride()
的调用以查看您当地时区的时间。
import java.awt.Color;
import java.awt.EventQueue;
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.time.ZonedDateTime;
import java.util.Date;
import java.util.TimeZone;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.axis.DateAxis;
import org.jfree.chart.plot.XYPlot;
import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer;
import org.jfree.data.general.SeriesException;
import org.jfree.data.time.Second;
import org.jfree.data.time.TimeSeries;
import org.jfree.data.time.TimeSeriesCollection;
import org.jfree.data.xy.XYDataset;
import org.jfree.ui.ApplicationFrame;
/** @see */
public class CSVTest {
public static void main(String[] args) {
EventQueue.invokeLater(() -> {
ApplicationFrame frame = new ApplicationFrame("CSVTest");
CSVTest test = new CSVTest();
frame.add(test.createChart("Temperature profile"));
frame.pack();
frame.setLocationRelativeTo(null);;
frame.setVisible(true);
});
}
private ChartPanel createChart(String chartTitle) {
JFreeChart chart = ChartFactory.createTimeSeriesChart(chartTitle,
"Time", "Temperature", createDataset(), true, true, false);
ChartPanel chartPanel = new ChartPanel(chart);
XYPlot plot = chart.getXYPlot();
plot.setBackgroundPaint(Color.WHITE);
XYLineAndShapeRenderer r = (XYLineAndShapeRenderer) plot.getRenderer();
r.setBaseShapesVisible(true);
DateAxis axis = (DateAxis) plot.getDomainAxis();
SimpleDateFormat df = new SimpleDateFormat("HH:mm:ssX");
df.setTimeZone(TimeZone.getTimeZone("UTC"));
axis.setDateFormatOverride(df);
return chartPanel;
}
private XYDataset createDataset() {
TimeSeries series = new TimeSeries("Temperature");
TimeSeriesCollection dataset = new TimeSeriesCollection(series);
try (FileReader fr = new FileReader("temp.csv");
BufferedReader br = new BufferedReader(fr)) {
String line;
while ((line = br.readLine()) != null) {
String[] s = line.split(",");
ZonedDateTime zdt = ZonedDateTime.parse(s[0]);
Second second = new Second(Date.from(zdt.toInstant()));
series.add(second, Double.valueOf(s[2]));
}
} catch (IOException | SeriesException e) {
System.err.println("Error: " + e);
}
return dataset;
}
}