ruby - Ruby中 block 的使用顺序是什么

标签 ruby block

我正在创建一个 gem 来支持命令行中的一些邮件。我用了一些 gem 。 我正在使用 Mail Gem .正如您在 mail gem 的描述中看到的那样。

mail = Mail.new do
  from    'mikel@test.lindsaar.net'
  to      'you@test.lindsaar.net'
  subject 'This is a test email'
  body    File.read('body.txt')
end

在代码块中,我调用了 Mail 类中的方法(from、to、subject、body)。这是有道理的,所以我在自己的邮件程序类中构建它

def initialize(mail_settings, working_hours)
  @mail_settings = mail_settings
  @working_hours = working_hours
  @mailer = Mail.new do
    to mail_settings[:to]
    from mail_settings[:from]
    subject mail_settings[:subject]
    body "Start #{working_hours[:start]} \n\
          Ende #{working_hours[:end]}\n\
          Pause #{working_hours[:pause]}"
  end
end

这看起来很简单。只需调用 block 并填写我通过构造函数获得的值。 现在我的问题来了。

我试图将邮件的正文构造放到一个单独的方法中。但是我不能在 gem 的 Mail 构造函数中使用它。

module BossMailer
  class Mailer
  def initialize(mail_settings, working_hours)
    @mail_settings = mail_settings
    @working_hours = working_hours
    @mailer = Mail.new do
      to mail_settings[:to]
      from mail_settings[:from]
      subject mail_settings[:subject]
      body mail_body
    end
  end

  def mail
    @mailer.delivery_method :smtp, address: "localhost", port: 1025
    @mailer.deliver
  end

  def mail_body
    "Start #{working_hours[:start]} \n\
    Ende #{working_hours[:end]}\n\
    Pause #{working_hours[:pause]}"
  end
end

结束

这个错误来自这段代码。 enter image description here

这意味着我不能在此 block 中使用我的类方法或类变量(以 @a 开头)。

问题

block 中的执行顺序是什么?如果我设置变量 @mail_settings,我就不能在 block 中使用它。 Ruby 是否在 Mail 类中搜索 @mail_settings 我将 block 提供给该类?为什么我可以通过 block 使用 BossMailer::Mailer 构造函数中的给定参数并且没有出现错误?

如果我使用和变量将内容解析到 block 中,为什么这会起作用? (body_content = mail_body) 有效!

def initialize(mail_settings, working_hours)
  @mail_settings = mail_settings
  @working_hours = working_hours
  body_content = mail_body
  @mailer = Mail.new do
    to mail_settings[:to]
    from mail_settings[:from]
    subject mail_settings[:subject]
    body body_content
  end
end

最佳答案

一切都与上下文有关。

mail = Mail.new do
  from    'mikel@test.lindsaar.net'
  to      'you@test.lindsaar.net'
  subject 'This is a test email'
  body    File.read('body.txt')
end

fromto 方法(以及其余方法)是关于 Mail::Message 的方法实例。为了能够以这种漂亮的 DSL 方式调用它们,传递给构造函数的 block 是 instance_eval'ed。 .

这意味着在此 block 内部,self 不再是您的邮件程序,而是一条邮件消息。因此,您的邮件程序方法不可访问。

代替 instance_eval,他们可以只有 yieldblock.call,但这不会使 DSL 成为可能。

至于为什么局部变量起作用:这是因为 ruby​​ block 是词法范围的闭包(意思是,它们保留了声明的局部上下文。如果从 block 所在的位置可以看到局部变量已定义,它会在调用 block 时记住变量及其值)

替代方法

不要使用 block 形式。使用这个:https://github.com/mikel/mail/blob/0f9393bb3ef1344aa76d6dac28db3a4934c65087/lib/mail/message.rb#L92-L96

mail = Mail.new
mail['from'] = 'mikel@test.lindsaar.net'
mail[:to]    = 'you@test.lindsaar.net'
mail.subject 'This is a test email'
mail.body    = 'This is a body'

代码

尝试注释/取消注释某些行。

class Mail
  def initialize(&block)
    # block.call(self) # breaks DSL
    instance_eval(&block) # disconnects methods of mailer
  end

  def to(email)
    puts "sending to #{email}"
  end

end

class Mailer
  def admin_mail
    # get_recipient = 'vasya@example.com'
    Mail.new do
      to get_recipient
    end
  end

  def get_recipient
    'sergio@example.com'
  end
end


Mailer.new.admin_mail

关于ruby - Ruby中 block 的使用顺序是什么,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31650909/

相关文章:

ruby-on-rails - 我如何在邮件程序之外使用 Devise 的 `confirmation_url`?

ruby - Ruby 中的哈希语法

ruby-on-rails - 清理 Controller 以加速应用程序

oracle:COMMENT 的 Oracle 等效数据类型是什么?

c++ - WriteFile() block (通过命名管道从 C++ 客户端写入到 C# 服务器)

ruby - 在 Ruby 中用键盘诅咒数组浏览

ruby - 重构对 Ruby 的 RestClient 的 CURL 请求

css - 使用 .htaccess 阻止目录

java - 与 Objective-C block 相比,Java 匿名类的局限性

Ruby block - 除非和如果比较