H'academy
  • H'academy
  • Web
    • Back-end
      • Spring Framework
        • Spring 개발환경 구축
          • Project 생성
          • Maven 설정
          • Maven 저장소 위치 설정
          • Apache Tomcat 연동
          • Lombok 설정
        • Spring MVC Project
          • pom.xml
          • web.xml
          • root-context.xml
          • servlet-context.xml
          • DispatcherServlet
        • Controller
          • File Upload / Download
          • RestController(작성중)
        • Spring Test
          • Assert 단정문
          • Spring 환경 연동 테스트
          • MockMvc Test
        • Spring AOP
          • AOP 용어
          • Pointcut Expression
          • Filter
          • Interceptor
          • Filter vs Interceptor
          • Controller Advice(작성중)
        • Spring JDBC
          • DBCP
          • myBatis
            • ResultMap 활용
        • ETC
          • Logging
            • Tomcat Logging 설정
          • Property 관리
          • E-mail
            • Gmail 변경사항
          • Scheduler
          • Websocket
      • Spring Boot
        • STS 다운로드
        • 프로젝트 생성
        • 프로젝트 구조
        • Spring Devtools
        • Spring Controller
          • JSP 연동
            • * JSP 생성불가 문제
            • Spring Boot 3.x
          • Model
          • 요청 파라미터 처리
          • 경로 변수 처리
          • 요청 메소드
          • Redirect
          • Spring Actuator
        • Spring Boot JDBC
          • Database 정보
          • DAO Pattern
          • ORM Framework
            • myBatis
              • Mapper 생성
            • Spring Data JPA
        • Spring AOP
          • AspectJ
          • Filter
          • Interceptor
          • ControllerAdvice
        • REST API
        • Spring WebSocket
        • Send Mail
        • Logging
        • Spring Boot Test
        • Lombok
      • Java EE
        • 개발환경 구성
          • Server와 Client
          • Web Server와 WAS
        • Dynamic Web Project
      • Apache Tomcat
        • Apache Tomcat 구조
        • Tomcat User 생성
        • Tomcat Manager 설정
      • Maven
      • Jenkins
        • Installation
          • Github Webhook 설정
          • Maven 연동
        • Project setting
          • 소스 코드 관리 설정
          • 빌드 유발 설정
          • 빌드 작업 설정
      • OpenAPI(swagger)
        • swagger 2.x (legacy)
        • swagger 3.x (boot)
        • Swagger UI 꾸미기
        • SpringDoc
    • Front-end
      • HTML
        • Heading
        • <A>
        • <IMG>
        • <DIV>
        • <TABLE>
        • <FORM>
          • <INPUT>
          • <SELECT>
          • <TEXTAREA>
          • <FIELDSET>
        • List
      • CSS
        • CSS 적용 방식
          • 크기(size)
            • 픽셀(px)
          • 색상(color)
        • CSS 선택자
          • 스타일 우선순위
        • CSS 주요 속성
          • 크기 속성
            • border
            • margin
            • padding
            • box-sizing
          • 배경 속성
          • 글자 속성
            • font
            • text
          • 배치 속성
            • position
            • display
            • overflow
            • float
            • flexbox
        • CSS Reset
      • JS
        • syntax
          • let, const
          • 구조 분해 할당
          • spread operator
          • Object copy
          • this
          • module
        • jQuery
          • structure
          • function
          • ajax
            • ajax file upload
        • VueJS
          • CDN(Vue3)
            • 기본 구조
            • Vue Devtools
            • jQuery vs VueJS
            • Vue instance
              • data
                • IME issue
              • v-model
              • template
              • computed
              • watch
                • deep watch
                • vs Computed
              • methods
            • Vue directive
              • v-text
              • v-html
              • v-bind
                • v-bind style
                • v-bind class
              • v-on
                • once
                • prevent
              • v-show
              • v-if
              • v-for
          • Vue-cli 3
            • 프로젝트 구조
            • src
              • main.js
              • App.vue
              • HelloWorld.vue
            • Vue Router 4
              • History mode
              • 404 not found
              • Guard
                • beforeEach
            • Vuex
              • 이동 횟수 측정
            • axios
              • interceptor
          • Vue-electron
        • ReactJS
          • CDN(React18)
            • 템플릿 설명
            • JSX
            • render
            • Handling Events
              • onClick
              • onInput
            • Component
              • 클래스 컴포넌트
                • state
                • props
                • event
                  • Mouse
                  • Input
                • lifecycle
              • 함수형 컴포넌트
                • useState
                • useEffect
                • useReducer
                • useMemo
                • props
                • 조건부 렌더링
          • CRA
            • 자원 경로 설정
            • src 절대 경로 설정
            • react-router
            • Context API
            • Electron 설정
        • Webpack
          • NodeJS
          • Git
    • Ubuntu server
      • Jenkins
    • 테스트 도구
      • JMeter
        • Test Plan
        • Test Plan(with login)
        • Plugin 설치
    • WebRTC
  • Base Language
    • Java
      • Java 실행 과정
        • 개발 환경 구축
        • 샘플 코드 작성
        • 컴파일/실행
        • 환경 변수 설정
      • Java 초급
        • Hello world
        • 자료와 정보
        • 변수
          • 정수
            • 정수의 저장 원리
            • 정수형 변수 생성
          • 실수
            • 실수의 저장 원리
            • 실수형 변수 생성
          • 논리
            • 논리형 변수 생성
          • 문자
          • 문자열
            • 문자열 변수 생성
          • 사용자 정의 자료형
        • 연산자
          • 대입 연산
          • 부호 연산
          • 산술 연산
          • 복합 대입 연산
          • 증감 연산
          • 쉬프트 연산
          • 비트 연산
          • 비교 연산
          • 논리 연산
          • 삼항 연산
          • 변환 연산
        • 표준 입출력
          • 표준 출력
          • 표준 오류 출력
          • 표준 입력
            • Scanner 입력
        • 제어문
          • if 구문
            • 단일 조건
            • 두 가지의 조건
            • 여러 가지의 조건
          • switch~case 구문
          • while 구문
          • do~while 구문
          • for 구문
        • 랜덤(Random)
        • 배열
          • 1차원 배열
          • 2차원 배열
          • 다차원 배열
          • Sort
            • 선택 정렬
            • 버블 정렬
            • 삽입 정렬
      • Java 중급
        • 객체 지향 프로그래밍
        • 클래스
        • 인스턴스
        • 필드
        • 메소드
          • 설정(setter) 메소드
          • 반환(getter) 메소드
        • 생성자
        • 접근 제한
          • private
          • package
          • protected
          • public
        • 정적(static)
        • 불변(final)
        • 상수(constant)
        • 상속
          • this와 super의 이해
          • 멤버 변수 설정
          • 멤버 메소드 설정
          • 생성자 설정
        • 추상화
          • 추상 클래스
          • 인터페이스
        • 다형성
        • 중첩 클래스
          • 일반 중첩 클래스
          • static 중첩 클래스
          • 지역 중첩 클래스
          • 익명 중첩 클래스
      • Java API(작성중)
        • java.lang
        • java.util
        • java.text
        • java.time
        • java.io
        • java.net
        • java.sql
        • java.awt
        • javax.swing
      • Java 고급
        • 제네릭(Generic)
        • 열거형(Enum)
        • Annotation Type
          • Retention
          • Target
          • Nested Annotation
        • Java Reflection
          • Intro
          • Class
          • Field
          • Constructor
          • Method
          • Package
      • JDBC
        • 드라이버 로드
        • 연결 생성
          • JDBC Driver 유형
        • JdbcTemplate
      • Java 버전별 변경사항
        • Java 1.8
          • Lambda Expression
          • Stream API
          • java.time 패키지
        • Java 9
        • Java 12
          • switch~case
        • Java 13
          • Text Block
    • Arduino
      • 학습 준비
        • Reference
        • 아두이노 온라인 시뮬레이터
        • 전기의 이해
        • 그라운드(GND)의 이해
      • 개발 환경 구축
        • CH340 보드 설치
        • Editor 글꼴 설정
      • 아두이노 제어
        • 코드의 구성
        • 시리얼 출력
        • 디지털 출력
          • LED 출력
        • PWM 출력
        • 시리얼 입력
        • 디지털 입력
          • PIR 센서
        • 아날로그 입력
          • 온도 센서(LM35DZ)
          • 조도 센서
          • 가스 센서
        • 피에조 부저
        • 서보 모터 제어
          • 시리얼 서보 모터 제어
          • 스위치 서보 모터 제어
        • 온습도 센서(DHT-11)
        • LCD
      • 아두이노 통신
        • 블루투스(Bluetooth)
          • HC-06
            • 온도 센서 활용
          • HM-10
        • 와이파이(Wi-Fi)
          • ESP-01
            • WiFiEsp 라이브러리 추가
            • WiFiEsp 네트워크 스캔
        • 이더넷(Ethernet)
        • ArduinoJson 라이브리 사용
    • Regular Expression
  • Database
    • Oracle
      • 설치
        • SQL Plus
        • SQL Developer
        • DBeaver
        • Docker(Mac OS)
      • SQL
        • 관리자 명령
          • 사용자 관리
          • 권한 관리
          • Encoding 설정
          • 서비스 포트 설정
        • 사용자 명령
          • 테이블 관리
            • 테이블 항목 제약 조건
            • 외래키 제약조건
            • 데이터 유형
          • 시퀀스 관리
          • 데이터 관리
            • 데이터 추가
            • 데이터 조회
              • 오라클 내장 함수
              • 그룹 조회
              • Top N Query
            • 데이터 수정
            • 데이터 삭제
          • 뷰 관리
          • 인덱스 관리
          • 집합 연산
          • 테이블 조인
            • 등가 조인
            • 외부 조인
        • HR 계정
        • 기타 명령
          • 컬럼 간격 조정
      • 내보내기 및 불러오기
        • EXPDP, IMPDP
        • Docker로 실행중인 경우
      • 기타
        • 자동 세션 제거
        • Tablespace 관리
        • Unlock user
  • 면접대비
    • Q&A
      • Java
      • Database
      • Web
      • IT 일반상식
  • ETC
    • Eclipse IDE
      • 설치
      • 화면 설명
      • 환경 설정
      • 프로젝트 생성
      • 소스파일 작성 및 실행
    • Github
      • Github 계정 관리
      • Github 저장소 관리
        • Profile
        • Collaborators
        • Code
        • Issues
        • Pull requests
        • Action
          • Vue 3 Deploy Action
      • Personal Access Token
      • Git ignore 설정
        • Spring 개발용 샘플
      • Github Eclipse 연동
        • Branches
        • Project
        • Team project
          • Dynamic web project
          • Spring Boot Project
          • Branch 생성
          • Pull Request
          • Branch 보호
          • Branch 다시 생성
    • Figma
      • Team
      • Design
      • Component
    • 과정평가형
      • 실기 모의고사 1회
      • 실기 모의고사 2회
      • 실기 모의고사 3회
      • 실기 모의고사 4회
    • VScode
      • Live Server 실행 오류
      • Github 계정 변경
    • Naver ToastUI Editor
      • Editor 만들기
        • Editor options
          • File upload
      • Viewer 만들기
    • 문자 인코딩 방식
    • OBS
      • 녹화 잡음 제거 설정
    • Semantic Version
