我有以下型号:
class Lyric < ActiveRecord::Base
belongs_to :user
belongs_to :song
after_create :add_to_song
end
class Song < ActiveRecord::Base
belongs_to :user
has_many :lyrics
end
这个想法是用户可以为一首歌添加任意数量的歌词。如果为该用户尚不存在的新歌曲输入歌词,则会为该用户创建一首新歌曲。这是通过调用 after_create 方法 'add_to_song' 来实现的,该方法检查用户是否有该歌曲的歌词:
def add_to_song
sl = self.song_line
# Check for adjacent songs
prior_song = Song.where(:user_id => self.user.id,
:title=> sl.title,
:artist => sl.artist,
:last_line => sl.linenum-1).first
next_song = Song.where(:user_id => self.user.id,
:title=> sl.title,
:artist => sl.artist,
:frst_line => sl.linenum+1).first
# Case 1 - No existing song
if !prior_song && !next_song
song = Song.create!(:user_id => self.user.id,
:length => 1,
:title=> sl.title,
:artist => sl.artist,
:frst_line => sl.linenum,
:last_line => sl.linenum )
self.update_attribute( :song_id, song.id )
# Case 2 - Lyric is between two songs -> merge songs
elsif prior_song && next_song
prior_song.absorb( next_song, self )
# Case 3 - Lyric is new first lyric of existing song
elsif next_song
next_song.expand( self )
# Case 4 - Lyric is new last lyric of existing song
else
prior_song.expand( self )
end
end
如果链接歌词是由用户添加的,那么 add_to_song 方法还会将两首“歌曲”合二为一。换句话说,如果用户拥有一首歌曲的第 1 行和第 3 行,则在她添加同一首歌曲的第 2 行之前,它们将被视为两首不同的歌曲。
问题
当用户同时从同一首歌曲中添加多个歌词时(通过从搜索结果中选择其中的一些歌词),MySQL 中偶尔会出现竞争条件,并且为同一首歌曲实例化两个歌曲模型,即使歌词彼此相邻并且应该组合成一首“歌曲”。(不幸的结果是歌词以正确的顺序呈现。)
我已经阅读了关于乐观与悲观锁定等的无穷无尽的帖子,并尝试了各种选择,但似乎无法摆脱这个问题。每次用户创建歌词时锁定整个 Song 表似乎有点过头了。
这是防止这种情况发生的唯一方法吗?(这似乎对性能造成了巨大影响)。我的架构中是否存在根本性错误?我想这是许多项目中的常见问题,但据我所知,它似乎并没有出现太频繁。似乎任何时候在 after_create 方法中实例化父关联时,如果父模型(在本例中为 Song)的创建依赖于另一个子模型(在本例中,歌词)。