Kolay hata ayıklama için Rails'teki bir nesnenin içeriğini nasıl yazdırabilirim?


101

Sanırım PHP'nin eşdeğerini print_r()(insan tarafından okunabilir yazdırın) almaya çalışıyorum ; şu anda ham çıktı:

ActiveRecord::Relation:0x10355d1c0

Ne yapmalıyım?


Görmediyseniz ( benimkinden hemen önce gönderilen bir yanıtı kabul ettiğiniz için ), PHP'de debug () işlevinin tam olarak print_r () gibi çalıştığını unutmayın .
Andrew

1
Sadece bu sayfaya daha sonra debug () ile gelen herkes için modası geçmiş ve artık bir işlev olarak dahil edilmemiştir. Çalışmayacak. ( Bunu sayfanın daha da aşağısına işaret ettiği için stackoverflow.com/users/231309/irongaze-com'a kredi verin .)
0112

Yanıtlar:


213

Genelde önce denerim .inspect, bu bana istediğimi vermezse, geçiş yaparım .to_yaml.

class User
  attr_accessor :name, :age
end

user = User.new
user.name = "John Smith"
user.age = 30

puts user.inspect
#=> #<User:0x423270c @name="John Smith", @age=30>
puts user.to_yaml
#=> --- !ruby/object:User
#=> age: 30
#=> name: John Smith

Umarım yardımcı olur.


6
Kayıtların bazı YAML çıktılarının görmek istediğimden daha fazla veri (meta veri, belki?) Gösterdiğini buldum. Kullanacağım bir kaydın YAML versiyonunu arıyorsanız y record_name.attributes. #yiçin bir takma addır to_yaml.
Tass

9

modelinizdeki to_s yöntemini tanımlayın. Örneğin

class Person < ActiveRecord::Base
  def to_s
    "Name:#{self.name} Age:#{self.age} Weight: #{self.weight}"
  end
end

Sonra, #puts ile yazdırmaya gittiğinizde, bu değişkenlerle birlikte bu dizeyi gösterecektir.


Ya içerdiği değişkenlerin ne olduğunu bilmiyorsanız?
cjm2671

Daha spesifik olabilir misin? Ya bir değişken bir dizi veya karmadır diyorsunuz? #To_s bunu hallederdi.
Chris Ledet

Bu doğru ifade edilmemiştir. puts my_model_instancearamayacak to_s. Bunu açıkça yapmanız gerekecek:puts my_model_instance.to_s
thisismydesign

6

Rails'de , görünümü hata ayıklama 'Helper ActionView :: Helpers :: DebugHelper'ı kullanarak yazdırabilirsiniz.

#app/view/controllers/post_controller.rb
def index
 @posts = Post.all
end

#app/view/posts/index.html.erb
<%= debug(@posts) %>

#start your server
rails -s

sonuçlar (tarayıcıda)

- !ruby/object:Post
  raw_attributes:
    id: 2
    title: My Second Post
    body: Welcome!  This is another example post
    published_at: '2015-10-19 23:00:43.469520'
    created_at: '2015-10-20 00:00:43.470739'
    updated_at: '2015-10-20 00:00:43.470739'
  attributes: !ruby/object:ActiveRecord::AttributeSet
    attributes: !ruby/object:ActiveRecord::LazyAttributeHash
      types: &5
        id: &2 !ruby/object:ActiveRecord::Type::Integer
          precision: 
          scale: 
          limit: 
          range: !ruby/range
            begin: -2147483648
            end: 2147483648
            excl: true
        title: &3 !ruby/object:ActiveRecord::Type::String
          precision: 
          scale: 
          limit: 
        body: &4 !ruby/object:ActiveRecord::Type::Text
          precision: 
          scale: 
          limit: 
        published_at: !ruby/object:ActiveRecord::AttributeMethods::TimeZoneConversion::TimeZoneConverter
          subtype: &1 !ruby/object:ActiveRecord::Type::DateTime
            precision: 
            scale: 
            limit: 
        created_at: !ruby/object:ActiveRecord::AttributeMethods::TimeZoneConversion::TimeZoneConverter
          subtype: *1
        updated_at: !ruby/object:ActiveRecord::AttributeMethods::TimeZoneConversion::TimeZoneConverter
          subtype: *1

6

Awesome_print cevherini kullanıyorum

Yani yazmanız yeterli:

ap @var

2
Bu kadar basit bir şey için bir mücevher kurmanın savunucusu değilim, ancak düzenli olarak Awesome Print'i kullanıyorum.
Tass

Bu gerçekten iyi, sonunda kodumda yaptığım şey şuydu: Rails.logger.ap someObject
Aleksandar Pavić

gem install awesome_print (Gemfile için düzenleme gerekmez)
Jay

4

.inspectaradığınız şey, IMO'dan çok daha kolay .to_yaml!

user = User.new
user.name = "will"
user.email = "will@example.com"

user.inspect
#<name: "will", email: "will@example.com">

3

pp de işi yapar, hiçbir mücevher gerektirmez.

@a = Accrual.first ; pp @a

#<Accrual:0x007ff521e5ba50
 id: 4,
 year: 2018,
 Jan: #<BigDecimal:7ff521e58f08,'0.11E2',9(27)>,
 Feb: #<BigDecimal:7ff521e585d0,'0.88E2',9(27)>,
 Mar: #<BigDecimal:7ff521e58030,'0.0',9(27)>,
 Apr: #<BigDecimal:7ff521e53698,'0.88E2',9(27)>,
 May: #<BigDecimal:7ff521e52fb8,'0.8E1',9(27)>,
 June: #<BigDecimal:7ff521e52900,'0.8E1',9(27)>,
 July: #<BigDecimal:7ff521e51ff0,'0.8E1',9(27)>,
 Aug: #<BigDecimal:7ff521e51bb8,'0.88E2',9(27)>,
 Sep: #<BigDecimal:7ff521e512f8,'0.88E2',9(27)>,
 Oct: #<BigDecimal:7ff521e506c8,'0.0',9(27)>,
 Nov: #<BigDecimal:7ff521e43d38,'0.888E3',9(27)>,
 Dec: #<BigDecimal:7ff521e43478,'0.0',9(27)>,

Ayrıca bir nesnenin iki örneğini de yazdırabilirsiniz:

 pp( Accrual.first , Accrual.second)
`
`
`

2

inspectharika ama bazen yeterince iyi değil. Örneğin BigDecimalböyle yazdırır: #<BigDecimal:7ff49f5478b0,'0.1E2',9(18)>.

Neyin yazdırıldığını tam olarak kontrol edebilmek için yöntemleri to_sveya inspectyöntemleri yeniden tanımlayabilirsiniz . Veya gelecekteki geliştiricilerin kafasını fazla karıştırmamak için kendi kartınızı yaratın.

  class Something < ApplicationRecord

    def to_s
      attributes.map{ |k, v| { k => v.to_s } }.inject(:merge)
    end

  end

Bu, to_stüm özniteliklere bir yöntem (yani ) uygulayacaktır . Bu örnek çirkinlerden kurtulacaktır BigDecimals.

Ayrıca, yalnızca birkaç özelliği yeniden tanımlayabilirsiniz:

  def to_s
    attributes.merge({ my_attribute: my_attribute.to_s })
  end

Ayrıca ikisinin bir karışımını oluşturabilir veya bir şekilde ilişkilendirme ekleyebilirsiniz.


-3

Kullanmanız gerekiyor debug(@var). Aynen "print_r" gibidir.


5
Bu bir şey değil, en azından Ruby 1.9.x - NoMethodError: main için tanımlanmamış yöntem `` debug '': Object
Irongaze.com
Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.