Powered by GitBook
On this page
  • Top N Query
  • Top N Query의 필요성
  • rownum
  • rownum의 문제
  • 서브쿼리(sub query)를 이용한 해결
  • 계속되는 문제
  • 최종 문제 해결
  • 결론
  1. Database
  2. Oracle
  3. SQL
  4. 사용자 명령
  5. 데이터 관리
  6. 데이터 조회

Top N Query

Previous그룹 조회Next데이터 수정

Last updated 3 years ago

Top N Query

이 문서에서는 Top N Query의 필요성과 이용 방법에 대해서 살펴본다.

Top N Query의 필요성

Top N Query는 데이터 집합에서 원하는 행 구간을 조회하도록 하는 구문이다. 예를 들어 다음과 같은 사례에 적용할 수 있다.

  • 음원차트 Top 100

  • 인기순위 Top 10

  • 판매량 Top 3

  • 31번부터 40번까지 게시글

rownum

Oracle DBMS 에서는 select를 통해 나온 결과집합에 rownum이라는 속성을 추가하여 행 번호를 부여한다. 이 rownum을 이용하여 원하는 데이터 구간을 조회할 수 있는 구문을 작성한다. 사용할 데이터 테이블은 을 사용한다.

select rownum, product.* from product;

조회하면 다음과 같은 결과집합이 출력된다.

