我正在创建一个 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
结束
这意味着我不能在此 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
from
、to
方法(以及其余方法)是关于 Mail::Message
的方法实例。为了能够以这种漂亮的 DSL 方式调用它们,传递给构造函数的 block 是 instance_eval'ed。 .
这意味着在此 block 内部,self
不再是您的邮件程序,而是一条邮件消息。因此,您的邮件程序方法不可访问。
代替 instance_eval
,他们可以只有 yield
或 block.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/