728x90

리눅스에서 제공하는 파일 인코딩 변환을 사용해봤는데 모두 ANSI 로 된 인코딩 모드를 UTF-8 로 변환하는 걸 제대로 처리하지 못한다.

 

Rocky Linux 9.5 및 CentOS 7.9 에서 모두 해봤는데 한글이 깨져서 변환처리를 한다.

 

파일을 나누어서 EditPlus 가 읽어들일 수 있는 용량 크기로 만들어서 인코딩 모드를 변환하는 것이 가장 확실한 방법이다.

 

 

 

728x90
블로그 이미지

Link2Me

,
728x90

건축데이터 민간개방 시스템 https://open.eais.go.kr/main/main.do 사이트에서 자료를 받는 방법이다.

 

 

 

건축물대장 표제부 데이터 크기가 너무 커서 EditPlus 와 같은 툴로 편집할 수가 없다. 그래서 리눅스에 파일을 올리고 파일 분할을 한 다음에 다시 EditPlus로 열어 인코딩 모드를 ANSI 에서 UTF8로 변경 저장해야 한다. 안그러면 한글이 깨져 업로드된다.

설명의 엑셀 파일을 다운로드 해서 순서대로 테이블을 설계한 것이 아래 코드이다.

 

건축물대장 표제부 데이터를 일괄 업로드하는 방법이다.

테이블 설계하는 것은 아래 제시한 것으로 이용하면 된다.

수도권 데이터만 필요해서 업로드 후 나머지 데이터는 삭제 처리했다.

준공일자가 잘못 나온 것도 모두 삭제처리했다.

 

######################################################################################
건축물대장 표제부
######################################################################################
cd /home/httpd/htdocs/sample/uploads
split -n l/3 ---additional-suffix=.txt mart_djy_03.txt djypart_
 
# editplus 에디터 툴로 열어서 ANSI 를 UTF8 로 변경하여 저장한다.
 
mysql -uroot -p
SHOW VARIABLES LIKE 'character_set%';
 
use testdb;
 
