데이터베이스: 쿼리 빌더
- 시작하기
- 결과 질의
- Selects-선택
- Joins-조인
- Unions-유니온
- Where-구문
- Ordering, Grouping, Limit, & Offset
- Inserts-삽입
- Updates-수정
- Deletes-삭제
- Pessimistic Locking
시작하기
데이터베이스 쿼리 빌더는 데이터베이스 쿼리들을 만들고 운영하는데 다양한 인터페이스의 편의기능를 제공합니다. 여러분들의 애플리케이션에서 운영하는 대부분의 데이터베이스 시스템에서 잘 작동합니다.
참고: 라라벨의 쿼리 빌더는 PDO 파라미터 바인딩을 사용하여 SQL injection 공격을 방지합니다. 따라서 쿼리에 바인딩할 문자열들을 따로 정리하고 전달할 필요가 없습니다.
결과 질의
한 테이블에서 모든 행들을 가져오기
쿼리 구성을 시작하기 위해서 DB
파사드의 table
메소드를 사용합니다. table
메소드는 주어진 테이블에 대한 쿼리빌더 인스턴스를 반환하고, 결과를 가져올 수 있도록 다양한 제약을 메소드 체이닝 형태로 구성할 수 있습니다. 다음과 같이, 테이블의 모든 레코드를 get
을 통해서 가져올 수 있습니다.
<?php
namespace App\Http\Controllers;
use DB;
use App\Http\Controllers\Controller;
class UserController extends Controller
{
/**
* Show a list of all of the application's users.
*
* @return Response
*/
public function index()
{
$users = DB::table('users')->get();
return view('user.index', ['users' => $users]);
}
}
raw 쿼리와 같이 get
메소드는 각각의 결과에 대응하는 PHP StdClass
객체 인스턴스의 배열을 결과로 반환합니다. 컬럼을 나타내는 객체의 속성에 엑세스 하여 컬럼의 값을 확인할 수 있습니다:
foreach ($users as $user) {
echo $user->name;
}
테이블에서 하나의 결과 / 컬럼 가져오기
데이터베이스 테이블에서 하나의 row을 가져오고자 한다면, first
메소드를 사용하면 됩니다. 이 메소드는 하나의 StdClass
객체를 반환할 것입니다:
$user = DB::table('users')->where('name', 'John')->first();
echo $user->name;
전체 row 가 필요하지 않다면, value
메소드를 사용하여 레코드에서 하나의 값만 추출할 수 있습니다. 메소드는 컬럼의 직접적인 값을 반환할 것입니다:
$email = DB::table('users')->where('name', 'John')->value('email');
테이블에서 결과데이터 분할하기
데이터베이스 레코드가 많은 작업을 수행해야 한다면, chunk
메소드를 사용하는 것을 고려하십시오. 이 메소드는 한번에 결과에 대한 하나의 작은 "chunk" 를 획득하고, 각각의 chunk를 Closure
를 통해서 처리합니다. 이 메소드는 다수의 레코드를 처리하는 아티즌 명령어를 작성하는데 유용합니다. 예를 들어, 전체 users
테이블에서 한번에 100개의 레코드를 가져온다고 해보겠습니다.
DB::table('users')->chunk(100, function($users) {
foreach ($users as $user) {
//
}
});
Closure
에서 false
를 반환하여, 더이상의 chunk를 처리하지 않도록 중단할 수 있습니다:
DB::table('users')->chunk(100, function($users) {
// Process the records...
return false;
});
컬럼 값의 목록 조회하기
한개의 컬럼의 값들을 포함하고 있는 배열을 조회하고자 한다면, lists
메소드를 사용할 수 있습니다. 이 예제에서는 사용자의 역할의 이름의 배열을 조회할 것입니다:
$titles = DB::table('roles')->lists('title');
foreach ($titles as $title) {
echo $title;
}
반환될 배열에서 사용될 특정한 컬럼 키를 지정할 수도 있습니다:
$roles = DB::table('roles')->lists('title', 'name');
foreach ($roles as $name => $title) {
echo $title;
}
Aggregates-집계문
쿼리빌더는 또한 count
, max
, min
, avg
그리고 sum
과 같은 다양한 집계 관련 메소드를 제공합니다. 이 메소드들은 쿼리를 생성한 뒤에 호출하면 됩니다:
$users = DB::table('users')->count();
$price = DB::table('orders')->max('price');
물론, 쿼리를 생성하는데 이 메소드들과 다른 조건문을 합쳐서 사용할 수도 있습니다:
$price = DB::table('orders')
->where('finalized', 1)
->avg('price');
Selects-선택
Select문 지정하기
당연하게도, 항상 데이터베이스의 테이블에서 모든 컬럼을 조회하고자 하지는 않을 것입니다. select
메소드를 사용하여, 쿼리에서 select
문을 위한 지정을 할 수 있습니다.
$users = DB::table('users')->select('name', 'email as user_email')->get();
The distinct
method allows you to force the query to return distinct results:
distinct
메소드는 쿼리가 고유한 결과를 반환하도록 강제할 수 있습니다:
$users = DB::table('users')->distinct()->get();
이미 쿼리빌더 인스턴스를 가지고 있고, 존재하는 select 구문에 선택할 컬럼을 추가하고자 한다면, addSelect
메소드를 사용할 수 있습니다:
$query = DB::table('users')->select('name');
$users = $query->addSelect('age')->get();
직접 질의구문(Raw Expressions)
때로는, 쿼리에서 Raw Expressions를 사용하고자 할 수도 있습니다. 이러한 구문들은 쿼리에 문자열 형태로 주입될 것이기 때문에, SQL 인젝션 공격을 유발하지 않도록 주의하시기 바랍니다! raw expression 을 생성하기 위해서는 DB::raw
메소드를 사용할 수 있습니다:
$users = DB::table('users')
->select(DB::raw('count(*) as user_count, status'))
->where('status', '<>', 1)
->groupBy('status')
->get();
Joins-조인
Inner Join 구문
쿼리 빌더를 사용해서 조인문(join statement)을 만들수 있습니다. 기본적인 SQL "inner join" 을 수행하기 위해서는 쿼리 빌더 인스턴스에 join
메소드를 사용하면 됩니다. join
메소드에 전달되는 첫번째 인자는 join을 수행할 테이블의 이름이며, 구 이후는 join 을 실행할 때 컬럼의 제약 조건입니다. 물론, 다음과 같이 하나의 쿼리에서 여러 테이블을 join 할 수도 있습니다:
$users = DB::table('users')
->join('contacts', 'users.id', '=', 'contacts.user_id')
->join('orders', 'users.id', '=', 'orders.user_id')
->select('users.*', 'contacts.phone', 'orders.price')
->get();
Left Join 구문
"inner join" 대신 "left join" 을 수행하고자 한다면, leftJoin
메소드를 사용하십시오. leftJoin
메소드는 join
메소드와 동일한 구성을 가집니다:
$users = DB::table('users')
->leftJoin('posts', 'users.id', '=', 'posts.user_id')
->get();
복작한 Join 구문
보다 복잡한 join 구문도 지정할 수 있습니다. 이렇게 하기 위해서, join
메소드의 두번째 인자로 Closure
를 전달하십시오. Closure
는 JoinCaluse
객체를 전달 받아, join
구문에 제약사항을 지정할 것입니다.
DB::table('users')
->join('contacts', function ($join) {
$join->on('users.id', '=', 'contacts.user_id')->orOn(...);
})
->get();
join 구문에 "where" 을 사용하고자 한다면, join 에 where
와 orWhere
메소드를 사용하면 됩니다. 두개의 컬럼을 비교하는 대신 이러한 메소드들은 값과 컬럼을 비교합니다:
DB::table('users')
->join('contacts', function ($join) {
$join->on('users.id', '=', 'contacts.user_id')
->where('contacts.user_id', '>', 5);
})
->get();
Unions-유니온
쿼리 빌더는 두 개의 쿼리를 "union" 하는 간편한 방법을 제공합니다. 다음과 같이 처음에 쿼리를 생성한 다음, 두 번째 쿼리를 결합하기 위하여 union
메소드를 사용할 수 있습니다:
$first = DB::table('users')
->whereNull('first_name');
$users = DB::table('users')
->whereNull('last_name')
->union($first)
->get();
unionAll
메소드도 union
메소드와 같은 방법으로 사용할 수 있습니다.
Where 절
간단한 Where 절
쿼리에 where
절을 추가하기 위해서는, 쿼리빌더 인스턴스에 where
메소드를 사용하면 됩니다. 가장 기본적인 where
메소드 호출 방법에는 3가지 인자가 필요합니다. 첫번째 인자는 컬럼의 이름입니다. 두번째 인자는 데이터베이스가 지원하는 연산자입니다. 세번째 인자는 컬람에 대해 비교하는 값입니다.
다음과 같이 "votes" 컬럼이 100과 일치하는 값을 가진 쿼리를 수행합니다:
$users = DB::table('users')->where('votes', '=', 100)->get();
보다 편리하게 사용하기 위해서, 컬럼이 주어진 값과 동일한가를 확인하고자 한다면, where
메소드의 두번째 인자로 바로 값을 전달할 수도 있습니다:
$users = DB::table('users')->where('votes', 100)->get();
물론, where
절을 작성할 때 다양한 다른 연산자를 사용할 수 있습니다:
$users = DB::table('users')
->where('votes', '>=', 100)
->get();
$users = DB::table('users')
->where('votes', '<>', 100)
->get();
$users = DB::table('users')
->where('name', 'like', 'T%')
->get();
Or 구문
쿼리에 or
절을 추가하는 것과 같이 where 제약에 더해서 체이닝 형태로 사용할 수 있습니다. orWhere
메소드는 where
메소드와 동일한 인자를 전달 받습니다:
$users = DB::table('users')
->where('votes', '>', 100)
->orWhere('name', 'John')
->get();
추가적인 Where 구문
whereBetween
whereBetween
메소드는 컬럼의 값이 두개의 값 사이의 값인지 확인합니다:
$users = DB::table('users')
->whereBetween('votes', [1, 100])->get();
whereNotBetween
whereNotBetween
메소드는 컬럼의 값이 두개의 값 사이가 아닌지 확인합니다:
$users = DB::table('users')
->whereNotBetween('votes', [1, 100])
->get();
whereIn / whereNotIn
whereIn
메소드는 주어진 컬럼의 값이 주어진 배열안에 값에 속해 있는지 확인합니다:
$users = DB::table('users')
->whereIn('id', [1, 2, 3])
->get();
whereNotIn
메소드는 주어진 컬럼의 값이 주어진 배열안에 포함되어 있지 않은지 확인합니다:
$users = DB::table('users')
->whereNotIn('id', [1, 2, 3])
->get();
whereNull / whereNotNull
whereNull
메소드는 주어진 컬럼의 값이 NULL
인지 확인합니다:
$users = DB::table('users')
->whereNull('updated_at')
->get();
whereNotNull
메소드는 주어진 컬럼의 값이 NULL
이 아닌지 확인합니다:
$users = DB::table('users')
->whereNotNull('updated_at')
->get();
복잡한 Where 절
Parameter Grouping
가끔은 "where exists"나 중첩된 파라미터를 그룹으로 묶어서 수행하는것과 같이, 복잡한 조건문을 사용해야 할 필요가 있습니다. 라라벨의 쿼리 빌더는 이 경우 다음과 같이 사용할 수 있습니다. 다음의 괄호 안에 제약조건을 그룹으로 묶는 예제를 살펴보겠습니다.
DB::table('users')
->where('name', '=', 'John')
->orWhere(function ($query) {
$query->where('votes', '>', 100)
->where('title', '<>', 'Admin');
})
->get();
위에서 보시다 시피, orWhere
메소드에 전달된 Closure
가 쿼리빌더의 제약조건을 그룹으로 묶고 있습니다. 이 Closure
는 괄호로 포함된 제약조건을 설정하는데 사용할 쿼리빌더 인스턴스를 전달받습니다. 이 예제는 다음과 같은 SQL을 생성할 것입니다:
select * from users where name = 'John' or (votes > 100 and title <> 'Admin')
Exists 구문
whereExists
메소드는 SQL 쿼리에 where exist
를 작성할 수 있도록 합니다. whereExists
메소드는 쿼리에 "exist" 구문을 정의할 수 있도록 쿼리 빌더를 인자로 받아들이는 Closure
를 인자로 받아들입니다.
DB::table('users')
->whereExists(function ($query) {
$query->select(DB::raw(1))
->from('orders')
->whereRaw('orders.user_id = users.id');
})
->get();
위의 쿼리는 다음과 같은 SQL을 생성합니다:
select * from users
where exists (
select 1 from orders where orders.user_id = users.id
)
Ordering, Grouping, Limit, & Offset
orderBy
orderBy
메소드는 주어진 컬럼에 대한 쿼리의 결과를 정렬합니다. orderBy
메소드의 첫번째 인자는 정렬하고자 하는 컬럼이며, 두번째 인자는 asc
또는 desc
의 정렬 방식을 제어하는 인자가 됩니다.
$users = DB::table('users')
->orderBy('name', 'desc')
->get();
groupBy / having / havingRaw
groupBy
와 hanving
메소드는 쿼리 결과를 그룹화하는데 사용합니다. having
메소드는 where
메소드와 사용장법이 비슷합니다:
$users = DB::table('users')
->groupBy('account_id')
->having('account_id', '>', 100)
->get();
havingRaw
메소드는 having
절의 값으로 raw 문자열을 설정하는데 사용됩니다. 다음과 같이, $2,500보다 많은 매출을 올린 부서를 찾을 수도 있습니다.
$users = DB::table('orders')
->select('department', DB::raw('SUM(price) as total_sales'))
->groupBy('department')
->havingRaw('SUM(price) > 2500')
->get();
skip / take
쿼리에서 반환되는 결과의 갯수를 제한하거나, 주어진 갯수만큼 결과를 건너뛰기(OFFSET
) 위해서는, skip
과 take
메소드를 사용하면 됩니다:
$users = DB::table('users')->skip(10)->take(5)->get();
Inserts-삽입
쿼리 빌더는 데이터베이스 테이블에 레코드를 추가하는 insert
메소드를 제공합니다. insert
메소드는 컬럼의 이름과 값으로 이루어진 배열을 인자로 전달받습니다.
DB::table('users')->insert(
['email' => '[email protected]', 'votes' => 0]
);
또한 한번의 insert
메소드 호출로 테이블에 여러 레코드를 추가하기 위해서 배열의 배열을 전달 할 수도 있습니다. 각각의 배열은 테이블에 추가되고자 하는 row 를 나타냅니다.
DB::table('users')->insert([
['email' => '[email protected]', 'votes' => 0],
['email' => '[email protected]', 'votes' => 0]
]);
Auto-Incrementing IDs
테이블이 auto-incrementing id를 가지고 있다면 insertGetId
메소드를 사용하여 레코드를 추가하고, 추가된 ID를 획득할 수 있습니다:
$id = DB::table('users')->insertGetId(
['email' => '[email protected]', 'votes' => 0]
);
주의: PostgreSQL에서 insertGetId 메소드를 사용하는 경우 auto-incrementing 컬럼의 이름은
id
이어야 합니다. 다른 이름의 ID를 원한다면는insertGetId
메서드의 두 번째 인자로 시퀀스 이름을 전달 하십시오.
Updates-수정
데이터베이스에 레코드를 삽입하는 것에 더해서 쿼리 빌더는 당연히 이미 존재하는 레코드를 update
메소드를 사용하여 변경할 수 있습니다. update
메소드는 insert
메소드와 마찬가지로, 업데이트 하기 위한 컬럼과 컬럼에 대한 값의 쌍으로 이루어진 배열을 인자로 전달받습니다. update
쿼리에 where
구문을 사용하여 범위를 제한할 수도 있습니다:
DB::table('users')
->where('id', 1)
->update(['votes' => 1]);
컬럼값의 증가 / 감소
쿼리빌더는 주어진 컬럼의 값을 증가 또는 감소하는 편리한 메소드를 제공합니다. 이는 간단한 단축키와 같으며, 수종으로 update
구문을 작성하는 것과 비교하면 보다 간결한 인터페이스를 제공합니다.
이 두개의 메소드는 최소한 하나의 인자를 전달 받습니다: 수정하고자 하는 컬럼. 두번째 인자는 선택적으로 전달되며, 컬럼이 증가되어야할 / 감소되어야할 양을 나타냅니다.
DB::table('users')->increment('votes');
DB::table('users')->increment('votes', 5);
DB::table('users')->decrement('votes');
DB::table('users')->decrement('votes', 5);
또한 이 작업을 수행동안 업데이트 되어야할 컬럼을 추가적으로 지정할 수도 있습니다.
DB::table('users')->increment('votes', 1, ['name' => 'John']);
Deletes-삭제
당연하게도, 쿼리빌더는 테이블에서 레코드를 삭제하는데 delete
메소드를 사용할 수 있습니다.
DB::table('users')->delete();
delete
구문에서는 delete
메소드를 호출하기 전에 where
절을 추가하여 제약사항을 설정할 수 있습니다.
DB::table('users')->where('votes', '<', 100)->delete();
모든 데이터를 삭제하고 auto-incrementing ID를 0으로 초기화 하는 것과 같이, 테이블 전체를 비우고자 한다면 truncate
메소드를 사용할 수 있습니다:
DB::table('users')->truncate();
Pessimistic Locking
쿼리빌더는 또한 여러분이 select
구문에 “Pessimistic Locking-배타적 잠금”을 설정을 할 수 있도록 도와주는 몇가지 기능을 포함하고 있습니다. "공유 lock"과 함께 구문을 실행하기 위해서, 쿼리에 sharedLock
메소드를 사용할 수 있습니다. 공유 lock 은 선택된 row 가 트랜젝션이 커밋되기 전까지 수정되는 것을 방지합니다.
DB::table('users')->where('votes', '>', 100)->sharedLock()->get();
이대신에, lockForUpdate
메소드를 사용할 수 있습니다. "수정을 위한" lock 은 row 가 수정되는 것 또는 다른 공유 lock 에 의해서 선택되는 것을 방지합니다.
DB::table('users')->where('votes', '>', 100)->lockForUpdate()->get();