有很多关于Octave的dlmread
性能缓慢的报告。我曾希望这个问题在3.2.4中得到解决,但当我尝试加载一个大约8*4百万(总共32百万)大小的CSV文件时,它也需要非常长的时间。我在网上搜索,但找不到解决方法。是否有人知道一个好的解决方法?
有很多关于Octave的dlmread
性能缓慢的报告。我曾希望这个问题在3.2.4中得到解决,但当我尝试加载一个大约8*4百万(总共32百万)大小的CSV文件时,它也需要非常长的时间。我在网上搜索,但找不到解决方法。是否有人知道一个好的解决方法?
我曾经遇到过同样的问题,而且我会使用R语言,所以我的解决方案是使用R中的"read.csv"函数,然后使用R包"R.matlab"来写入一个".mat"文件,最后在Octave中加载这个文件。
"read.csv"也可能会很慢,但在我的情况下这个方法非常有效。
function alldata = load_data(filename)
fid = fopen(filename,'r');
s=0;
data=[];
alldata=[];
save "temp.mat" alldata;
if fid == -1
disp("Couldn't find file mydata");
else
while (~feof(fid))
line = fgetl(fid);
[t1,t2,t3,t4,d] = sscanf(line,'%i:%i:%i:%i %f', "C"); #reading time as hh:mm:ss:ms and data as float
s++;
t = (t1 * 3600000 + t2 * 60000 + t3 * 1000 + t4);
data = [data; t, d];
if (mod(s,10000) == 0)
#disp(s), disp(" "), disp(t), disp(" "), disp(d), disp("\n");
disp(s);
fflush(stdout);
end
if (mod(s,50000) == 0)
load "temp.mat";
alldata=[alldata; data];
data=[];
save "temp.mat" alldata;
disp("data saved");
fflush(stdout);
end
end
disp(s);
load "temp.mat";
alldata=[alldata; data];
save "temp.mat" alldata;
disp("data saved");
fflush(stdout);
end
fclose(fid);
这里是我正在使用的解决方法。
我发现sscanf不会像上面所示那样解析输入行。此外,我没有使用临时文件。
我的.csv文件有大量的行。它们以18行的标题开头,然后是一个数据块,每个块有135列。以下代码已经过测试。我的文件还以dd/mm/yyyy hh:mm字段开始每一行。这也将通过使用try/catch捕获错误的行并指出它们的位置。
我的.csv文件来自一个客户,他将PARCView负载转储到Excel文件中。
function [tags,descr,alldata] = fbcsvread(filename)
fid = fopen(filename,'r');
s = 0;
data=[];
alldata=zeros(1,135);
if fid==-1
disp("Couldn't find file %s\n",filename);
else
linecount = 1;
while (~feof(fid))
line = fgetl(fid);
data2 = zeros(1,135);
if linecount == 1
tags = strsplit(line,",");
elseif linecount == 2
descr = strsplit(line,",");
elseif linecount >= 19
data = strsplit(line,",");
datetime = strsplit(char(data(1))," ");
modyyr = strsplit(char(datetime(1)),"/");
hrmin = strsplit(char(datetime(2)),":");
year1 = sscanf(char(modyyr(3)),"%d","C");
day1 = sscanf(char(modyyr(2)),"%d","C");
month1 = sscanf(char(modyyr(1)),"%d","C");
hour1 = sscanf(char(hrmin(1)),"%d","C");
minute1 = sscanf(char(hrmin(2)),"%d","C");
realtime = datenum(year1,month1,day1,hour1,minute1);
data2(1) = realtime;
for location = 2:134
try
data2(location) = sscanf(char(data(location)),"%f","C");
catch
printf("Error at %s %s\n",char(datetime(1)),char(datetime(2)) );
fflush(stdout);
end_try_catch
endfor
alldata(linecount-18,:) = data2;
if mod(linecount,50) == 0
printf(".");
fflush(stdout);
endif
endif
linecount = linecount + 1;
endwhile
fclose(fid);
endif
endfunction