类 CSV::Row
CSV::Row¶ ↑
CSV::Row 实例表示 CSV 表格行。(参见 类 CSV)。
该实例可能具有
-
字段:每个都是对象,不一定是字符串。
-
标题:每个都充当键,也不一定是字符串。
实例方法¶ ↑
CSV::Row 有三组实例方法
-
其自身内部定义的实例方法。
-
模块
Enumerable
包含的方法。 -
委托给类
Array
的方法。
创建 CSV::Row 实例¶ ↑
通常,通过解析具有标题的 CSV 源来创建新的 CSV::Row 实例
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) table.each {|row| p row }
输出
#<CSV::Row "Name":"foo" "Value":"0"> #<CSV::Row "Name":"bar" "Value":"1"> #<CSV::Row "Name":"baz" "Value":"2">
你也可以直接创建一行。参见 ::new
。
标题¶ ↑
与 CSV::Table 一样,CSV::Row 也有标题。
通过解析 CSV 源创建的 CSV::Row 从表中继承其标题
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table.first row.headers # => ["Name", "Value"]
您还可以创建一个带有标题的新行;与哈希中的键一样,标题不必是字符串
row = CSV::Row.new([:name, :value], ['foo', 0]) row.headers # => [:name, :value]
即使添加到具有标题的表中,新行也会保留其标题
table << row # => #<CSV::Table mode:col_or_row row_count:5> row.headers # => [:name, :value] row[:name] # => "foo" row['Name'] # => nil
访问字段¶ ↑
您可以使用其整数索引(数组样式)或其标题(哈希样式)来访问 CSV::Row 中的字段。
使用 []
方法获取字段
row = CSV::Row.new(['Name', 'Value'], ['foo', 0]) row[1] # => 0 row['Value'] # => 0
row = CSV::Row.new(['Name', 'Value'], ['foo', 0]) row # => #<CSV::Row "Name":"foo" "Value":0> row[0] = 'bar' row['Value'] = 1 row # => #<CSV::Row "Name":"bar" "Value":1>
属性
用于比较相等性的内部数据格式。
公共类方法
返回从参数 headers
和 fields
构建的新 CSV::Row 实例;两者都应该是数组;请注意,字段不必是字符串
row = CSV::Row.new(['Name', 'Value'], ['foo', 0]) row # => #<CSV::Row "Name":"foo" "Value":0>
如果数组长度不同,则较短的数组将填充为 nil
row = CSV::Row.new(['Name', 'Value', 'Date', 'Size'], ['foo', 0]) row # => #<CSV::Row "Name":"foo" "Value":0 "Date":nil "Size":nil>
每个 CSV::Row 对象要么是字段行,要么是标题行;默认情况下,新行是字段行;对于上面创建的行
row.field_row? # => true row.header_row? # => false
如果可选参数 header_row
被指定为 true
,则创建的行是标题行
row = CSV::Row.new(['Name', 'Value'], ['foo', 0], header_row = true) row # => #<CSV::Row "Name":"foo" "Value":0> row.field_row? # => false row.header_row? # => true
# File lib/csv/row.rb, line 105 def initialize(headers, fields, header_row = false) @header_row = header_row headers.each { |h| h.freeze if h.is_a? String } # handle extra headers or fields @row = if headers.size >= fields.size headers.zip(fields) else fields.zip(headers).each(&:reverse!) end end
公共实例方法
向 self
添加一个字段;返回 self
如果参数是 2 元素数组 [header, value]
,则添加一个具有给定 header
和 value
的字段
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row << ['NAME', 'Bat'] row # => #<CSV::Row "Name":"Foo" "Name":"Bar" "Name":"Baz" "NAME":"Bat">
如果参数是哈希,则每个 键值
对都将作为具有标题 key
和值 value
的字段添加。
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row << {NAME: 'Bat', name: 'Bam'} row # => #<CSV::Row "Name":"Foo" "Name":"Bar" "Name":"Baz" NAME:"Bat" name:"Bam">
否则,给定的 value
将作为没有标题的字段添加。
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row << 'Bag' row # => #<CSV::Row "Name":"Foo" "Name":"Bar" "Name":"Baz" nil:"Bag">
# File lib/csv/row.rb, line 389 def <<(arg) if arg.is_a?(Array) and arg.size == 2 # appending a header and name @row << arg elsif arg.is_a?(Hash) # append header and name pairs arg.each { |pair| @row << pair } else # append field value @row << [nil, arg] end self # for chaining end
如果 other
是一个 /CSV::Row,并且其字段(标题和值)与 self
的顺序相同,则返回 true
;否则返回 false
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] other_row = table[0] row == other_row # => true other_row = table[1] row == other_row # => false
# File lib/csv/row.rb, line 633 def ==(other) return @row == other.row if other.is_a? CSV::Row @row == other end
分配给给定index
或header
的字段值;返回value
。
按整数索引分配字段值
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table[0] row[0] = 'Bat' row[1] = 3 row # => #<CSV::Row "Name":"Bat" "Value":3>
如果index
为负,则从最后一列向后计数
row[-1] = 4 row[-2] = 'Bam' row # => #<CSV::Row "Name":"Bam" "Value":4>
如果正index
不在行中,则用nil:nil
扩展行
row[4] = 5 row # => #<CSV::Row "Name":"bad" "Value":4 nil:nil nil:nil nil:5>
如果负index
太小(离零太远),则引发IndexError
。
按标题分配字段值(首次找到)
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row['Name'] = 'Bat' row # => #<CSV::Row "Name":"Bat" "Name":"Bar" "Name":"Baz">
按标题分配字段值,忽略offset
前导字段
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row['Name', 2] = 4 row # => #<CSV::Row "Name":"Foo" "Name":"Bar" "Name":4>
按(新)标题追加新字段
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table[0] row['New'] = 6 row# => #<CSV::Row "Name":"foo" "Value":"0" "New":6>
# File lib/csv/row.rb, line 339 def []=(*args) value = args.pop if args.first.is_a? Integer if @row[args.first].nil? # extending past the end with index @row[args.first] = [nil, value] @row.map! { |pair| pair.nil? ? [nil, nil] : pair } else # normal index assignment @row[args.first][1] = value end else index = index(*args) if index.nil? # appending a field self << [args.first, value] else # normal header assignment @row[index][1] = value end end end
返回适合于模式匹配的新数组,其中包含该行的值。
# File lib/csv/row.rb, line 682 def deconstruct fields end
返回适合于模式匹配的新哈希,其中仅包含作为参数指定的键。
# File lib/csv/row.rb, line 667 def deconstruct_keys(keys) if keys.nil? to_h else keys.to_h { |key| [key, self[key]] } end end
从self
中删除指定字段;如果字段存在,则返回 2 元素数组[header, value]
。
如果给出了整数参数index
,则删除并返回偏移量index
处的字段,或者如果字段不存在,则返回nil
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.delete(1) # => ["Name", "Bar"] row.delete(50) # => nil
否则,如果给出了单个参数header
,则删除并返回具有给定标题的第一个找到的字段,如果字段不存在,则返回一个新的空数组
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.delete('Name') # => ["Name", "Foo"] row.delete('NAME') # => []
如果给出了参数header
和整数参数offset
,则删除并返回具有给定标题的第一个找到的字段,其index
至少与offset
一样大
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.delete('Name', 1) # => ["Name", "Bar"] row.delete('NAME', 1) # => []
# File lib/csv/row.rb, line 451 def delete(header_or_index, minimum_index = 0) if header_or_index.is_a? Integer # by index @row.delete_at(header_or_index) elsif i = index(header_or_index, minimum_index) # by header @row.delete_at(i) else [ ] end end
从self
中删除块选定的字段;返回self
。
删除块返回真值字段
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.delete_if {|header, value| value.start_with?('B') } # => true row # => #<CSV::Row "Name":"Foo"> row.delete_if {|header, value| header.start_with?('B') } # => false
如果没有给出块,则返回一个新的枚举器
row.delete_if # => #<Enumerator: #<CSV::Row "Name":"Foo">:delete_if>
# File lib/csv/row.rb, line 476 def delete_if(&block) return enum_for(__method__) { size } unless block_given? @row.delete_if(&block) self # for chaining end
查找并返回嵌套对象中由index_or_header
和specifiers
指定的对象。
嵌套对象可以是各种类的实例。请参阅挖掘方法。
示例
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table[0] row.dig(1) # => "0" row.dig('Value') # => "0" row.dig(5) # => nil
# File lib/csv/row.rb, line 715 def dig(index_or_header, *indexes) value = field(index_or_header) if value.nil? nil elsif indexes.empty? value else unless value.respond_to?(:dig) raise TypeError, "#{value.class} does not have \#dig method" end value.dig(*indexes) end end
使用每个头值对调用块;返回 self
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.each {|header, value| p [header, value] }
输出
["Name", "Foo"] ["Name", "Bar"] ["Name", "Baz"]
如果没有给出块,则返回一个新的枚举器
row.each # => #<Enumerator: #<CSV::Row "Name":"Foo" "Name":"Bar" "Name":"Baz">:each>
# File lib/csv/row.rb, line 610 def each(&block) return enum_for(__method__) { size } unless block_given? @row.each(&block) self # for chaining end
返回 header
指定的字段值。
使用单个参数 header
,返回该头部的字段值(首先找到)
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.fetch('Name') # => "Foo"
如果头部不存在,则引发异常 KeyError
。
如果给出了参数 header
和 default
,则在头部存在时返回该头部的字段值(首先找到),否则返回 default
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.fetch('Name', '') # => "Foo" row.fetch(:nosuch, '') # => ""
如果给出了参数 header
和一个块,则在头部存在时返回该头部的字段值(首先找到);否则调用该块并返回其返回值
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.fetch('Name') {|header| fail 'Cannot happen' } # => "Foo" row.fetch(:nosuch) {|header| "Header '#{header} not found'" } # => "Header 'nosuch not found'"
# File lib/csv/row.rb, line 258 def fetch(header, *varargs) raise ArgumentError, "Too many arguments" if varargs.length > 1 pair = @row.assoc(header) if pair pair.last else if block_given? yield header elsif varargs.empty? raise KeyError, "key not found: #{header}" else varargs.first end end end
返回给定 index
或 header
的字段值。
按整数索引获取字段值
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table[0] row.field(0) # => "foo" row.field(1) # => "bar"
如果index
为负,则从最后一列向后计数
row.field(-1) # => "0" row.field(-2) # => "foo"
如果 index
超出范围,则返回 nil
row.field(2) # => nil row.field(-3) # => nil
按头部获取字段值(首先找到)
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.field('Name') # => "Foo"
按头部获取字段值,忽略 offset
前导字段
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.field('Name', 2) # => "Baz"
如果头部不存在,则返回 nil
。
# File lib/csv/row.rb, line 203 def field(header_or_index, minimum_index = 0) # locate the pair finder = (header_or_index.is_a?(Integer) || header_or_index.is_a?(Range)) ? :[] : :assoc pair = @row[minimum_index..-1].public_send(finder, header_or_index) # return the field if we have a pair if pair.nil? nil else header_or_index.is_a?(Range) ? pair.map(&:last) : pair.last end end
如果 value
是此行中的一个字段,则返回 true
,否则返回 false
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.field?('Bar') # => true row.field?('BAR') # => false
# File lib/csv/row.rb, line 589 def field?(data) fields.include? data end
如果这是字段行,则返回 true
,否则返回 false
。
# File lib/csv/row.rb, line 148 def field_row? not header_row? end
返回给定 specifiers
的字段值,它可以是以下任何组合
-
整数索引。
-
整数索引范围。
-
包含头和偏移量的 2 元素数组。
-
头。
-
头范围。
对于上述前四个案例中的specifier
,返回self.field(specifier)
的结果;请参见field
。
虽然可能存在任意数量的specifiers
,但此处示例将一次说明一个。
当说明符是整数index
时,返回self.field(index)
L
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.fields(1) # => ["Bar"]
当说明符是整数范围range
时,返回self.field(range)
row.fields(1..2) # => ["Bar", "Baz"]
当说明符是 2 元素数组array
时,返回self.field(array)
L
row.fields('Name', 1) # => ["Foo", "Bar"]
当说明符是头header
时,返回self.field(header)
L
row.fields('Name') # => ["Foo"]
当说明符是头范围range
时,从range.start
和range.end
的索引中形成一个新的范围new_range
,并返回self.field(new_range)
source = "Name,NAME,name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.fields('Name'..'NAME') # => ["Foo", "Bar"]
如果未给出参数,则返回所有字段
row.fields # => ["Foo", "Bar", "Baz"]
# File lib/csv/row.rb, line 530 def fields(*headers_and_or_indices) if headers_and_or_indices.empty? # return all fields--no arguments @row.map(&:last) else # or work like values_at() all = [] headers_and_or_indices.each do |h_or_i| if h_or_i.is_a? Range index_begin = h_or_i.begin.is_a?(Integer) ? h_or_i.begin : index(h_or_i.begin) index_end = h_or_i.end.is_a?(Integer) ? h_or_i.end : index(h_or_i.end) new_range = h_or_i.exclude_end? ? (index_begin...index_end) : (index_begin..index_end) all.concat(fields.values_at(new_range)) else all << field(*Array(h_or_i)) end end return all end end
如果存在具有给定header
的字段,则返回true
,否则返回false
。
# File lib/csv/row.rb, line 279 def has_key?(header) !!@row.assoc(header) end
如果这是标题行,则返回true
,否则返回false
。
# File lib/csv/row.rb, line 140 def header_row? @header_row end
返回此行的标题
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table.first row.headers # => ["Name", "Value"]
# File lib/csv/row.rb, line 160 def headers @row.map(&:first) end
如果存在,则返回给定头的索引;否则返回nil
。
使用单个参数header
,返回具有给定header
的第一个找到的字段的索引
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.index('Name') # => 0 row.index('NAME') # => nil
使用参数header
和offset
,返回具有给定header
的第一个找到的字段的索引,但忽略前offset
个字段
row.index('Name', 1) # => 1 row.index('Name', 3) # => nil
# File lib/csv/row.rb, line 573 def index(header, minimum_index = 0) # find the pair index = headers[minimum_index..-1].index(header) # return the index at the right offset, if we found one index.nil? ? nil : index + minimum_index end
调用父类方法。
# File lib/csv/row.rb, line 130 def initialize_copy(other) super_return_value = super @row = @row.collect(&:dup) super_return_value end
返回一个 ASCII 兼容的 String,显示
-
Class
CSV::Row。 -
头值对。
示例
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table[0] row.inspect # => "#<CSV::Row \"Name\":\"foo\" \"Value\":\"0\">"
# File lib/csv/row.rb, line 740 def inspect str = ["#<", self.class.to_s] each do |header, field| str << " " << (header.is_a?(Symbol) ? header.to_s : header.inspect) << ":" << field.inspect end str << ">" begin str.join('') rescue # any encoding error str.map do |s| e = Encoding::Converter.asciicompat_encoding(s.encoding) e ? s.encode(e) : s.force_encoding("ASCII-8BIT") end.join('') end end
将给定的每个 values
追加到 self
中作为字段;返回 self
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.push('Bat', 'Bam') row # => #<CSV::Row "Name":"Foo" "Name":"Bar" "Name":"Baz" nil:"Bat" nil:"Bam">
# File lib/csv/row.rb, line 410 def push(*args) args.each { |arg| self << arg } self # for chaining end
将行作为 CSV String
返回。不包括头
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table[0] row.to_csv # => "foo,0\n"
# File lib/csv/row.rb, line 694 def to_csv(**options) fields.to_csv(**options) end
返回通过将 self
中的每个头值对作为 Hash 中的键值对添加而形成的新 Hash。
source = "Name,Value\nfoo,0\nbar,1\nbaz,2\n" table = CSV.parse(source, headers: true) row = table[0] row.to_h # => {"Name"=>"foo", "Value"=>"0"}
保留头顺序,但忽略重复头
source = "Name,Name,Name\nFoo,Bar,Baz\n" table = CSV.parse(source, headers: true) row = table[0] row.to_h # => {"Name"=>"Foo"}
# File lib/csv/row.rb, line 653 def to_h hash = {} each do |key, _value| hash[key] = self[key] unless hash.key?(key) end hash end