이 기사에서는 MySQL 데이터베이스 성능 향상을 위한 아이디어를 살펴보고 8가지 측면에서 구체적인 솔루션을 제공합니다.
1. 가장 적합한 필드 속성을 선택하십시오.
MySQL은 대량의 데이터에 대한 액세스를 잘 지원할 수 있지만 일반적으로 데이터베이스의 테이블이 작을수록 쿼리 실행 속도가 빨라집니다. 따라서 테이블을 생성할 때 더 나은 성능을 얻기 위해 테이블의 필드 너비를 최대한 작게 설정할 수 있습니다. 예를 들어 우편 번호 필드를 정의할 때 이를 CHAR(255)로 설정하면 데이터베이스에 불필요한 공간이 추가됩니다. CHAR(6)은 괜찮기 때문에 VARCHAR 유형을 사용해도 중복됩니다. 마찬가지로 가능하다면 BIGIN 대신 MEDIUMINT를 사용하여 정수 필드를 정의해야 합니다.
효율성을 높이는 또 다른 방법은 가능한 경우 필드를 NOT NULL로 설정하여 나중에 쿼리를 실행할 때 데이터베이스가 NULL 값을 비교할 필요가 없도록 하는 것입니다.
"지방" 또는 "성별"과 같은 일부 텍스트 필드의 경우 ENUM 유형으로 정의할 수 있습니다. 왜냐하면 MySQL에서는 ENUM 유형이 숫자 데이터로 처리되고, 숫자 데이터는 텍스트 유형보다 훨씬 빠르게 처리되기 때문입니다. 이런 방식으로 데이터베이스의 성능을 향상시킬 수 있습니다.
2. 하위 쿼리(Sub-Queries) 대신 조인(JOIN)을 사용하십시오.
MySQL은 4.1부터 SQL 하위 쿼리를 지원합니다. 이 기술을 사용하면 SELECT 문을 사용하여 쿼리 결과의 단일 열을 만든 다음 이 결과를 다른 쿼리의 필터 조건으로 사용할 수 있습니다. 예를 들어 기본 고객 정보 테이블에서 주문이 없는 고객을 삭제하려면 하위 쿼리를 사용하여 먼저 판매 정보 테이블에서 주문을 한 모든 고객의 ID를 검색한 후 그 결과를 다음으로 전달하면 됩니다. 아래와 같이 기본 쿼리:
DELETE FROM customerinfo
WHERE CustomerID NOT in (SELECT CustomerID FROM salesinfo)
하위 쿼리를 사용하면 논리적으로 한 번에 여러 단계를 완료해야 하는 많은 SQL 작업을 완료할 수 있으며 트랜잭션 또는 테이블 잠금을 피할 수도 있고 작성하기도 쉽습니다. 그러나 경우에 따라 하위 쿼리를 보다 효율적인 조인(JOIN)으로 대체할 수 있습니다. 예를 들어 주문 기록이 없는 모든 사용자를 검색하려는 경우 다음 쿼리를 사용할 수 있습니다.
SELECT * FROM customerinfo
CustomerID가 없는 곳(판매 정보에서 CustomerID 선택)
연결(JOIN)을 사용하여 이 쿼리를 완료하면 속도가 훨씬 빨라집니다. 특히 salesinfo 테이블에 CustomerID에 대한 인덱스가 있는 경우 쿼리는 다음과 같습니다.
SELECT * FROM customerinfo
왼쪽 가입 salesinfoON customerinfo.CustomerID=salesinfo.
고객ID
salesinfo.CustomerID가 NULL인 경우
데이터 경로를 설정하고 데이터베이스 데이터 파일을 공유 NFS 디렉터리(NAS 서버)에
배치해야 서비스를 정상적으로 시작하고 중지할 수 있습니다.
1125 vi /etc/ my.cnf
[mysqld]
#데이터베이스 데이터 설치 위치
datadir=/data/mysqldata
#innoDB 엔진 설치 위치
innodb_data_home_dir = /usr/local/mysql/data
innodb_log_group_home_dir= /usr/local/mysql/data
innodb_data_file_path=ibdata1: 50M;ibdata2:50M:autoextend
1106 cp ./support-files/mysql.server /etc/rc.d/init.d/
vi /etc/rc.d/init.d/mysql.server
컴파일 관련 222번째부터 시작하는 항목 두 줄은 서버의 로컬 디렉터리에 PID 파일을 배치합니다:
pid_file=/usr/local/mysql/data/mysqlmanager-`/bin/hostname`.pid
server_pid_file=/usr/local/mysql /data/`/bin/hostname`.pid는
MySQL의 기본 데이터베이스를 설치합니다:
1123 mount 10.4.66.251:/data /data
1124 mkdir /data/mysqldata
1127 ./scripts/mysql_install_db --user=mysql
1145 chown -R mysql .mysql /data/mysqldata/
if 정상이면 mysql이 정상적으로 시작되는 것을 볼 수 있습니다.
1146 /etc/rc.d/init.d/mysql.server start
1146 /etc/rc.d/init.d/mysql.server;
HA 고가용성 구성을중지하고
설정하지 마십시오. NFS 마운트 및 서버 시작 시 mysql 서비스가 자동으로 실행됩니다.
5. 테이블 잠금
트랜잭션은 데이터베이스의 무결성을 유지하는 데 매우 좋은 방법이지만 독점성으로 인해 때때로 영향을 줍니다. 특히 대규모 애플리케이션 시스템에서 매우 오랜 시간 동안 데이터베이스 성능이 저하됩니다. 트랜잭션이 실행되는 동안 데이터베이스가 잠기므로 다른 사용자 요청은 트랜잭션이 끝날 때까지만 기다릴 수 있습니다. 소수의 사용자만이 데이터베이스 시스템을 사용한다면 거래에 미치는 영향은 큰 문제가 되지 않으나, 전자상거래 웹사이트에 접속하는 등 수천 명의 사용자가 동시에 데이터베이스 시스템에 접속한다면 거래에 미치는 영향은 심각할 것입니다. 응답 지연.
실제로 어떤 경우에는 테이블을 잠그면 더 나은 성능을 얻을 수 있습니다. 다음 예에서는 잠금 테이블 방법을 사용하여 이전 예의 트랜잭션 기능을 완료합니다.
LOCK TABLE 인벤토리 쓰기
재고에서 수량 선택
WHEREItem='책';
...
UPDATE 재고 SET 수량=11
WHEREItem='책';
테이블 잠금 해제
여기서는 SELECT 문을 사용하여 초기 데이터를 검색하고, 일부 계산을 통해 UPDATE 문을 사용하여 테이블에 새 값을 업데이트합니다. WRITE 키워드가 포함된 LOCK TABLE 문은 UNLOCK TABLES 명령이 실행되기 전에 인벤토리에 대한 다른 액세스가 삽입, 업데이트 또는 삭제되지 않도록 보장합니다.
6.
외래 키를 사용하여 테이블을 잠그는 방법은 데이터의 무결성을 유지할 수 있지만 데이터의 관련성을 보장할 수는 없습니다. 이때 외래 키를 사용할 수 있습니다. 예를 들어 외래 키를 사용하면 각 판매 레코드가 기존 고객을 가리키도록 할 수 있습니다. 여기서 외래 키는 customerinfo 테이블의 CustomerID를 salesinfo 테이블의 CustomerID에 매핑할 수 있습니다. 유효한 CustomerID가 없는 레코드는 salesinfo에 업데이트되거나 삽입되지 않습니다.
테이블 만들기 고객 정보
(
고객 ID INT NOT NULL ,
기본 키(고객 ID)
) 유형 = INNODB
판매정보 생성
;
(
SalesID INT NOT NULL,
고객 ID INT는 NULL이 아닙니다.
PRIMARY KEY(고객ID, 판매ID),
외래 키(CustomerID) 참조 customerinfo
(CustomerID) DELETECASCADE 사용 중
) 유형 = INNODB;
예제에서 "ON DELETE CASCADE" 매개변수를 참고하세요. 이 매개변수는 customerinfo 테이블의 고객 레코드가 삭제될 때 salesinfo 테이블의 고객과 관련된 모든 레코드도 자동으로 삭제되도록 보장합니다. MySQL에서 외래 키를 사용하려면 테이블을 생성할 때 테이블 유형을 트랜잭션에 안전한 InnoDB 유형으로 정의해야 합니다. 이 유형은 MySQL 테이블의 기본 유형이 아닙니다. 정의된 방법은 CREATE TABLE 문에 TYPE=INNODB를 추가하는 것입니다. 예에 표시된 것처럼.
7. 인덱스를 사용하는 것은
데이터베이스 성능을 향상시키는 일반적인 방법입니다. 특히 명령을 실행할 때 쿼리 문에 MAX(), MIN() 및 ORDERBY가 포함된 경우 데이터베이스 서버가 인덱스 없이 특정 행을 훨씬 빠르게 검색할 수 있습니다. 성능 향상이 더 분명해졌습니다. 그렇다면 어떤 필드를 색인화해야 할까요? 일반적으로 인덱스는 JOIN, WHERE 판단 및 ORDER BY 정렬에 사용되는 필드에 구축되어야 합니다. 중복된 값이 많이 포함된 데이터베이스의 필드를 색인화하지 마십시오. ENUM 유형 필드의 경우 customerinfo의 "province".. 필드와 같이 많은 수의 중복 값이 있을 가능성이 매우 높습니다. 반대로 이러한 필드에 인덱스를 작성하는 것은 도움이 되지 않습니다. 데이터베이스 성능을 줄입니다. 테이블을 생성할 때 적절한 인덱스를 동시에 생성할 수도 있고, 나중에 ALTER TABLE 또는 CREATE INDEX를 사용하여 인덱스를 생성할 수도 있습니다. 게다가 MySQL은
전체 텍스트 인덱싱 및 검색은 버전 3.23.23부터 지원됩니다. 전체 텍스트 인덱스는 MySQL의 FULLTEXT 유형 인덱스이지만 MyISAM 유형 테이블에만 사용할 수 있습니다. 대규모 데이터베이스의 경우 FULLTEXT 인덱스 없이 테이블에 데이터를 로드한 다음 ALTER TABLE 또는 CREATE INDEX를 사용하여 인덱스를 생성하는 것이 매우 빠릅니다. 그러나 이미 FULLTEXT 인덱스가 있는 테이블에 데이터를 로드하면 실행 프로세스가 매우 느려집니다.
8. 최적화된 쿼리문
대부분의 경우 인덱스를 사용하면 쿼리 속도가 향상되지만, SQL문을 잘못 사용하면 인덱스가 제 역할을 하지 못하게 됩니다. 다음은 주의를 기울여야 할 몇 가지 측면입니다. 첫째, 동일한 유형의 필드 간에 비교 작업을 수행하는 것이 가장 좋습니다. MySQL 버전 3.23 이전에는 이는 필수 조건이기도 했습니다. 예를 들어, 인덱스된 INT 필드는 BIGINT 필드와 비교할 수 없지만 특수한 경우 CHAR 유형 필드와 VARCHAR 유형 필드의 크기가 동일한 경우 비교할 수 있습니다. 둘째, 인덱스된 필드에서 작동하는 함수를 사용하지 마십시오.
예를 들어, DATE 유형 필드에 YEAE() 함수를 사용하면 인덱스가 제대로 작동하지 않습니다. 따라서 다음 두 쿼리는 동일한 결과를 반환하지만 후자가 전자보다 훨씬 빠릅니다.
SELECT * FROM 주문 WHERE YEAR(OrderDate)<2001;
SELECT * FROM order WHERE OrderDate<"2001-01-01";
숫자 필드를 계산할 때도 동일한 상황이 발생합니다.
SELECT * FROM Inventory WHERE Amount/7<24;
SELECT * FROM Inventory WHERE Amount<24*7;
위의 두 쿼리도 동일한 결과를 반환하지만 후자의 쿼리는 이전 쿼리보다 훨씬 빠릅니다. 셋째, 문자 필드를 검색할 때 LIKE 키워드와 와일드카드를 사용하는 경우가 있습니다. 이 접근 방식은 간단하지만 시스템 성능이 저하되기도 합니다. 예를 들어 다음 쿼리는 테이블의 모든 레코드를 비교합니다.
SELECT * 도서에서
"MySQL%"와 같은 WHERE 이름
그러나 다음 쿼리를 사용하면 반환되는 결과는 동일하지만 속도가 훨씬 빠릅니다. ..
SELECT * FROM books
WHERE name>="MySQL" 및 name<"MySQM"
마지막으로, MySQL이 쿼리에서 자동 유형 변환을 수행하지 않도록 주의해야 합니다. 변환 프로세스로 인해 인덱스도 무효화되기 때문입니다.