CREATE TABLE djy_blding_tsect (
  pkCode varchar(33NOT NULL COMMENT '관리건축물대장PK',
  regstrGbCd varchar(1DEFAULT NULL COMMENT '대장구분코드',
  regstrGbCdNm varchar(100DEFAULT NULL COMMENT '대장구분코드명',
  regstrKindCd varchar(1DEFAULT NULL COMMENT '대장종류코드',
  regstrKindCdNm varchar(100DEFAULT NULL COMMENT '대장종류코드명',
  jiAddress varchar(230NOT NULL COMMENT '대지위치',
  stAddress varchar(230DEFAULT NULL COMMENT '도로명대지위치',
  bldNm varchar(100DEFAULT NULL COMMENT '건물명',
  sigunguCd varchar(5NOT NULL COMMENT '시군구코드',
  bjdongCd varchar(5NOT NULL COMMENT '법정동코드',
  platGbCd char(1NOT NULL DEFAULT '0' COMMENT '대지구분코드',
  bun varchar(4NOT NULL COMMENT '번',
  ji varchar(4NOT NULL COMMENT '지',
  splotNm varchar(200DEFAULT NULL COMMENT '특수지명',
  block varchar(20DEFAULT NULL COMMENT '블록',
  lot varchar(20DEFAULT NULL COMMENT '로트',
  bylotCnt int(11DEFAULT 0 COMMENT '외필지수',
  naRoadCd varchar(12DEFAULT NULL COMMENT '새주소도로코드',
  naBjdongCd varchar(5DEFAULT NULL COMMENT '새주소법정동코드',
  naUgrndCd char(1DEFAULT '0' COMMENT '새주소지상지하코드',
  naMainBun int(11DEFAULT NULL COMMENT '새주소본번',
  naSubBun int(11DEFAULT NULL COMMENT '새주소부번',
  dongNm varchar(100DEFAULT NULL COMMENT '동명칭',
  mainAtchGbCd char(1DEFAULT '0' COMMENT '주부속구분코드',
  mainAtchGbCdNm varchar(100DEFAULT NULL COMMENT '주부속구분코드명',
  platArea double NOT NULL DEFAULT 0 COMMENT '대지면적(㎡)',
  archArea double NOT NULL DEFAULT 0 COMMENT '건축면적(㎡)',
  bcRat double NOT NULL DEFAULT 0 COMMENT '건폐율(%)',
  totArea double NOT NULL DEFAULT 0 COMMENT '연면적(㎡)',
  vlRatEstmTotArea double NOT NULL DEFAULT 0 COMMENT '용적률산정연면적(㎡)',
  vlRat double NOT NULL DEFAULT 0 COMMENT '용적률(%)',
  strctCd char(1DEFAULT NULL COMMENT '구조코드',
  strctCdNm varchar(100DEFAULT NULL COMMENT '구조코드명',
  etcStrct varchar(500DEFAULT NULL COMMENT '기타구조',
  mCode int(11DEFAULT NULL COMMENT '주용도코드',
  mCodeNM varchar(100DEFAULT NULL COMMENT '주용도코드명',
  etcType text DEFAULT NULL COMMENT '기타용도',
  roofCd varchar(2DEFAULT NULL COMMENT '지붕코드',
  roofCdNm varchar(100DEFAULT NULL COMMENT '지붕코드명',
  etcRoof varchar(500DEFAULT NULL COMMENT '기타지붕',
  hhldCnt int(11DEFAULT 0 COMMENT '세대수(세대)',
  fmlyCnt int(11DEFAULT 0 COMMENT '가구수(가구)',
  heit double NOT NULL DEFAULT 0 COMMENT '높이(m)',
  gFlrCnt int(5DEFAULT 0 COMMENT '지상층수',
  ugFlrCnt int(2DEFAULT 0 COMMENT '지하층수',
  rideUseElvtCnt int(11DEFAULT 0 COMMENT '승용승강기수',
  emgenUseElvtCnt int(11DEFAULT 0 COMMENT '비상용승강기수',
  atchBldCnt int(11DEFAULT 0 COMMENT '부속건축물수',
  atchBldArea double NOT NULL DEFAULT 0 COMMENT '부속건축물면적(㎡)',
  totDongTotArea double NOT NULL DEFAULT 0 COMMENT '총동연면적(㎡)',
  indrMechUtcnt int(11DEFAULT 0 COMMENT '옥내기계식대수(대)',
  indrMechArea double NOT NULL DEFAULT 0 COMMENT '옥내기계식면적(㎡)',
  oudrMechUtcnt int(11DEFAULT 0 COMMENT '옥외기계식대수(대)',
  oudrMechArea double NOT NULL DEFAULT 0 COMMENT '옥외기계식면적(㎡)',
  indrAutoUtcnt int(11DEFAULT 0 COMMENT '옥내자주식대수(대)',
  indrAutoArea double NOT NULL DEFAULT 0 COMMENT '옥내자주식면적(㎡)',
  oudrAutoUtcnt int(11DEFAULT 0 COMMENT '옥외자주식대수(대)',
  oudrAutoArea double NOT NULL DEFAULT 0 COMMENT '옥외자주식면적(㎡)',
  pmsDay varchar(8DEFAULT NULL COMMENT '허가일',
  stcnsDay varchar(8DEFAULT NULL COMMENT '착공일',
  useAprDay varchar(8DEFAULT NULL COMMENT '사용승인일',
  pmsnoYear varchar(4DEFAULT NULL COMMENT '허가번호년',
  pmsnoKikCd char(7DEFAULT NULL COMMENT '허가번호기관코드',
  pmsnoKikCdNm varchar(100DEFAULT NULL COMMENT '허가번호기관코드명',
  pmsnoGbCd varchar(4DEFAULT NULL COMMENT '허가번호구분코드',
  pmsnoGbCdNm varchar(100DEFAULT NULL COMMENT '허가번호구분코드명',
  hoCnt int(11DEFAULT 0 COMMENT '호수(호)',
  engrGrade varchar(4DEFAULT NULL COMMENT '에너지효율등급',
  engrRat double DEFAULT 0 COMMENT '에너지절감율',
  engrEpi int(11DEFAULT 0 COMMENT 'EPI점수',
  gnBldGrade char(1DEFAULT NULL COMMENT '친환경건축물등급',
  gnBldCert int(11DEFAULT 0 COMMENT '친환경건축물인증점수',
  itgBldGrade char(1DEFAULT NULL COMMENT '지능형건축물등급',
  itgBldCert int(11DEFAULT 0 COMMENT '지능형건축물인증점수',
  regDate varchar(8NOT NULL COMMENT '생성일자',
  rsDsgnApplyYn char(1DEFAULT '0' COMMENT '내진설계적용여부',
  rsAblty varchar(20DEFAULT NULL COMMENT '내진능력'
ENGINE=MyISAM DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci COMMENT='건축물대장 표제부';
 
ALTER TABLE djy_blding_tsect
  ADD PRIMARY KEY (pkCode),
  ADD KEY jiAddress (jiAddress),
  ADD KEY jCode (sigunguCd,bjdongCd,platGbCd,bun,ji);
COMMIT;
 
 
LOAD DATA LOCAL INFILE '/home/httpd/htdocs/sample/uploads/djypart_00.txt' 
INTO TABLE djy_blding_tsect 
FIELDS TERMINATED BY '|' 
ENCLOSED BY '"'
LINES TERMINATED BY '\n';
 
 
LOAD DATA LOCAL INFILE '/home/httpd/htdocs/sample/uploads/djypart_01.txt' 
INTO TABLE djy_blding_tsect 
FIELDS TERMINATED BY '|' 
ENCLOSED BY '"'
LINES TERMINATED BY '\n';
 
 
LOAD DATA LOCAL INFILE '/home/httpd/htdocs/sample/uploads/djypart_02.txt' 
INTO TABLE djy_blding_tsect 
FIELDS TERMINATED BY '|' 
ENCLOSED BY '"'
LINES TERMINATED BY '\n';
 
 
DELETE FROM djy_blding_tsect WHERE SUBSTRING_INDEX(jiAddress, ' '1NOT IN ('서울특별시''경기도''인천광역시');
 
# 준공일자가 8자리가 아닌 것은 모두 삭제
DELETE FROM djy_blding_tsect WHERE length(useAprDay) != 8;
 
select pkCode, max(useAprDay) from djy_blding_tsect;
 
# 준공일자가 현재 날짜보다 큰 숫자 즉 잘못 기입된 것은 삭제처리
DELETE FROM djy_blding_tsect WHERE useAprDay > '20240901';
 
OPTIMIZE TABLE djy_blding_tsect;
cs

 

 

 

728x90

'SQL' 카테고리의 다른 글

리눅스 파일 인코딩 변환  (0) 2025.01.20
건축인허가 층별 데이터 일괄 업로드하는 방법  (2) 2024.09.08
MariaDB 경로 변경  (0) 2024.08.08
DB 공간 부족으로 테이블 백업  (0) 2024.07.30
MySQL 접속 경과시간 Query  (0) 2024.05.01
블로그 이미지

Link2Me

,
728x90

건축인허가 층별 데이터를 DB에 업로드하는 방법을 PHP 코드로 구현해서 하려고 하니 엄두가 나지 않아서 MariaDB에서 제공하는 대용량 LOCAL FILE 업로드하는 방법으로 했다.

이 방법으로 하면 약 1500만개 데이터를 DB TABLE 에 데이터 저장하는데 15분 이내로 가능하다.

 

https://open.eais.go.kr/opnsvc/opnSvcInqireView.do# 에서 파일을 다운로드 한다.

매월 업데이트되는 파일을 받아서 업로드해야 하는데 중복체크 이런 걸 하려면 시간도 너무 오래걸리기 때문에 매월 자료 받으면 기존 데이터는 일괄 삭제하고 신규 데이터를 업로드하는 방식으로 하는게 낫다.

아쉽게도 파일의 인코딩모드가 euckr 로 되어 있기 때문에 파일을 나누고 EditPlus 같은 툴을 이용하여 UTF-8로 인코딩 모드를 변경한 후에 데이터 업로드를 해야 정상적으로 한글이 깨지지 않는다.

 

그리고 파일 구조 엑셀 파일을 다운로드하여 테이블 설계를 먼저 해야 한다.

테이블 설계시에는 칼럼의 개수와 순서가 반드시 일치하도록 해야 한다.

 

건축데이터 민간 개방시스템에서 제공하는 자료는 실시간 자료는 아니고 2개월 늦은 데이터라고 보면 된다.

실시간 데이터를 원하면 직접 세움터 사이트에서 조회를 해서 엑셀로 받아서 봐야 한다.

 

# 건축데이터 민간 개방시스템 대용량 파일 제공하는 게시판에서 파일을 받으면 utf-8 로 되어 있지 않다.
# 그냥 대용량 파일 업로드를 했더니 한글이 깨진다.
 
# 리눅스 파일 나누기
-n l/N 옵션: 파일 줄을 고려하여 N등분 한다.
-d 옵션: 000102 형태로 숫자로 접미사를 만든다.
--additional-suffix 옵션: 나눠진 파일명 뒤에 접미사를 추가로 붙인다.
split -n l/3 ---additional-suffix=.txt mart_kcy_03.txt kcypart_
 
# editplus 파일로 열어서 ANSI 를 UTF8 로 변경하여 저장한다.
 
mysql -uroot -p
SHOW VARIABLES LIKE 'character_set%';
 
use testdb;
 
DROP TABLE dgo_blding_floor;
CREATE TABLE dgo_blding_floor (
  pkfCode varchar(33NOT NULL COMMENT '관리_층별개요_PK',
  pkdCode varchar(33DEFAULT NULL COMMENT '관리_동별개요_PK',
  pkCode varchar(33NOT NULL COMMENT '관리_허가대장_PK',
  Address varchar(250DEFAULT NULL COMMENT '지번주소',
  bldNM varchar(100DEFAULT NULL COMMENT '건물명',
  sigungu varchar(5DEFAULT NULL COMMENT '시군구코드',
  bjdong varchar(5DEFAULT NULL COMMENT '법정동코드',
  daeji varchar(1DEFAULT NULL COMMENT '대지산구분code',
  bun varchar(4DEFAULT NULL COMMENT '번',
  ji varchar(4DEFAULT NULL COMMENT '지',
  spNM varchar(200DEFAULT NULL COMMENT '특수지명',
  block varchar(20DEFAULT NULL COMMENT '블록',
  Loot varchar(20DEFAULT NULL COMMENT '로트',
  stCode varchar(2DEFAULT NULL COMMENT '구조_코드',
  stCodeNM varchar(100DEFAULT NULL COMMENT '구조_코드명',
  mCode varchar(5DEFAULT NULL COMMENT '주용도코드',
  mCodeNM varchar(100DEFAULT NULL COMMENT '주용도코드명',
  fNO int(4NOT NULL DEFAULT 0 COMMENT '층번호',
  fArea double NOT NULL DEFAULT 0 COMMENT '층면적',
  fCode varchar(2DEFAULT NULL COMMENT '층구분코드',
  fCodeNM varchar(100DEFAULT NULL COMMENT '층구분코드명',
  archCode varchar(4DEFAULT NULL COMMENT '건축구분코드',
  archCodeNM varchar(100DEFAULT NULL COMMENT '건축구분코드명',
  regDate varchar(8DEFAULT NULL COMMENT '생성일자'
ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
ALTER TABLE dgo_blding_floor
  ADD UNIQUE KEY pkfCode (pkfCode) USING BTREE,
  ADD KEY pkCode (pkCode) USING BTREE;
 
COMMIT;
 
 
LOAD DATA LOCAL INFILE '/home/httpd/htdocs/sample/uploads/kcypart_00.txt' 
INTO TABLE dgo_blding_floor 
FIELDS TERMINATED BY '|' 
ENCLOSED BY '"'
LINES TERMINATED BY '\n';
 
 
LOAD DATA LOCAL INFILE '/home/httpd/htdocs/sample/uploads/kcypart_01.txt' 
INTO TABLE dgo_blding_floor 
FIELDS TERMINATED BY '|' 
ENCLOSED BY '"'
LINES TERMINATED BY '\n';
 
 
LOAD DATA LOCAL INFILE '/home/httpd/htdocs/sample/uploads/kcypart_02.txt' 
INTO TABLE dgo_blding_floor 
FIELDS TERMINATED BY '|' 
ENCLOSED BY '"'
LINES TERMINATED BY '\n';
 
 
DELETE FROM dgo_blding_floor WHERE SUBSTRING_INDEX(Address, ' '1NOT IN ('서울특별시''경기도''인천광역시');
 

 

728x90

'SQL' 카테고리의 다른 글

리눅스 파일 인코딩 변환  (0) 2025.01.20
건축물대장 표제부 대용량 일괄 업로드 방법  (2) 2024.09.13
MariaDB 경로 변경  (0) 2024.08.08
DB 공간 부족으로 테이블 백업  (0) 2024.07.30
MySQL 접속 경과시간 Query  (0) 2024.05.01
블로그 이미지

Link2Me

,

MariaDB 경로 변경

SQL 2024. 8. 8. 07:41
728x90

mariadb의 공간 부족으로 인해 경로를 변경하는 방법이다.

 

먼저 할당된 공간을 확인한다.

 

기본으로 mariadb가 설치되는 경로는 /var/lib/mysql 이다.

 

공간이 home 디렉토리에 비해 상대적으로 적다.

공간이 충분한 home 디렉토리 하단으로 옮기겠다.

 

 

# 1) MariaDB 서비스 정지
sudo systemctl stop mariadb
 
# 2) 새로운 Data 디렉토리 생성 및 데이터 복사하기
# Data 디렉토리를 /home/data/mysql 로 한다고 할 때
 
sudo mkdir /home/data/
sudo rsync -av /var/lib/mysql /home/data/
sudo chown -R mysql:mysql /home/data/mysql
 
# 3) 서버 설정 파일 변경
vi /etc/my.cnf.d/server.cnf
[mysqld]
datadir=/home/data/mysql
socket=/home/data/mysql/mysql.sock
log-error=/home/data/mysql/error.log
 
[client]
port=3306
socket=/home/data/mysql/mysql.sock
 
4) MairaDB /root /home 디렉토리 접근 허용하여 ProtectHome true 를 false 로 변경한다.
vi /usr/lib/systemd/system/mariadb.service
ProtectHome=false 
#wq

 

 

위와 같이 경고 메시지가 나와도 다음 단계를 진행하면 정상적으로 처리된다.

 

5) mariadb 실행하면 경고 메시지가 출력된다.
systemctl start mariadb
 
6) reload
systemctl daemon-reload
 
7) 최종 서비스 시작
systemctl start mariadb
 
8) 데이터 디렉토리 위치 변경 확인
mysql -u root -p
 
MariaDB [(none)]> select @@datadir;
 
9) 기존 mysql 폴더 bak 폴더로 변경하기
sudo mv /var/lib/mysql /var/lib/mysql.bak

 

경로가 변경된 것을 확인할 수 있다.

 

 

728x90
블로그 이미지

Link2Me

,
728x90

DB 공간부족으로 당장은 불필요하지만 나중에 활용할 수 있는 DB 테이블을 적어둔다.

 

CREATE TABLE rawpasswd (
  idx int(11NOT NULL,
  userID varchar(9NOT NULL,
  passwd text NOT NULL,
  access_date timestamp NULL DEFAULT '0000-00-00 00:00:00' COMMENT '접속시간',
  display int(2NOT NULL DEFAULT 1
ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
ALTER TABLE rawpasswd
  ADD PRIMARY KEY (idx);
 
ALTER TABLE rawpasswd
  MODIFY idx int(11NOT NULL AUTO_INCREMENT;
COMMIT;
 

 

전국의 동까지의 정보를 추출한 DB 테이블 SQL 파일

addrDB.sql
2.27MB

 

 

728x90
블로그 이미지

Link2Me

,
728x90

일정시간이 경과한 데이터는 삭제처리하는 로직을 만들기 위해서 구현해 본 쿼리다.

 

# 2시간 이내에 있는 데이터만 추출

select idx,userID,access_date from accessLog where access_date > subtime(current_timestamp(), '2:0:0') and access_date < current_timestamp();

 

# 2시간 경과한 데이터만 추출

select idx,userID,access_date from accessLog where access_date < subtime(current_timestamp(), '2:0:0');

 

# 2시간 경과한 데이터 삭제

delete from accessLog where access_date < subtime(current_timestamp(), '2:0:0');

 

 

 

 

728x90

'SQL' 카테고리의 다른 글

MariaDB 경로 변경  (0) 2024.08.08
DB 공간 부족으로 테이블 백업  (0) 2024.07.30
접속로그 통계 (신규, 중복 동시 처리)  (0) 2023.05.23
MySQL foreign key 예제  (0) 2022.05.16
[MySQL] 샘플 데이터베이스 설치하기  (0) 2022.04.28
블로그 이미지

Link2Me

,
728x90

접속로그 통계 구현을 위해서 로그를 쌓고 있는 테이블에서 데이터를 가공해야 한다.

접속로그 테이블에는 접속실패, 접속성공 등 모든 접속 시도 데이터를 저장해야 한다.

CREATE TABLE `rb_accessLog` (
  `uid` int(11NOT NULL,
  `ipaddr` varchar(20NOT NULL,
  `access_date` datetime NOT NULL COMMENT '접속시간',
  `userID` varchar(60NOT NULL,
  `userNM` varchar(20DEFAULT NULL,
  `OS` varchar(60DEFAULT NULL,
  `browser` varchar(30DEFAULT NULL,
  `hit` int(11NOT NULL DEFAULT 0,
  `success` int(1NOT NULL DEFAULT 1 COMMENT '성공실패',
  `date` char(8NOT NULL,
  `time` varchar(10NOT NULL,
  `YM` char(6NOT NULL,
  `MD` char(4NOT NULL,
  `year` char(4NOT NULL,
  `month` char(2NOT NULL,
  `day` char(2NOT NULL,
  `route` int(1NOT NULL DEFAULT 0 COMMENT '접속루트',
  `errCode` int(2NOT NULL DEFAULT 0 COMMENT '에러코드',
  `display` tinyint(2NOT NULL DEFAULT 1
ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
 
ALTER TABLE `rb_accessLog`
  ADD PRIMARY KEY (`uid`) USING BTREE,
  ADD KEY `date` (`date`),
  ADD KEY `YM` (`YM`),
  ADD KEY `MD` (`MD`);
 
 
ALTER TABLE `rb_accessLog`
  MODIFY `uid` int(11NOT NULL AUTO_INCREMENT;
COMMIT;
 

 

 

하지만 접속통계는 로그인 성공에 대한 것만 일자별로 사용자 기준으로 접속하는 테이블을 구현하면 될 것이다.

CREATE TABLE `rb_accessLog_tmp` (
  `uid` int(11NOT NULL,
  `date` char(8NOT NULL,
  `userID` varchar(60NOT NULL,
  `userNM` varchar(20DEFAULT NULL,
  `YM` char(6NOT NULL,
  `MD` char(4NOT NULL,
  `year` char(4NOT NULL,
  `month` char(2NOT NULL,
  `day` char(2NOT NULL,
  `hit` int(11NOT NULL DEFAULT 0,
  `display` tinyint(2NOT NULL DEFAULT 1
ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
 
ALTER TABLE `rb_accessLog_tmp`
  ADD PRIMARY KEY (`uid`) USING BTREE,
  ADD UNIQUE KEY `date_userID` (`date`,`userID`);
 
ALTER TABLE `rb_accessLog_tmp`
  MODIFY `uid` int(11NOT NULL AUTO_INCREMENT;
COMMIT;

 

 

데이터를 추출하여 넣은 SQL 쿼리문은 다음과 같다.

INSERT INTO rb_accessLog_tmp (date, userID, userNM, YM, MD, year, month, day,hit) 
(select date, userID, userNM, YM, MD, year, month, day, count(*) as hit 
from rb_accessLog where success=1 group by date, userID) 
ON DUPLICATE KEY UPDATE date = VALUES(date), 
userID = VALUES(userID), hit=VALUES(hit)

없는 자료는 Insert 하고 중복된 자료는 특정 칼럼만 업데이트한다.

달라질 칼럼은 hit 칼럼밖에 없을 것이다.

 

중복체크하는 중요한 Key는 date_userID 인덱스 설정이다.

 

이 테이블에서 다시 일별 사용자수 접속 통계, 접속수 통계를 구분하는 테이블을 만들어보자.

CREATE TABLE rb_access_Stats (
  uid int(11NOT NULL,
  date char(8NOT NULL,
  YM char(6NOT NULL,
  day char(2NOT NULL,
  IDCnt int(11NOT NULL DEFAULT 0,
  dailyCnt int(11NOT NULL DEFAULT 0
ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
ALTER TABLE rb_access_Stats
  ADD PRIMARY KEY (uid),
  ADD UNIQUE KEY date (date) USING BTREE;
 
ALTER TABLE rb_access_Stats
  MODIFY uid int(11NOT NULL AUTO_INCREMENT;
COMMIT;

 

UNIQUE KEY 설정은 필수 사항!!!

# 일일 통계 생성 
INSERT INTO rb_access_Stats (date,YM,day,IDCnt,dailyCnt) 
(select date, YM, day ,count(distinct userID) as IDCnt,sum(hit) as dailyCnt 
from rb_accessLog_tmp group by date)
ON DUPLICATE KEY UPDATE date = VALUES(date), IDCnt = VALUES(IDCnt), 
dailyCnt = VALUES(dailyCnt)

 

728x90

'SQL' 카테고리의 다른 글

DB 공간 부족으로 테이블 백업  (0) 2024.07.30
MySQL 접속 경과시간 Query  (0) 2024.05.01
MySQL foreign key 예제  (0) 2022.05.16
[MySQL] 샘플 데이터베이스 설치하기  (0) 2022.04.28
MariaDB 멀티 인덱스(index) 설정  (0) 2022.04.05
블로그 이미지

Link2Me

,

MySQL foreign key 예제

SQL 2022. 5. 16. 06:30
728x90

SQL 엔진에 따른 참조무결성 여부 확인 및 테이블 간에 참조 무결성 확인

##########################################################################
# 테이블 삭제 순서가 중요하다.
# DB 엔진을 MyISAM 으로 했을 경우와 InnoDB 로 했을 경우 차이점을 살펴본다.
 
DROP TABLE IF EXISTS buy, person;
CREATE TABLE person (
  userID varchar(20NOT NULL COMMENT '사용자ID',
  userNM varchar(30NOT NULL COMMENT '이름',
  mobileNO varchar(60NOT NULL COMMENT '휴대폰번호',
  reg_date timestamp NOT NULL DEFAULT current_timestamp() COMMENT '등록일자',
  display tinyint(2NOT NULL DEFAULT 1,
  PRIMARY KEY (userID)
ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
INSERT INTO person (userID, userNM, mobileNO) VALUES
('20221001''홍길동''010-1000-1001'),
('20221002''이순신''010-1000-1003'),
('20221003''강감찬''010-1000-1004'),
('20221004''신시아''010-1000-1005'),
('20221005''나현우''010-1000-1006'),
('20221006''홍석천''010-2000-1011'),
('20221007''김남일''010-2000-1012'),
('20221008''김남길''010-2000-1014'),
('20221009''박나래''010-2000-1015');
 
-- ALTER TABLE person ADD PRIMARY KEY (userID);
 
CREATE TABLE buy (
  id int(11NOT NULL AUTO_INCREMENT PRIMARY KEY,
  userID varchar(20NOT NULL,
  prod_name varchar(20NOT NULL,
  FOREIGN KEY (userID) REFERENCES person (userID)
)ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
INSERT INTO buy VALUES(NULL,'20221001','지갑');
INSERT INTO buy VALUES(NULL,'20221001','맥북');
INSERT INTO buy VALUES(NULL,'20221003','USB');
INSERT INTO buy VALUES(NULL,'20221004','가방');
INSERT INTO buy VALUES(NULL,'20221005','휴대폰');
INSERT INTO buy VALUES(NULL,'20221005','의류');
INSERT INTO buy VALUES(NULL,'20221005','운동화');
INSERT INTO buy VALUES(NULL,'20221006','노트북');
 
SELECT p.userID, p.userNM, b.prod_name 
FROM buy b INNER JOIN person p ON b.userID = p.userID;
 
UPDATE person SET userID='20221010' WHERE userID='20221001';
-- 에러가 발생하는지 확인한다.
 
##########################################################################
DROP TABLE IF EXISTS buy, person;
CREATE TABLE person (
  userID varchar(20NOT NULL COMMENT '사용자ID',
  userNM varchar(30NOT NULL COMMENT '이름',
  mobileNO varchar(60NOT NULL COMMENT '휴대폰번호',
  reg_date timestamp NOT NULL DEFAULT current_timestamp() COMMENT '등록일자',
  display tinyint(2NOT NULL DEFAULT 1,
  PRIMARY KEY (userID)
ENGINE=InnoDB DEFAULT CHARSET=utf8;
 
INSERT INTO person (userID, userNM, mobileNO) VALUES
('20221001''홍길동''010-1000-1001'),
('20221002''이순신''010-1000-1003'),
('20221003''강감찬''010-1000-1004'),
('20221004''신시아''010-1000-1005'),
('20221005''나현우''010-1000-1006'),
('20221006''홍석천''010-2000-1011'),
('20221007''김남일''010-2000-1012'),
('20221008''김남길''010-2000-1014'),
('20221009''박나래''010-2000-1015');
 
-- ALTER TABLE person ADD PRIMARY KEY (userID);
 
CREATE TABLE buy (
  id int(11NOT NULL AUTO_INCREMENT PRIMARY KEY,
  userID varchar(20NOT NULL,
  prod_name varchar(20NOT NULL,
  FOREIGN KEY (userID) REFERENCES person (userID)
)ENGINE=InnoDB DEFAULT CHARSET=utf8;
 
INSERT INTO buy VALUES(NULL,'20221001','지갑');
INSERT INTO buy VALUES(NULL,'20221001','맥북');
INSERT INTO buy VALUES(NULL,'20221003','USB');
INSERT INTO buy VALUES(NULL,'20221004','가방');
INSERT INTO buy VALUES(NULL,'20221005','휴대폰');
INSERT INTO buy VALUES(NULL,'20221005','의류');
INSERT INTO buy VALUES(NULL,'20221005','운동화');
INSERT INTO buy VALUES(NULL,'20221006','노트북');
 
SELECT p.userID, p.userNM, b.prod_name 
FROM buy b INNER JOIN person p ON b.userID = p.userID;
 
UPDATE person SET userID='20221010' WHERE userID='20221001';
-- 에러가 발생하는지 확인한다.
 
DELETE FROM person WHERE userID='20221001';
-- 에러가 발생하는지 확인한다.
 
##########################################################################
-- 기준 테이블의 열이 변경되는 경우 : 회원 테이블의 userID 변경
-- alter table [추가할테이블명] add constraint [제약조건명] foreign key(컬럼명) 
  references [부모테이블명] (PK컬럼명) [ON DELETE CASCADE / ON UPDATE CASECADE];
 
DROP TABLE IF EXISTS buy, person;
CREATE TABLE person (
  userID varchar(20NOT NULL COMMENT '사용자ID',
  userNM varchar(30NOT NULL COMMENT '이름',
  mobileNO varchar(60NOT NULL COMMENT '휴대폰번호',
  reg_date timestamp NOT NULL DEFAULT current_timestamp() COMMENT '등록일자',
  display tinyint(2NOT NULL DEFAULT 1,
  PRIMARY KEY (userID)
ENGINE=InnoDB DEFAULT CHARSET=utf8;
 
INSERT INTO person (userID, userNM, mobileNO) VALUES
('20221001''홍정민''010-1000-1001'),
('20221002''이순신''010-1000-1003'),
('20221003''강감찬''010-1000-1004'),
('20221004''신시아''010-1000-1005'),
('20221005''나현우''010-1000-1006'),
('20221006''홍석천''010-2000-1011'),
('20221007''김남일''010-2000-1012'),
('20221008''김남길''010-2000-1014'),
('20221009''박나래''010-2000-1015');
 
CREATE TABLE buy (
  id int(11NOT NULL AUTO_INCREMENT PRIMARY KEY,
  userID varchar(20NOT NULL,
  prod_name varchar(20NOT NULL
)ENGINE=InnoDB DEFAULT CHARSET=utf8;
 
ALTER TABLE buy 
  ADD constraint fk_userID 
  FOREIGN KEY(userID) REFERENCES person (userID) 
  ON UPDATE CASCADE 
  ON DELETE CASCADE;
 
INSERT INTO buy VALUES(NULL,'20221001','지갑');
INSERT INTO buy VALUES(NULL,'20221001','맥북');
INSERT INTO buy VALUES(NULL,'20221003','USB');
INSERT INTO buy VALUES(NULL,'20221004','가방');
INSERT INTO buy VALUES(NULL,'20221005','휴대폰');
INSERT INTO buy VALUES(NULL,'20221005','의류');
INSERT INTO buy VALUES(NULL,'20221005','운동화');
INSERT INTO buy VALUES(NULL,'20221006','노트북');
 
SHOW INDEX FROM buy;
 
UPDATE person SET userID='20221010' WHERE userID='20221001';
 
DELETE FROM person WHERE userID='20221003';
 
-- foreign key 확인
select * from information_schema.table_constraints where table_name = 'buy';
 
-- foreign key 삭제
ALTER TABLE buy 
  DROP foreign key fk_userID;
 
-- foreign key 삭제 확인
select * from information_schema.table_constraints where table_name = 'buy';
 
use test;
SHOW INDEX FROM buy;
 
DELETE FROM person WHERE userID='20221004';
 
###############################################################################

 

 

728x90
블로그 이미지

Link2Me

,
728x90

MySQL 샘플 데이터베이스 설치하는 방법이다.

 

https://www.mysqltutorial.org/mysql-sample-database.aspx

 

MySQL Sample Database

This page provides you with a MySQL sample database that helps you to practice with MySQL effectively and quickly. You can download the sample database and load it into your MySQL Server.

www.mysqltutorial.org

위 사이트에 접속하면 받을 수가 있는데 문제는 Import 하면 에러가 발생하더라.

전체적인 구조는 위와 같다.

 

Legacy Create 테이블 생성 방식(?)이라 그런지 몰라도 테이블 순서가 매우 중요하다.

sql 파일을 Editor로 열어보면, Foreign Key 가 설정되어 있고 employees 테이블을 참조하고 있다.

customer 테이블보다 먼저 employees 테이블이 먼저 Import 되어야 한다는 의미로 간주해야 한다.

CREATE TABLE `customers` (
  `customerNumber` int(11NOT NULL,
  `customerName` varchar(50NOT NULL,
  `contactLastName` varchar(50NOT NULL,
  `contactFirstName` varchar(50NOT NULL,
  `phone` varchar(50NOT NULL,
  `addressLine1` varchar(50NOT NULL,
  `addressLine2` varchar(50DEFAULT NULL,
  `city` varchar(50NOT NULL,
  `state` varchar(50DEFAULT NULL,
  `postalCode` varchar(15DEFAULT NULL,
  `country` varchar(50NOT NULL,
  `salesRepEmployeeNumber` int(11DEFAULT NULL,
  `creditLimit` decimal(10,2DEFAULT NULL,
  PRIMARY KEY (`customerNumber`),
  KEY `salesRepEmployeeNumber` (`salesRepEmployeeNumber`),
  CONSTRAINT `customers_ibfk_1` FOREIGN KEY (`salesRepEmployeeNumber`REFERENCES `employees` (`employeeNumber`)
ENGINE=InnoDB DEFAULT CHARSET=latin1;
 
 

 

이런 점을 고려하여 테이블 순서를 변경한 파일을 첨부한다.

mysqlsampledb.sql
0.20MB

 

위 파일을 Import 하면 정상적으로 테이블이 생성되고 데이터가 추가될 것이다.

 

DBWeaver TOOL 을 이용해서 접속해보자.

접속환경 : VirtualBox 기반 CentOS 7.9 MariaDB 10.5

DB IP address : 192.168.1.20 (사설 IP)

Access IP address : 192.168.1.25

 

-- 사용자 권한 부여
-- 비밀번호는 각자 수정하시라.
use mysql;
grant all privileges on cmodels.* to codefox@'192.168.1.25' identified by '비밀번호';
flush privileges;
 

 

 

 

 

728x90

'SQL' 카테고리의 다른 글

접속로그 통계 (신규, 중복 동시 처리)  (0) 2023.05.23
MySQL foreign key 예제  (0) 2022.05.16
MariaDB 멀티 인덱스(index) 설정  (0) 2022.04.05
MariaDB 대소문자 구분  (0) 2022.03.14
MySQL 중복 레코드 관리 방법  (0) 2022.03.02
블로그 이미지

Link2Me

,
728x90

테이블 설계 예시이다.

중복 체크를 하기 위해서 UNIQUE KEY 설정을 3개의 칼럼을 JOIN 으로 해서 하나의 KEY로 설정했다.

 

Engine 은 MyISAM 으로 한 이유는 Legacy PHP 로 코딩하는 경우, InnoDB 엔진으로 설정하면 원치 않는 에러 발생시 해결하기 난감한 경험을 겪어서이다.

CREATE TABLE voc (
  idx int(11NOT NULL,
  userID varchar(20NOT NULL COMMENT '고객ID',
  voctype varchar(100NOT NULL COMMENT 'VOC유형',
  vocCnt int(5NOT NULL DEFAULT 0 COMMENT 'VOC건수',
  YM varchar(12NOT NULL COMMENT '년월',
  CEndDate varchar(12DEFAULT NULL COMMENT '계약종료일',
  custtype varchar(20DEFAULT NULL COMMENT '고객분류',
  display int(2NOT NULL DEFAULT 1,
 reg_date timestamp NOT NULL DEFAULT current_timestamp() COMMENT '등록일자'
ENGINE=MyISAM DEFAULT CHARSET=utf8;
 
ALTER TABLE voc
  ADD PRIMARY KEY (idx),
  ADD UNIQUE KEY id_ym_type (userID,YM,voctype);
 
ALTER TABLE voc
  MODIFY idx int(11NOT NULL AUTO_INCREMENT;
COMMIT;
 

 

 

728x90

'SQL' 카테고리의 다른 글

MySQL foreign key 예제  (0) 2022.05.16
[MySQL] 샘플 데이터베이스 설치하기  (0) 2022.04.28
MariaDB 대소문자 구분  (0) 2022.03.14
MySQL 중복 레코드 관리 방법  (0) 2022.03.02
MySQL 샘플 DB 설치  (0) 2022.01.14
블로그 이미지

Link2Me

,

MariaDB 대소문자 구분

SQL 2022. 3. 14. 11:40
728x90

MariaDB 테이블명이 대소문자 구분이 되는 경우와 되지 않는 경우가 있다.

 

확인 방법

show variables like 'lower_case_table_names';

lower_case_table_names = 0

- UNIX 기반 시스템의 기본값

- 테이블이름, 별명 및 데이터베이스 이름이 대소문자를 구분하여 비교

 

 

lower_case_table_names = 1

- 테이블이름, 별명 및 데이터베이스 이름이 소문자로 저장되며 대소문자를 구분하지 않는다.

 

 

변경방법

vi /etc/my.cnf.d/server.cnf

lower_case_table_names=1

 

 

728x90
블로그 이미지

Link2Me

,
728x90

1. 개요

MySQL에는 아래 3가지 방법을 이용하여 중복 레코드를 관리할 수 있다.

  1. INSERT IGNORE ...
  2. REPLACE INTO ...
  3. INSERT INTO ... ON DUPLICATE UPDATE

각 방법의 특징을 요약하면 다음과 같다.

INSERT IGNORE ... 최초 등록된 레코드가 남아 있음
최초 등록된 레코드의 AUTO_INCREMENT 값은 변하지 않음
REPLACE INTO ... 최초 등록된 레코드가 삭제되고, 신규 레코드가 INSERT됨
AUTO_INCREMENT의 값이 변경됨
INSERT INTO ... ON DUPLICATE UPDATE INSERT IGNORE의 장점 포함함
중복 키 오류 발생 시, 사용자가 UPDATE될 값을 지정할 수 있음

방법 특징

2. 사전 조건 및 중복 처리 방법

중복 레코드 관리를 위해선 테이블에 UNIQUE INDEX가 필요하다.

자동으로 증가하는 Primary Key 로 중복관리를 하는 것은 쉽지 않다.

중복 레코드 관리를 위한 KEY가 2개의 칼럼 조합일 수도 있다.

 

create database androidsample default character set utf8 COLLATE utf8_general_ci;
 
CREATE TABLE IF NOT EXISTS person (
  id int(11NOT NULL,
  name varchar(20DEFAULT NULL,
  address varchar(150DEFAULT NULL,
  cnt int(4NOT NULL DEFAULT 0,
  PRIMARY KEY (id),
  UNIQUE INDEX (name) -- 중복 검사용 필드
ENGINE=InnoDB DEFAULT CHARSET=utf8mb3;
 
-- INSERT IGNORE INTO
-- INSERT IGNORE는 중복 키 에러가 발생했을 때 신규로 입력되는 레코드를 무시
INSERT IGNORE INTO person VALUES (NULL'홍길동''서울',0);
INSERT IGNORE INTO person VALUES (NULL'신시아''용인',0);
INSERT IGNORE INTO person VALUES (NULL'홍길동''평택',1);
INSERT IGNORE INTO person VALUES (NULL'신시아''서울',1);
INSERT IGNORE INTO person VALUES (NULL'홍길동''인천',2);
 
 
-- INSERT INTO ON DUPLICATE KEY UPDATE
INSERT INTO person VALUES (NULL'강감찬''대전'3)
           ON DUPLICATE KEY UPDATE address = VALUES(address), cnt=VALUES(cnt);
 
INSERT INTO person VALUES (NULL'홍길동''인천'5)
           ON DUPLICATE KEY UPDATE address = VALUES(address), cnt=VALUES(cnt);
 
INSERT INTO person(name,cnt) VALUES ('신시아'8)
           ON DUPLICATE KEY UPDATE cnt=VALUES(cnt);
 
INSERT INTO person VALUES (NULL'이순신''충청'9)
           ON DUPLICATE KEY UPDATE address = VALUES(address), cnt=VALUES(cnt);
 
-- 테이블 비우기
TRUNCATE person;
 
INSERT IGNORE INTO person VALUES (NULL'홍길동''서울',0);
INSERT IGNORE INTO person VALUES (NULL'신시아''용인',0);
INSERT IGNORE INTO person VALUES (NULL'홍길동''평택',1);
INSERT IGNORE INTO person VALUES (NULL'신시아''서울',1);
INSERT IGNORE INTO person VALUES (NULL'홍길동''인천',2);
 
-- REPLACE INTO
-- REPLACE INTO는 중복이 발생되었을 때 기존 레코드를 삭제하고 신규 레코드를 INSERT하는 방식
REPLACE INTO person VALUES (NULL'홍길동''서울',0);
REPLACE INTO person VALUES (NULL'신시아''용인',0);
REPLACE INTO person VALUES (NULL'홍길동''평택',1);
REPLACE INTO person VALUES (NULL'신시아''서울',1);
REPLACE INTO person VALUES (NULL'홍길동''인천',2);
 

 

UPDATE하고자 할 때는 항상 column=VALUES(column)와 같이 적어줘야 한다는 점이다.

 

두가지 케이스로 테스트를 해보면서, id 값이 어떻게 변경되는지 확인해 보시라.

 

Primary Key 값의 변동없이 Insert INTO, Update를 처리하려면 PHP 등과 같은 별도의 백엔드 언어를 통해서 처리하는 코드를 구현해야 한다.

728x90

'SQL' 카테고리의 다른 글

MariaDB 멀티 인덱스(index) 설정  (0) 2022.04.05
MariaDB 대소문자 구분  (0) 2022.03.14
MySQL 샘플 DB 설치  (0) 2022.01.14
DB 접속툴 DBeaver Community 설치 및 DB 접속  (0) 2021.12.03
회원 테이블(members SQL) 예시  (0) 2021.11.09
블로그 이미지

Link2Me

,

MySQL 샘플 DB 설치

SQL 2022. 1. 14. 13:58
728x90

MySQL 샘플 DB가 필요해서 설치하는 과정을 적어둔다.

 

https://github.com/datacharmer/test_db 에 접속해서 파일을 다운로드한다.

test_db-master.zip 라는 파일이 다운로드 된다.

 

이제 PC에 설치된 MariaDB 에 접속해야 한다.

CMD 창을 관리자권한으로 열어야만 접속이 되더라.

 

Windows키 + R 을 누르면

 

 

cmd 를 입력하고 Ctrl + Shift + Enter키를 누르면 관리자권한으로 접속된 CMD창이 뜬다.

 

이제 MariaDB가 설치된 폴더로 이동한다.

그리고 다운로드 받은 파일을 해당 폴더로 옮겼다.

mysql -u root -p

패스워드 입력

을 하면 DB에 접속된다.

use mysql;

source employees.sql;

를 하면 DB와 테이블이 생성되고 샘플 데이터가 업로드된다.

 

 

테이블이 생성된 걸 확인할 수 있다.

 

 

 

이제 DBeaver 접속툴로 접속해 본 이미지를 확인해보자.

 

데이터 확인

 

쿼리문 연습은 phpMyAdmin 에서 해도 된다.

728x90

'SQL' 카테고리의 다른 글

MariaDB 대소문자 구분  (0) 2022.03.14
MySQL 중복 레코드 관리 방법  (0) 2022.03.02
DB 접속툴 DBeaver Community 설치 및 DB 접속  (0) 2021.12.03
회원 테이블(members SQL) 예시  (0) 2021.11.09
SELECT Query 예제  (0) 2021.09.22
블로그 이미지

Link2Me

,
728x90

DB 접속은 보통 phpMyAdmin 으로 접속하기 때문에 사용할 일이 없었다.

꼭 DB 접속툴을 이용하여 접속해야 하는 환경에서 HeidiSQL 접속을 시도했는데 접속이 어렵다.

그래서 DB툴을 검색하니 DBeaver 가 있어서 설치하고 접속을 해봤다.

HeidiSQL은 테스트 환경이 IP TIME 공유기 환경에서 해봐야 편의성 여부를 알 수 있을 듯 싶다.

 

보통은 DB 접속을 원격에서 하면 DB에서 원격 접속을 허용해야 하는데 DBeaver는 localhost 접속으로 가능하게 만들었더라. SSH 접속을 통해서 로컬상에서 DB에 접속하는 방법으로 접속을 하는 거 같다.

 

https://dbeaver.io/ 에서 파일 다운로드 받아서 설치하면 아래와 같은 단계를 거쳐서 DB접속이 가능하다.

 

 

 

 

 

DB가 접속된 걸 확인할 수 있다.

 

설치한 pythondb의 ER 다이어그램을 실행시켜 봤다.

DB 테이블 구조는 https://link2me.tistory.com/2072 에 올려진 테이블 구조를 샘플로 등록했다.

아직 데이터는 하나도 생성하지 않아서 데이터는 없다.

 

윈도우 뿐만 아니라 MAC에서도 이용할 수 있어 편리한 DB 툴인 거 같다.

 

ER Diagram 기능은 https://dbeaver.com/docs/wiki/ER-Diagrams/ 를 참조하면 될 듯 싶다.

728x90

'SQL' 카테고리의 다른 글

MySQL 중복 레코드 관리 방법  (0) 2022.03.02
MySQL 샘플 DB 설치  (0) 2022.01.14
회원 테이블(members SQL) 예시  (0) 2021.11.09
SELECT Query 예제  (0) 2021.09.22
MySQL 테이블 스토리지 엔진 변경  (0) 2021.04.01
블로그 이미지

Link2Me

,
728x90

회원 테이블 예시이다.

비밀번호는 RSA 암호화 비밀번호를 사용하고, SALT 20자를 적용하여 보안을 고려한다.

access 칼럼은 관리자가 userID를 접속을 불허할 것인지, 비밀번호를 5회 이상 틀릴 경우 접속이 차단되도록 설정했다가 일정 시간이 경과하면 자동으로 접속 허용 상태로 변경처리할 것인지 정하는 용도이다.

가입일자와 최근 접속일자를 구분하여 최근에 접속한 날짜를 기록한다.

로그인 실패횟수를 카운트하여 접속 실패횟수 초과 설정값을 넘어가면 접속을 차단하는 로직을 구현하면 된다.

CREATE TABLE `members` (
  `idx` int(11NOT NULL,
  `userID` varchar(20NOT NULL COMMENT '아이디',
  `userNM` varchar(50NOT NULL COMMENT '성명',
  `passwd` varchar(90NOT NULL COMMENT '비밀번호',
  `salt` varchar(20NOT NULL COMMENT '비밀번호 솔트',
  `telNO` varchar(16DEFAULT NULL COMMENT '사무실전화',
  `mobileNO` varchar(60DEFAULT NULL COMMENT '휴대폰번호',
  `email` varchar(56DEFAULT NULL COMMENT '이메일',
  `access` tinyint(2NOT NULL DEFAULT 0 COMMENT '접근권한:불허(1),승인대기(2),차단(3)',
  `regdate` varchar(12DEFAULT NULL COMMENT '가입일자',
  `regtime` varchar(10DEFAULT NULL COMMENT '가입시간',
  `access_date` date DEFAULT NULL COMMENT '최근접속일자',
  `access_failed_count` tinyint(4NOT NULL DEFAULT 0 COMMENT '로그인실패횟수',
  `display` tinyint(2NOT NULL DEFAULT 1 COMMENT '숨김(0) 보임(1)'
ENGINE=InnoDB DEFAULT CHARSET=utf8;
 
--
-- Indexes for table `members`
--
ALTER TABLE `members`
  ADD PRIMARY KEY (`idx`),
  ADD UNIQUE KEY `userID` (`userID`);
 
--
-- AUTO_INCREMENT for table `members`
--
ALTER TABLE `members`
  MODIFY `idx` int(11NOT NULL AUTO_INCREMENT;
COMMIT;

 

Hash 와 SALT 개념은 https://st-lab.tistory.com/100 에 잘 설명되어 있더라.

코드가 Java 코드 예시이긴 해도 개념 이해하는데 이보다 훌륭한 자료 찾기는 쉽지 않을 듯 싶다.

728x90
블로그 이미지

Link2Me

,

SELECT Query 예제

SQL 2021. 9. 22. 19:21
728x90

(예제) 테이블에서 장학금을 받는 학생의 학번과 장학금 내역을 출력하라.

SELECT studentNO, scholarship

FROM tablename

WHERE scholarship > 0;

 

예제) 장학금을 백만원 이상 지급 받는 학생 중에서 2회 이상 지급받은 학생의 학번과 지급 받는 횟수를 학번 내림 차순으로 출력하라. ( 실행 순서 : FROM 절 → WHERE 절 → GROUP BY 절 → HAVING 절 → SELECT 절 → ORDER BY 절)

SELECT studentNO, count(*)

FROM tablename

WHERE scholarship > 1000000

GROUP BY studentNO

having count(*) > 1

ORDER BY studentNO DESC;

 

※ GROUP BY 절을 사용할 때에는 count, avg, min, max, sum 과 같은 그룹 함수를 같이 사용해야 한다.

※ WHERE 절은 FROM 절에서 생성된 중간 테이블에서 동작하고,

   HAVING 절은 GROUP BY 절에서 생성된 중간 테이블에서 동작한다.

   SELECT 절은 최종 결과 테이블에 표현될 열을 지정하기 위해서 사용된다.

   HAVING 절의 조건에 있는 수식은 통계 함수를 가질 수 있지만

   WHERE 절에 있는 조건 수식에는 통계함수를 가질 수 없다.

만약 결과 테이블에 서로 다른 테이블로부터 데이터를 가져오려 한다면

   FROM 절에 여러 개의 테이블을 지정해야 한다.

FROM 절에 동일한 테이블 이름이 한번 이상 사용될 때는 반드시 가명을 사용해야 한다.

 

예제) 1990년 이후에 출생한 여학생의 학번, 이름, 주민등록번호를 출력하라

※ substring(id_num,1,2) 라고 하면 앞 2자리를 반환한다. 1부터 시작된다.

SELECT studentNO, userNM, id_num

FROM tablename

WHERE substring(id_num, 8,1) = 2 and birth_year > '1990';

 

예제) 성별이 남자가 아닌 학생의 학번, 이름을 출력하라.

SELECT studentNO, userNM

FROM tablename

WHERE NOT substring(id_num, 8, 1) = 1;

 

예제) 나이가 가장 많은 학생을 제외한 나머지 학생의 학번, 이름, 주민등록번호를 출력하다.

SELECT studentNO, userNM, id_num

FROM tablename

WHERE birth_year > any (select birth_year from tablename);

 

예제) 주민등록번호의 앞 2자리의 값이 서로 다른 숫자를 가지는 것은 몇명인지 출력하라.

SELECT count(distinct substring(id_num,1,2)

FROM tablename;

 

예제) 학년별, 주야 인원을 출력하라. 순서는 학년별 오름차순, 주야 오름 차순이다.

※ 2개 이상의 열에 대한 그룹화

   count(*) 함수는 서로 다른 모든 행에 대해 연산하지 않고, 각 그룹화된 행에 의존하여 실행된다.

SELECT grade, daynight, count(*)

FROM tablename

GROUP BY grade, daynight

ORDER BY grade, daynight;

 

예제) 재학중에 납부한 등록금의 전체 납부 금액이 30,000,000 원 이상인 각 학생에 대하여 출력하라.

SELECT studentNO, sum(fee)

FROM tablename

GROUP BY studentNO

having sum(fee) >= 30,000,000;

 

테이블에서 테이블로 행을 복사

- INSERT INTO 절에 있는 열의 수는 SELECT 절에 있는 수식의 수와 동일해야 한다.

- INSERT INTO 절에 있는 자료형은 SELECT 절에 있는 수식의 자료형과 일치해야 한다.

 

728x90
블로그 이미지

Link2Me

,
728x90

ALTER TABLE `테이블명` ENGINE = MYISAM;

ALTER TABLE `테이블명` ENGINE = INNODB;

 

DB가 이상한 것인지는 정확하게 모르겠으나 Lagacy PHP 에서 INNODB 엔진으로 설정한 테이블에 데이터가 저장되지 않는 거 같더라.

localhost에서는 잘 저장되는데, 원격 서버에서 DB 접속할 때 auto commit 이 안되어서 그런지 안되더라.

이것 때문에 원인 찾느라 엄청 삽질을 했다.

 

PDO 방식의 코드는 기본적으로 INNODB 를 고려한 데이터 Insert, Update가 처리되도록 코드가 구현되어 있는데

Lagacy PHP 에서는 commit 이 자동으로 안되는지 모르겠다.

직접 DB에 접속하여 쿼리를 실행하면 잘 저장되는 걸 확인했기 때문에 원인이 뭘까 이 고민 저고민 하다가

혹시 엔진 때문인가 싶어서 MyISAM 으로 엔진 변경하고 테스트했더니 잘 저장되네.

728x90

'SQL' 카테고리의 다른 글

회원 테이블(members SQL) 예시  (0) 2021.11.09
SELECT Query 예제  (0) 2021.09.22
MySQL SUBTIME 예제 (30분전까지 데이터 불러오는 쿼리)  (0) 2021.03.24
MySQL DB 백업  (0) 2020.06.13
SQL 테이블 칼럼 추가/삭제 방법  (0) 2020.03.14
블로그 이미지

Link2Me

,
728x90

SUBTIME(start_value, time) 은 start_value 에서 time만큼을 차감한 결과를 돌려준다.

SELECT SUBTIME('2015-08-01 01:30:10','02:50:20') AS subtime
// subtime -> 2015-07-31 22:39:50

SELECT SUBTIME('2015-08-01 01:30:10','2015-08-01 02:50:20') AS subtime
// subtime -> NULL

SELECT SUBTIME('2015-08-01 01:30:10.126587','00:50:20') AS subtime
// subtime -> 2015-08-01 00:39:50.126587

SELECT SUBTIME('2015-08-01 01:30:10.126587','00:50:20.020202') AS subtime
// subtime -> 2015-08-01 00:39:50.106385

SELECT SUBTIME('01:30:10.126587','02:50:20.020202') AS subtime
// subtime -> -01:20:09.893615

SELECT SUBTIME('01:30:10.126587','-02:50:20.020202') AS subtime
// subtime -> 04:20:30.146789

SELECT SUBTIME("10:24:21", "5");

결과는 10:24:16 ("5"는 5초를 의미)

 

CURDATE() 현재 연-월-일, CURTIME() 현재 시:분:초,

NOW()와 SYSDATE() 현재 연-월-일 시:분:초 출력

현재시간으로 부터 30분까지의 결과를 반환하고자 한다면....

where regdate > subtime(current_timestamp(), '0:30:0') and regdate < current_timestamp()

 

다른 방법으로

where regdate >= date_add(now(), interval -30 minutes)

 

where regdate >= date_add(now(), interval -1 day)  // 1일전

728x90

'SQL' 카테고리의 다른 글

SELECT Query 예제  (0) 2021.09.22
MySQL 테이블 스토리지 엔진 변경  (0) 2021.04.01
MySQL DB 백업  (0) 2020.06.13
SQL 테이블 칼럼 추가/삭제 방법  (0) 2020.03.14
테이블명 변경  (0) 2019.03.11
블로그 이미지

Link2Me

,

MySQL DB 백업

SQL 2020. 6. 13. 00:00
728x90

MySQL/MariaDB 백업, 복구에 대한 명령어를 적어둔다.


## 모든 DB 백업
mysqldump -uroot -p --all-databases > alldbbackup.sql

## 모든 DB 복구
mysql -u root -p < alldbbackup.sql


## 특정 DB 백업

mysqldump -uroot -p --databases kchart > kchart.sql


# kchart DB에서 특정 테이블 여러개 백업
mysqldump -uroot -p kchart allowHosts > allowHosts.sql
mysqldump -uroot -p kchart allowHosts > allowHosts.sql
mysqldump -uroot -p kchart allowHosts SYS_MEMBER > sys_allow_backup.sql

## DB 접속

mysql -u root -p


// DB 생성
create database kchart default character set utf8;



## root 비밀번호 설정
use mysql;
update user set password=password('설정할root비번') where user='root';
flush privileges;

## DB 사용권한 등록

## username, password, dbname을 각각 상황에 맞게 수정한다.

use mysql;
create user username@localhost identified by 'password';
grant all privileges on dbname.* to username@localhost;
flush privileges;


### 사용자 현황 검색

use mysql;
select host, user, password from user;



## DB 백업본 업로드
mysql -u root -p
비밀번호 입력
use kchart;
source partmember.sql;



728x90
블로그 이미지

Link2Me

,
728x90

테이블 칼럼을 phpMyAdmin 을 이용하여 추가하다보니 명령어를 잘 모르겠다.

그래서 명령어를 추가해 둔다.


칼럼 추가

ALTER TABLE 테이블명
ADD 추가할 컬럼명  데이터 유형;


ALTER TABLE members ADD  isUpdate TINYINT( 2 ) NOT NULL ,

ADD isDelete TINYINT( 2 ) NOT NULL ,
ADD OStype TINYINT( 2 ) NOT NULL ,
ADD phoneBrand VARCHAR( 20 ) NULL ,
ADD phoneModel VARCHAR( 20 ) NULL ,
ADD level TINYINT( 2 ) NOT NULL DEFAULT '1',
ADD chosung VARCHAR( 10 ) NULL ,
ADD ismobileNO TINYINT( 2 ) NOT NULL DEFAULT '1'


칼럼 변경

ALTER TABLE members

CHANGE isDelete isDelete TINYINT( 2 ) NOT NULL DEFAULT '0'


칼럼 삭제

ALTER TABLE 테이블명
DROP 삭제할 컬럼명;

ALTER TABLE members
  DROP isUpdate,
  DROP isDelete;


특정 칼럼 다음에 필드 추가

ALTER TABLE members ADD isUpdate TINYINT( 2 ) NOT NULL DEFAULT '0' AFTER isGanbu ,
ADD isDelete TINYINT( 2 ) NOT NULL DEFAULT '0' AFTER isUpdate


칼럼 순서 변경

ALTER TABLE members modify column corpID VARCHAR( 12 ) AFTER idx;


728x90

'SQL' 카테고리의 다른 글

MySQL SUBTIME 예제 (30분전까지 데이터 불러오는 쿼리)  (0) 2021.03.24
MySQL DB 백업  (0) 2020.06.13
테이블명 변경  (0) 2019.03.11
MySQL 사용자 권한 부여  (0) 2018.11.26
MySQL 두 테이블 불일치 데이터 찾는 SQL  (0) 2018.08.18
블로그 이미지

Link2Me

,