데이터베이스: 쿼리 빌더

시작하기

데이터베이스 쿼리 빌더는 데이터베이스 쿼리들을 만들고 운영하는데 다양한 인터페이스의 편의기능를 제공합니다. 여러분들의 애플리케이션에서 운영하는 대부분의 데이터베이스 시스템에서 잘 작동합니다.

참고: 라라벨의 쿼리 빌더는 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 를 전달하십시오. ClosureJoinCaluse 객체를 전달 받아, join 구문에 제약사항을 지정할 것입니다.

DB::table('users')
        ->join('contacts', function ($join) {
            $join->on('users.id', '=', 'contacts.user_id')->orOn(...);
        })
        ->get();

join 구문에 "where" 을 사용하고자 한다면, join 에 whereorWhere 메소드를 사용하면 됩니다. 두개의 컬럼을 비교하는 대신 이러한 메소드들은 값과 컬럼을 비교합니다:

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

groupByhanving 메소드는 쿼리 결과를 그룹화하는데 사용합니다. 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) 위해서는, skiptake 메소드를 사용하면 됩니다:

$users = DB::table('users')->skip(10)->take(5)->get();

Inserts-삽입

쿼리 빌더는 데이터베이스 테이블에 레코드를 추가하는 insert 메소드를 제공합니다. insert 메소드는 컬럼의 이름과 값으로 이루어진 배열을 인자로 전달받습니다.

DB::table('users')->insert(
    ['email' => 'john@example.com', 'votes' => 0]
);

또한 한번의 insert 메소드 호출로 테이블에 여러 레코드를 추가하기 위해서 배열의 배열을 전달 할 수도 있습니다. 각각의 배열은 테이블에 추가되고자 하는 row 를 나타냅니다.

DB::table('users')->insert([
    ['email' => 'taylor@example.com', 'votes' => 0],
    ['email' => 'dayle@example.com', 'votes' => 0]
]);

Auto-Incrementing IDs

테이블이 auto-incrementing id를 가지고 있다면 insertGetId 메소드를 사용하여 레코드를 추가하고, 추가된 ID를 획득할 수 있습니다:

$id = DB::table('users')->insertGetId(
    ['email' => 'john@example.com', '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();