Rails.cache.fetch不适用于json数组

时间:2018-01-12 22:46:59

标签: ruby-on-rails ruby caching jsonapi-resources

我正在尝试使用Rail.cache.fetch功能缓存服务器API响应,但它似乎总是'nil'而不是缓存数据。下面的方法运行一次,然后总是失败,当我试图通过@my_api_data_变量时,“访问nil类”类错误

def index
  api_statuses_

end
def api_statuses_
  Rails.cache.fetch('api_repsonse_m_', expires_in: 3.minute) do


      @mypairs_ = ['ticker1','ticker2', 'ticker3', 'ticker4', 'ticker5', 'ticker6', 'ticker7', 'ticker8', 'ticker9', 'ticker10']

      @my_api_data_ = []
      @my_api_data_poplr = []
      url = 'https://myserver.com/ticker/'

      @mypairs_.each do |item_|
        uri = URI(url + item_)
        response = Net::HTTP.get(uri)
        begin
          #response is like this {"status":"valu1","time":"valu2","data":"valu3"}
          myresult = JSON.parse(response)
        rescue JSON::ParserError

        end
        #Rails.logger.info myresult
        @my_api_data_.push(myresult);
      end

  end
end

我希望也许有人知道为什么每次都变坏了

我尝试在缓存块结束前添加“@my_api_data_”行。这与我在原始情况下得到的错误相同:

I, [2018-01-13T19:29:14.384766 #5716]  INFO -- : Processing by Private::MydataController#index as HTML
I, [2018-01-13T19:29:14.403774 #5716]  INFO -- :   Rendered private/mydata/index.html.slim within layouts/mydatalayout_ (0.7ms)
I, [2018-01-13T19:29:14.403919 #5716]  INFO -- : Completed 500 Internal Server Error in 19ms
F, [2018-01-13T19:29:14.405077 #5716] FATAL -- : 
ActionView::Template::Error (undefined method `each' for nil:NilClass):
    37:         .row
    38:           /! Top head coin-boxes
    39:           .market-top-coin-boxes
    40:             - @my_api_data_.each do |pair_data|

2 个答案:

答案 0 :(得分:1)

这里的问题似乎是你必须在块的末尾返回@my_api_data_。让我们检查下面的代码:

def api_statuses_
  Rails.cache.fetch('api_repsonse_m_', expires_in: 3.minute) do
    @mypairs_ = ['ticker1','ticker2', 'ticker3', 'ticker4', 'ticker5', 'ticker6', 'ticker7', 'ticker8', 'ticker9', 'ticker10']

    @my_api_data_ = []
    @my_api_data_poplr = []
    url = 'https://myserver.com/ticker/'

    @mypairs_.each do |item_|
      uri = URI(url + item_)
      response = Net::HTTP.get(uri)
      begin
        #response is like this {"status":"valu1","time":"valu2","data":"valu3"}
        myresult = JSON.parse(response)
      rescue JSON::ParserError
      end
      #Rails.logger.info myresult
      @my_api_data_.push(myresult);
    end
    @my_api_data
  end
end

请注意,第一次调用api_statuses_方法时,将执行所有被阻止的操作,并且返回值将被写入api_repsonse_m_键下的缓存中。因此,下次您调用api_statuses_时,如果expires_in中的时间尚未过期,它将返回缓存中的值save,并且不会执行该块。

另外,您能否告诉我们您在何处以及如何使用此方法?显示代码和错误日志可以很好地理解问题。

已更新:您似乎没有正确使用此功能。您应该在api_statuses_控制器下的索引方法中调用MydataController方法,并将返回值保存到将在视图中使用的变量中。

例如:

module Private
  class MydataController
    def index 
      @api_statuses = api_statuses_
    end
  end
end

然后,在您看来:

.row
  .market-top-coin-boxes
    - @api_statuses.each do |pair_data|

答案 1 :(得分:1)

为什么你在视图中使用'@my_api_data_'?

使用

def index
  @result = api_statuses_
end

在视图文件中使用@result。注意,如果有缓存命中,则传递给Rails.cache.fetch的块不会运行。