posts
테이블의 글을 게시판별로 분류하기 위해서는 Bulletin
모델과 Post
모델 사이에 일대다의 관계선언을 해 줄 필요가 있다.
app/models/bulletin.rb
파일을 열고 아래와 같이 추가한다.
class Bulletin < ActiveRecord::Base
has_many :posts, dependent: :destroy
end
app/models/post.rb
파일을 열고 아래와 같이 추가한다.
class Post < ActiveRecord::Base
belongs_to :bulletin
end
두 모델 간의 관계선언을 할 때는 단복수에 주의해야 한다. 위에서와같이
has_many
다음에는 항상 복수형(posts
)을 지정해야 하고,belongs_to
다음에는 항상 단수형(bulletin
)으로 지정해야 한다.
관계선언을 하는 매크로 스타일의 클래스 메소드인 has_many
와 belongs_to
는 매우 직관적이어서 별도의 설명이 필요치 않다. 단, has_many
의 경우 dependent
옵션을 지정할 수 있는데, :destroy
로 지정할 경우, 특정 bulletin
레코드를 삭제할 때 이 게시판에 속하는 모든 posts
도 동시에 삭제된다.
단, 이와 같이 두 모델의 관계를 선언하는 것만으로 실제 DB 테이블이 자동으로 연결되지 않는다. 즉, 관계형 데이터베이스에서 두 테이블이 관계를 가지기 위해서는 자식 테이블 필드 중에 부모 테이블의
id
를 외래키(foreign key)로 가지고 있어야 한다.
액티브레코드를 통해서 이 두 모델이 연결되는 각각의 테이블을 물리적으로 연결하기 위해서는 posts
테이블에 bulletin_id
(모델명 + 'id')란 정수형(integer형)의 필드를 추가해 주어야 한다.
레일스의 모든 모델 클래스는 해당 테이블의
primary key
가id
인 것으로 가정한다. 이것은 레일스의 규칙이다. 물론id
이외의 다른 키를primary key
로 사용할 수 있지만, 부가적인 작업을 더 해주어야 하기 때문에 불편하다. 따라서 레일스 나라에서 살려면 레일스의 법을 준수할 필요가 있는 것이다.
따라서 posts
테이블에 bulletin_id
필드를 추가하기 위해 마이그레이션 파일을 작성한다.
$ bin/rails g migration add_bulletin_id_to_posts bulletin_id:integer:index
invoke active_record
create db/migrate/20150130114743_add_bulletin_id_to_posts.rb
bulletin_id:integer:index
와 같이 추가할 필드명과 데이터형 다음에 index
옵션을 지정하면 해당 필드에 대한 인덱스 파일이 지정되며, 이는 빠른 검색을 가능하게 한다.
생성된 마이그레이션 파일(db/migrate/(생성된 일자가 포함된 일련의 숫자)_add_bulletin_id_to_posts.rb
)은 아래와 같다.
class AddBulletinIdToPosts < ActiveRecord::Migration
def change
add_column :posts, :bulletin_id, :integer
add_index :posts, :bulletin_id
end
end
이 마이그레이션 파일에 대해서 db:migrate
작업을 한다.
$ bin/rake db:migrate
== 20140509005154 AddBulletinIdToPosts: migrating =============================
-- add_column(:posts, :bulletin_id, :integer)
-> 0.0035s
-- add_index(:posts, :bulletin_id)
-> 0.0004s
== 20140509005154 AddBulletinIdToPosts: migrated (0.0040s) ====================
지금까지 작업한 것이 제대로 동작하는지를 확인하기 위해서 터미널에서 아래와 같이 레일스 콘솔을 실행해 보자.
$ bin/rails console
Loading development environment (Rails 4.2.0)
irb(main):001:0> bulletin = Bulletin.new
=> #<Bulletin id: nil, title: nil, description: nil, created_at: nil, updated_at: nil>
그리고 bulletin.post
까지 입력한 후 <tab>
키를 눌러보면 사용할 수 있는 메소드들을 볼 수 있다.
irb(main):002:0> bulletin.post
bulletin.post_ids bulletin.post_ids= bulletin.posts bulletin.posts=
이 메소드들은 두 모델 관계선언으로 인해 자동으로 생성된다.
bulletin.post_ids
: 이 메소드는 특정bulletin
객체에 속하는 모든post
객체들의id
값들을 배열로 반환한다. 현재는 빈배열을 반환할 것이다.bulltein.post_ids=
: 이 메소드는post
객체들의id
값들을 요소로하는 배열을 할당해 주어, 해당post
객체들이 이bulletin
객체의 자식 객체들로 등록되도록 한다.bulletin.posts
: 이 메소드는 특정bulletin
객체에 속하는 모든post
객체들을 배열로 반환한다. 현재는 빈배열(#<ActiveRecord::Associations::CollectionProxy []>
)을 반환할 것이다.bulletin.posts=
: 이 메소드는post
객체들를 요소로하는 배열를 할당해 주어, 해당post
객체들이 이bulletin
객체의 자식 객체들로 등록되도록 한다.
이상으로 두 모델의 관계선언을 완성하였다.
특정 게시판에 하나의 글을 추가한다고 가정해 보자. 우선, 레일스 콘솔을 열고 앞에서 생성했던 공지사항
이라는 bulletin
객체를 불러 온다.
$ bin/rails console
Loading development environment (Rails 4.2.0)
irb(main):001:0> bulletin = Bulletin.first
Bulletin Load (0.1ms) SELECT "bulletins".* FROM "bulletins" ORDER BY "bulletins"."id" ASC LIMIT 1
=> #<Bulletin id: 1, title: "공지사항", description: "공지사항을 입력하는 게시판입니다. ", created_at: "2015-01-30 10:55:47", updated_at: "2015-01-30 10:55:47">
그리고 post
객체도 하나 생성하자.
irb(main):003:0> post = Post.create title:"레일스 가이드라인 책 집필", content:"초보자를 위한 레일스"
(0.0ms) begin transaction
SQL (0.2ms) INSERT INTO "posts" ("content", "created_at", "title", "updated_at") VALUES (?, ?, ?, ?) [["content", "초보자를 위한 레일스 가이드라인"], ["created_at", "2014-05-04 09:06:54.879852"], ["title", "레일스 가이드라인 책 집필"], ["updated_at", "2014-05-04 09:06:54.879852"]]
(1.3ms) commit transaction
=> #<Post id: 2, title: "레일스 가이드라인 책 집필", content: "초보자를 위한 레일스 가이드라인", created_at: "2014-05-04 09:06:54", updated_at: "2014-05-04 09:06:54", bulletin_id: nil>
현재는 post.bulletin_id
값이 nil
이기 때문에,
post
객체와 bulletin
객체가 물리적으로 연결되지 않은 상태이다.
irb(main):004:0> post.bulletin_id
=> nil
이 문제를 해결하기 위해서 post.bulletin_id=
메소드에 bulletin.id
값을 할당한다.
irb(main):005:0> bulletin.id
=> 1
irb(main):006:0> post.bulletin_id = bulletin.id
=> 1
irb(main):007:0> post.save
(0.1ms) begin transaction
SQL (0.4ms) UPDATE "posts" SET "bulletin_id" = ?, "updated_at" = ? WHERE "posts"."id" = ? [["bulletin_id", 1], ["updated_at", "2015-01-30 11:59:53.609051"], ["id", 1]]
(1.6ms) commit transaction
=> true
이제 아래와 같이 두 모델의 관계선언이 제대로 설정되었는지를 확인해 보자.
irb(main):010:0> bulletin.posts
Post Load (0.2ms) SELECT "posts".* FROM "posts" WHERE "posts"."bulletin_id" = ? [["bulletin_id", 1]]
=> #<ActiveRecord::Associations::CollectionProxy [#<Post id: 2, title: "레일스 가이드라인 책 집필", content: "초보자를 위한 레일스", created_at: "2015-02-01 06:42:13", updated_at: "2015-02-01 06:42:56", bulletin_id: 1>]>
지금까지 bulletin
객체에 임의의 post
객체를 추가하는 과정을 보았다. 왠지 모르게 번잡스러운 느낌이 든다.
두 모델 클래스에서 관계선언을 한 경우에는 이러한 과정을 간단하게 해결할 수 있다.
irb(main):011:0> post = bulletin.posts.create title:"두번째 글", content: "관계선언을 이용하여 글을 등록합니다"
(0.1ms) begin transaction
SQL (0.3ms) INSERT INTO "posts" ("title", "content", "bulletin_id", "created_at", "updated_at") VALUES (?, ?, ?, ?, ?) [["title", "두번째 글"], ["content", "관계선언을 이용하여 글을 등록합니다"], ["bulletin_id", 1], ["created_at", "2015-02-01 06:45:34.648475"], ["updated_at", "2015-02-01 06:45:34.648475"]]
(1.7ms) commit transaction
=> #<Post id: 3, title: "두번째 글", content: "관계선언을 이용하여 글을 등록합니다", created_at: "2015-02-01 06:45:34", updated_at: "2015-02-01 06:45:34", bulletin_id: 1>
irb(main):012:0> bulletin.posts
=> #<ActiveRecord::Associations::CollectionProxy [#<Post id: 2, title: "레일스 가이드라인 책 집필", content: "초보자를 위한 레일스", created_at: "2015-02-01 06:42:13", updated_at: "2015-02-01 06:42:56", bulletin_id: 1>, #<Post id: 3, title: "두번째 글", content: "관계선언을 이용하여 글을 등록합니다", created_at: "2015-02-01 06:45:34", updated_at: "2015-02-01 06:45:34", bulletin_id: 1>]>
irb(main):014:0> bulletin.posts.size
=> 2
즉, bulletin.posts.create
와 같이 post
를 생성하면 생성되는 post
객체의 bulletin_id
속성이 bulletin.id
값으로 자동으로 지정되기 때문에, post.bulletin_id=
에 bulletin.id
값을 할당하는 과정이 필요 없게 된다.