조건을 부여해도 rownum은 순차적으로 부여된다.

select * from product where price <= 2000;

결과집합은 다음과 같이 출력되며, rownum은 결과집합에 순서대로 부여되는 것을 확인할 수 있다.

따라서 다음과 같은 조건의 사용도 가능하다.

select rownum, product.* from product where rownum <= 3;

rownum의 문제

우선 다음 구문을 실행한 결과를 살펴본 뒤 문제점에 대해서 알아보도록 한다.

select rownum, product.* from product order by price desc;

가격이 비싼 순서대로 정렬하는 구문이며, 다음과 같은 출력 결과가 나온다.

rownum이 순서대로 출력되지 않는데 이는 rownum의 문제보다는 select의 실행되는 순서 때문에 발생하는 문제라고 보는 것이 맞다.

select는 다음과 같은 순서로 해석된다.

select  		---(1)
rownum, product.* 	---(3)
from product		---(2)
order by price desc;	---(4)

여기서 중요한 부분은 정렬이 가장 마지막에 해석된다는 점인데, rownum이 부여되고 나서 정렬이 이루어지기 때문에 우리가 원하는 방식으로 rownum이 출력되지 않는 현상이 발생한다.

서브쿼리(sub query)를 이용한 해결

순서가 문제라면 서브쿼리(sub query)를 이용하여 순서를 변경해줄 수 있다. 현재 rownum이 순서대로 부여되지 않는 이유는 rownum을 부여한 이후 정렬이 이루어지기 때문이므로, 정렬을 한 이후 rownum을 부여할 수 있도록 정렬 코드를 서브쿼리로 실행한다.

