Thinkphp는 중국의 주류 프레임워크이고, 이를 사용하는 사람들이 많을 것이라고 생각합니다. 오늘은 Thinkphp에서 분산 데이터베이스에 연결하는 방법을 살펴보겠습니다.
물론 모델을 사용하여 데이터베이스를 추가, 삭제, 수정 및 쿼리하는 방법을 설명하기 위해 여기 있는 것은 아닙니다. 우리는 thinkphp의 데이터베이스 작동을 더 잘 이해하는 데 도움이 될 수 있는 기본 연결 코드를 분석하고 있습니다. 향후 사용을 용이하게 하기 위해.
1. 단일 데이터베이스 연결
사용하면 단일 데이터베이스의 연결 구성이 매우 간단합니다. 구성 파일에서 일부 정보만 구성하면 됩니다.
1 2 3 4 5 6 7 | 'DB_TYPE' => 'mysql',
'DB_HOST' => '192.168.5.102',
'DB_NAME' => 'databasename',
'DB_USER' => 'user',
'DB_PWD' => 'password',
'DB_PORT' => '3306',
'DB_PREFIX' => 'onmpw_',
|
로그인 후 복사
설정이 완료된 후 사용하시면 됩니다. 기본값은 단일 데이터베이스 연결입니다.
2. 분산 데이터베이스 연결
단일 데이터베이스 연결은 매우 간단합니다. 분산 데이터베이스 연결을 집중적으로 살펴보겠습니다.
1 2 3 4 5 6 7 8 9 10 11 | 'DB_TYPE' => 'mysql',
'DB_HOST' => '192.168.5.191,192.168.5.88,192.168.5.103',
'DB_NAME' => 'test,test,test',
'DB_USER' => 'masteruser,slaveuser,slaveuser',
'DB_PWD' => 'masterpass,slavepass,slavepass',
'DB_PORT' => '3306',
'DB_PREFIX' => '',
'DB_DEPLOY_TYPE' => 1,
'DB_RW_SEPARATE' => true,
'DB_MASTER_NUM' => 1,
'DB_SLAVE_NO' => '',
|
로그인 후 복사
위 구성에 따라 분산 데이터베이스에 연결합니다.
다음 옵션을 살펴보겠습니다
'DB_HOST'
분산형 데이터베이스의 경우 여러 대의 서버가 있는 경우 여러 개의 서버 주소를 입력하고 각 주소를 쉼표로 구분해야 합니다. 마스터-슬레이브 배포인 경우 이전 주소는 마스터 데이터베이스의 주소여야 합니다.
다음 사용자 이름, 비밀번호, 청취 포트 등은 물론 가능한 한 많이 적어 두세요. 각 데이터베이스의 사용자 이름과 비밀번호가 동일한 경우 하나만 작성할 수 있습니다.
이 옵션을 구문 분석하는 코드는 다음과 같습니다
1 2 3 4 5 6 7 8 | $_config ['username'] = explode (',', $this ->config['username']);
$_config ['password'] = explode (',', $this ->config['password']);
$_config ['hostname'] = explode (',', $this ->config['hostname']);
$_config ['hostport'] = explode (',', $this ->config['hostport']);
$_config ['database'] = explode (',', $this ->config['database']);
$_config ['dsn'] = explode (',', $this ->config['dsn']);
$_config ['charset'] = explode (',', $this ->config['charset']);
‘DB_DEPLOY_TYPE’=>1
|
로그인 후 복사
1은 분산형을 의미하고, 0은 중앙 집중형(즉, 단일 서버)을 의미합니다.
이 옵션의 구현은 ThinkDbDirver
1 2 3 4 5 6 7 8 | protected function initConnect( $master =true) {
if (! empty ( $this ->config['deploy']))
$this ->_linkID = $this ->multiConnect( $master );
else
if ( ! $this ->_linkID ) $this ->_linkID = $this ->connect();
}
|
로그인 후 복사
$this->config['deploy']는 'DB_DEPLOY_TYPE' 구성 옵션을 나타냅니다. 위 구성은 사용 전에 구문 분석되었으며 구성 항목은 모두 In입니다. $this->config 배열입니다. 구성 파일을 구문 분석하는 방법은 여기서 소개하지 않습니다. 관심 있는 분은 ThinkDb 클래스를 참조하세요.
$this->multiConnect() 함수는 분산 연결에 사용됩니다. 'DB_DEPLOY_TYPE' 옵션을 1로 설정하면 이 함수가 실행됩니다. 그렇지 않으면 $this->connect() 함수를 직접 실행하세요.
true는 읽기와 쓰기가 분리되어 있음을 의미하고, false는 읽기와 쓰기가 분리되지 않음을 의미합니다.
여기서 읽기와 쓰기의 분리는 마스터-슬레이브 데이터베이스 시스템을 기반으로 한다는 점에 유의해야 합니다. 이 옵션을 true로 설정하면 마스터 데이터베이스는 쓰고 슬레이브 데이터베이스는 읽습니다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | if ( $this ->config['rw_separate']){
if ( $master )
$r = $m ;
else {
if ( is_numeric ( $this ->config['slave_no'])) {
$r = $this ->config['slave_no'];
} else {
$r = floor (mt_rand( $this ->config['master_num'], count ( $_config ['hostname'])-1));
}
}
} else {
$r = floor (mt_rand(0, count ( $_config ['hostname'])-1));
}
|
로그인 후 복사
$this->config['rw_separate'] true인 경우 읽기와 쓰기가 분리되고, false인 경우 읽기와 쓰기가 분리되지 않습니다. 읽기와 쓰기의 분리가 왜 주인-슬레이브여야 합니까? 슬레이브 서버는 쓰기가 불가능하고 읽기만 가능하기 때문에 슬레이브 서버에 데이터를 쓰면 데이터를 동기화할 수 없습니다. 이로 인해 데이터 불일치가 발생합니다. 그러므로 우리 시스템이 마스터-슬레이브라면 읽기-쓰기 분리를 사용해야 합니다. 즉, DB_RW_SEPARATE 옵션을 true로 설정해야 한다.
1 | 'DB_MASTER_NUM'=>1
|
로그인 후 복사
이 옵션 뒤의 숫자는 읽기 및 쓰기 분리 후 기본 서버 수를 나타냅니다. 따라서 이 옵션은 마스터-슬레이브 데이터베이스 시스템에서도 사용됩니다.
아래 코드는 메인 서버를 선택합니다.
1 | $m = floor (mt_rand(0, $this ->config['master_num']-1));
|
로그인 후 복사
마스터-슬레이브 데이터베이스에서 읽을 때 슬레이브 서버에서 읽을 핵심 코드를 선택하세요
1 | $r = floor (mt_rand( $this ->config['master_num'], count ( $_config ['hostname'])-1));
|
로그인 후 복사
여기서 $this->config['master_num']는 마스터 서버 수를 나타냅니다.
1 | 'DB_SLAVE_NO'=> ''
|
로그인 후 복사
데이터를 읽는 슬레이브 서버의 일련번호를 지정하세요. 설정하지 않으면 마스터 서버 수를 기준으로 슬레이브 서버 수를 계산한 후 무작위로 하나를 선택하여 읽어옵니다.
1 2 3 4 5 6 | if ( is_numeric ( $this ->config[ 'slave_no' ])) {
$r = $this ->config[ 'slave_no' ];
} else {
$r = floor (mt_rand( $this ->config['master_num'], count ( $_config ['hostname'])-1));
}
|
로그인 후 복사
위는 각 옵션의 기능 구현코드에 대한 간단한 설명입니다.
연결 부분을 살펴보겠습니다
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | if ( $m != $r ){
$db_master = array (
'username' => isset( $_config ['username'][ $m ])? $_config ['username'][ $m ]: $_config ['username'][0],
'password' => isset( $_config ['password'][ $m ])? $_config ['password'][ $m ]: $_config ['password'][0],
'hostname' => isset( $_config ['hostname'][ $m ])? $_config ['hostname'][ $m ]: $_config ['hostname'][0],
'hostport' => isset( $_config ['hostport'][ $m ])? $_config ['hostport'][ $m ]: $_config ['hostport'][0],
'database' => isset( $_config ['database'][ $m ])? $_config ['database'][ $m ]: $_config ['database'][0],
'dsn' => isset( $_config ['dsn'][ $m ])? $_config ['dsn'][ $m ]: $_config ['dsn'][0],
'charset' => isset( $_config ['charset'][ $m ])? $_config ['charset'][ $m ]: $_config ['charset'][0],
);
}
$db_config = array (
'username' => isset( $_config ['username'][ $r ])? $_config ['username'][ $r ]: $_config ['username'][0],
'password' => isset( $_config ['password'][ $r ])? $_config ['password'][ $r ]: $_config ['password'][0],
'hostname' => isset( $_config ['hostname'][ $r ])? $_config ['hostname'][ $r ]: $_config ['hostname'][0],
'hostport' => isset( $_config ['hostport'][ $r ])? $_config ['hostport'][ $r ]: $_config ['hostport'][0],
'database' => isset( $_config ['database'][ $r ])? $_config ['database'][ $r ]: $_config ['database'][0],
'dsn' => isset( $_config ['dsn'][ $r ])? $_config ['dsn'][ $r ]: $_config ['dsn'][0],
'charset' => isset( $_config ['charset'][ $r ])? $_config ['charset'][ $r ]: $_config ['charset'][0],
);
return $this ->connect( $db_config , $r , $r == $m ? false : $db_master );
|
로그인 후 복사
이렇게 보면 위의 각 구성 옵션에 대한 코드를 소개할 때 $r과 $m의 역할을 다들 이해하셔야 할 것 같습니다.
이제 $r == $m ? false : $db_master를 살펴보겠습니다. 데이터베이스 읽기 및 쓰기가 분리되지 않고 읽기 및 쓰기가 서버인 경우 연결 함수에 전달된 값은 false입니다. 또는 마스터-슬레이브 쓰기가 분리된 경우 연결에 전달된 값도 false입니다. 위 코드에서 $r과 $m이 같지 않으면 $db_master가 설정된다는 것을 알 수 있습니다. 실제로 선택한 $r 서버에 장애가 발생하여 연결할 수 없는 경우에는 백업과 동일하게 $db_master에 연결됩니다.
connect() 함수의 세 번째 매개변수는 실제로 $db_config 서버 연결이 실패할 때 백업 연결을 선택할지 여부를 나타냅니다. False는 다시 연결되지 않음을 의미하고, 다른 값은 다시 연결을 의미합니다.
핵심 코드는 다음과 같습니다
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | try {
if ( empty ( $config ['dsn'])) {
$config ['dsn'] = $this ->parseDsn( $config );
}
if (version_compare(PHP_VERSION,'5.3.6','<=')){
$this ->options[PDO::ATTR_EMULATE_PREPARES] = false;
}
$this ->linkID[ $linkNum ] = new PDO( $config ['dsn'], $config ['username'], $config ['password'], $this ->options);
} catch (\PDOException $e ) {
if ( $autoConnection ){
trace( $e ->getMessage(),'','ERR');
return $this ->connect( $autoConnection , $linkNum );
} elseif ( $config ['debug']){
E( $e ->getMessage());
}
}
|
로그인 후 복사
이렇게 하면 마스터-슬레이브 유형의 경우 $r과 $m은 절대 동일하지 않습니다. 따라서 선택한 슬레이브 서버가 데이터를 읽을 때 장애가 발생하면 마스터 서버가 백업이 되며 결국 마스터 서버로 이동하여 데이터를 읽게 됩니다. 데이터 읽기의 적시성을 보장할 수 있습니다.
하지만 아직은 완벽하지 않다는 걸 늘 느껴요. 슬레이브 서버가 여러 개 있고 읽기 중에 선택한 슬레이브 서버와 마스터 서버가 모두 실패하면 데이터 읽기가 실패합니다. 이때 다른 슬레이브 서버를 다시 읽을 수 있으면 더 안전할 것입니다. 물론 현재 thinkphp의 기능은 매우 완벽하고 우리가 사용하기에 충분합니다. 하지만 앞으로도 thinkphp가 점점 더 완벽해지기를 바랍니다.
thinkphp를 개발에 사용하는 모든 분들에게 위의 소개가 도움이 되기를 바랍니다.
관련 권장 사항:
ThinkPHP5 프레임워크 기본 지식, 개발 사양 및 디렉터리 구조
ThinkPHP 파일 캐싱 클래스 코드 공유
ThinkPHP Db 및 모델 성능 평가
위 내용은 Thinkphp에서 분산 데이터베이스에 연결하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!