1

我正在尝试使用带有标题名称和非同质数据类型的 numpy genfromtxt 导入数据。每次我运行程序时,我都会收到错误:

Traceback (most recent call last):
    raise ValueError(errmsg)
ValueError: Some errors were detected !
    Line #8 (got 6 columns instead of 1)
    Line #9 (got 6 columns instead of 1)
    Line #10 (got 6 columns instead of 1)
    Line #11 (got 6 columns instead of 1)
    Line #12 (got 6 columns instead of 1)

我已经解决了这个问题 ,但它并没有解决我的问题。这是一个非常简单的问题,但我不知道出了什么问题。代码和数据包括:

代码

import numpy as np
data = np.genfromtxt('Data.dat', comments='#', delimiter='\t', names=True, dtype=None).transpose()
print data

制表符分隔的数据

# -----
# -----
# -----
# -----
# -----
# -----
# -----
column_1    column_2    column_3    column_4    column_5    column_6
1   2   3   A   1   F
4   3   2   B   2   G
1   4   3   C   3   H
5   6   4   D   4   I

更新

简而言之,我需要一种将skip_header之后的第一个有效行转换为具有可选参数names=True的第一个未注释的有效行的方法。

4

3 回答 3

2

When names=True,genfromtxt期望第一行(skip_header行后)包含字段名称,即使该行是注释。显然,在注释中指定字段名称是很常见的。如果在未注释的字段名称之前有可变数量的注释,则必须解决genfromtxt. 以下显示了您可以执行此操作的一种方法。

这是我的测试文件。(该文件以空格分隔。添加对制表符分隔文件delimiter='\t'的调用genfromtxt)。

In [12]: cat with_comments.dat
# Some
# comments
# here
foo bar baz
1.0 2.0 3.0
4.0 5.0 6.0
7.0 8.0 9.0

打开文件,并读取行,直到该行不是注释:

In [13]: f = open("with_comments.dat", "r")

In [14]: line = f.readline()

In [15]: while line.startswith('#'):
   ....:     line = f.readline()
   ....: 

line现在包含字段名称行:

In [16]: line
Out[16]: 'foo bar baz\n'

将其转换为名称列表:

In [17]: names = line.split()

将这些名称赋予 genfromtxt,然后阅读文件的其余部分:

In [18]: data = genfromtxt(f, names=names)

In [19]: data
Out[19]: 
array([(1.0, 2.0, 3.0), (4.0, 5.0, 6.0), (7.0, 8.0, 9.0)], 
      dtype=[('foo', '<f8'), ('bar', '<f8'), ('baz', '<f8')])

不要忘记关闭文件(或者更好的是,with("with_comments.dat", "r") as f:改用):

In [20]: f.close()
于 2015-03-02T17:25:27.350 回答
0

好的,一点点刺激已经揭示了答案。从genfromtxt()文档(http://docs.scipy.org/doc/numpy/user/basics.io.genfromtxt.html):

注意:此行为有一个值得注意的例外:如果可选参数 names=True,则将检查第一个注释行的名称。

因此,要使您的代码正常工作,您的数据应采用以下格式:

#column_1   column_2    column_3    column_4    column_5    column_6
#   -----
#   -----
#   -----
#   -----
#   -----
#   -----
1   2   3   A   1   F
4   3   2   B   2   G
1   4   3   C   3   H
5   6   4   D   4   I

或者,如果您有可变数量的标题/注释行,但列都是相同的,那么您可以在genfromtxt参数中定义列名:

data = np.genfromtxt(
    path, comments='#', delimiter='\t', 
    names='column_1,column_2,column_3,column_4,column_5,column_6',
    dtype=None
)

但是,通过使用comments关键字,genfromtxt将读取包含列标题的最后一个注释行之后的第一行。它将假定它是数据的一部分,因此您的 dtype 应该是字符串,因此您在此阶段的数据将如下所示:

array([('column_1', 'column_2', 'column_3', 'column_4', 'column_5', 'column_6'),
       ('1', '2', '3', 'A', '1', 'F'), ('4', '3', '2', 'B', '2', 'G'),
       ('1', '4', '3', 'C', '3', 'H'), ('5', '6', '4', 'D', '4', 'I')], 
      dtype=[('column_1', 'S8'), ('column_2', 'S8'), ('column_3', 'S8'), ('column_4', 'S8'), ('column_5', 'S8'), ('column_6', 'S8')])

如果你知道你的列应该是什么数据类型,你首先取一个不包括第一行的切片:

data1 = data[1:]

然后修改dtypes

data1.astype(np.dtype([('column_1', 'i4'),('column_2', 'i4'), ('column_3', 'i4'), ('column_4', 'S10'), ('column_5', 'i4'), ('column_6', 'S10')]))

输出:

array([(1, 2, 3, 'A', 1, 'F'), (4, 3, 2, 'B', 2, 'G'),
       (1, 4, 3, 'C', 3, 'H'), (5, 6, 4, 'D', 4, 'I')], 
      dtype=[('column_1', '<i4'), ('column_2', '<i4'), ('column_3', '<i4'), ('column_4', 'S10'), ('column_5', '<i4'), ('column_6', 'S10')])
于 2015-03-02T14:14:55.743 回答
0

根据以下文件genfromtxt

如果names为 True,则从第一行之后的第一个有效行读取字段名称skip_header

在您的示例中,您可以添加skip_header=7genfromtxt调用以使其工作。

于 2015-03-02T14:18:08.227 回答