我需要原子方式增加一个模型计数器并使用其新值(由Sidekiq作业处理)。
目前,我使用
Group.increment_counter :tasks_count, @task.id
在我的模型中,我会自动递增计数器的值。
但是,如果计数器的值达到了50,我还需要发送一个通知,有什么好的想法吗?锁定表/行或者有更简单的方法吗?
编辑/解决方案
基于mu is too short的回答和Rails的update_counters方法,我实现了一个实例方法(在PostgreSQL上测试过)。
def self.increment_counter_and_return_value(counter_name, id)
quoted_column = connection.quote_column_name(counter_name)
quoted_table = connection.quote_table_name(table_name)
quoted_primary_key = connection.quote_column_name(primary_key)
quoted_primary_key_value = connection.quote(id)
sql = "UPDATE #{quoted_table} SET #{quoted_column} = COALESCE(#{quoted_column}, 0) + 1 WHERE #{quoted_table}.#{quoted_primary_key} = #{quoted_primary_key_value} RETURNING #{quoted_column}"
connection.select_value(sql).to_i
end
使用方法如下:
Group.increment_counter_and_return_value(:tasks_count, @task.id)
它使用
RETURNING
在同一查询中获取新值。