select rownum, TMP.* from (
	select * from product order by price desc
) TMP;

서브쿼리의 위치를 보면 조회할 대상을 의미하는 from 뒤에 위치하는데, 이는 먼저 정렬을 한 결과를 TMP라고 부르고, product 테이블이 아닌 TMP에서 rownum을 붙여 재 조회 하겠다는 의미로 해석할 수 있다. 즉, 서브쿼리를 활용하여 정렬을 먼저 수행하도록 변경하였고, 출력 결과를 보면 문제가 해결되었음을 확인할 수 있다.

이를 이용하여 Top 3를 계산할 수 있다.

select rownum, TMP.* from (
    select * from product order by price desc
)TMP where rownum <= 3;

TMP는 별칭이며, 다른 별칭을 사용해도 무방하다.

계속되는 문제

아직 해결되지 않은 문제가 남아있다. Top 3, Top 5등 rownum이 1부터 조회되는 경우는 문제 없이 결과집합이 출력되지만, rownum이 1부터 조회되는 경우가 아니면 정상적으로 조회가 이루어지지 않는다.

select rownum, TMP.* from (
    select * from product order by price desc
)TMP where rownum between 3 and 5;

위 구문은 데이터가 있음에도 조회가 이루어지지 않는데, 이는 rownum의 특징과 연관이 있다.

rownum은 select 구문이 실행되어 결과집합이 만들어지면서 결과집합의 행에 부여되는 번호이다. 그리고 rownum은 1번부터 생성된다. 이는 다르게 말하면 1번이 생성되지 않으면 rownum은 생성되지 않는다는 것을 의미한다.

최종 문제 해결

rownum을 부여하면서 rownum을 조건으로 사용하면 1번부터 시작하는 경우가 아니면 조건이 성립하지 않는 문제를 해결하기 위해서 다시 한 번 서브쿼리를 사용한다. 서브쿼리를 사용하여 rownum을 부여한 뒤 조건으로 필터링 하도록 순서를 변경한다.

select * from (    
    select rownum rn, TMP.* from (
        select * from product order by price desc
    )TMP
) where rn between 3 and 5;

rownum 뒤에 rn이라는 별칭을 붙여 모호성을 방지하도록 설정한다. 별칭을 부여하지 않을 경우 세 번의 select가 각각의 rownum을 가지기 때문에 잘못된 rownum이 선택되어 결과가 나오지 않는다.

결론

결론적으로 Top N Query는 다음과 같은 공식을 사용한다.

select * from (
    select rownum rn, TMP.* from(
        -- 원하는 데이터를 조회하는 구문
    ) TMP
) where rn between 시작번호 and 종료번호;
상품 테이블(product)
rownum은 1번부터 부여된다.