Rails的validates工作原理

Rails validates工作原理

有两种Active Record对象:与数据库中的行对应的那些对象和不对应的那些对象。当您创建新对象时,例如使用该new方法,该对象尚不属于该数据库。一旦调用save该对象,它将被保存到相应的数据库表中。Active Record使用new_record?实例方法来确定对象是否已存在于数据库中。考虑以下简单的Active Record类:

class Person < ApplicationRecord
end

我们可以通过查看一些rails console输出来了解它是如何工作的:

$ bin/rails console
>> p = Person.new(name: "John Doe")
=> #<Person id: nil, name: "John Doe", created_at: nil, updated_at: nil>
>> p.new_record?
=> true
>> p.save
=> true
>> p.new_record?
=> false

创建和保存新记录将向INSERT数据库发送SQL 操作。更新现有记录将UPDATE改为发送SQL 操作。通常在将这些命令发送到数据库之前运行验证。如果任何验证失败,该对象将被标记为无效,Active Record将不执行INSERT或UPDATE操作。这可以避免在数据库中存储无效对象。您可以选择在创建,保存或更新对象时运行特定验证。

以下方法触发验证,并且仅当对象有效时才将对象保存到数据库:

  • create
  • create!
  • save
  • save!
  • update
  • update!

我的Rails代码

我在html表单中判断一个@diagnose_task.errors.any?以决定是否执行表单错误提示。

.card
  .card-body.pb-1
    = form_tag diagnose_create_path, remote: true do
      - if @diagnose_task.errors.any?
        #error_explanation.mr-2.mb-2.form-control.bg-white
          %h2= "发生了#{@diagnose_task.errors.count} 个错误"
          -#%h2= "#{pluralize(@diagnose_task.errors.count, "error")} prohibited this white_list from being saved:"
          %ul
            - @diagnose_task.errors.full_messages.each do |message|
              %li= message

提交表单时会调用@diagnose_task的create方法,而@diagnose_task对象在create的时候会在diagnose_task model中调用事先写好的判断用户输入ip地址是否合法的validate方法:

validates :desc, presence: true, format: { with: /调查线索:((25[0-5]|2[0-4]\d|((1\d{2})|([1-9]?\d)))\.){3}(25[0-5]|2[0-4]\d|((1\d{2})|([1-9]?\d)))/,
                                                                  message: "ip地址非法" }

当用户输入ip非法时validate方法会在diagnose_task.error中写入相应的值,并且@diagnose_task.create方法失败,重新render表单的html。从而实现表单错误信息提示,如下图所示:
Rails的validates工作原理