레이블이 oraclejava교육인 게시물을 표시합니다. 모든 게시물 표시
레이블이 oraclejava교육인 게시물을 표시합니다. 모든 게시물 표시

2013년 8월 8일 목요일

[오라클자바커뮤니티]에서 파라미터 넘기는 법

보통 <global-forwards>에서는 다음과 같은 방법을 매핑될 페이지를
입력합니다.

<forward name="insert_success"        path="BoardList.do" />

여기서 파라미터를 다음과 같이 입력해서 코딩하면 xml 파싱에러가 납니다.
파라미터가 하나는 들어가는데 두개이상이 안들어 가게 됩니다.

<forward name="order"        path="BoardList.do?code=00001" />
  => 에러 안남

<forward name="order"        path="BoardList.do?code=00001&page=1" />
  => 에러 발생

이 경우에는 다음과 같은 방법으로 코딩하면 다수의
파라미터값을 넘길 수 있습니다.

<forward name="board_insert_success"      path="/BoardList.do?run=BoardList & a m p ; current_page=1"/>

윗 부분의 & a m p ; <=요 부분의 띄어쓰기된 부분을 붙여주면 됩니다.
유용하게 쓰일때가 꼭 있을 거라는 생각이 들어서 올려봤습니다. ^^

[오라클자바닷넷커뮤니티교육, ORACLEJAVANEW.KR]간단한 Validator Framework 예제

간단한 Validator Framework 예제


오라클자바커뮤니티에서 설립한 오엔제이프로그래밍 실무교육센터
(오라클SQL, 튜닝, 힌트,자바프레임워크, 안드로이드, 아이폰, 닷넷 실무전문 강의)  




이번 예제는 이전의 강좌인 "Struts를 이용한 간단한 로그인 예제”의 기능에 Validation 기능을 간단히 추가해 보도록 하겠습니다. "Struts를 이용한 간단한 로그인 예제” 강좌에 나와 있는 내용에 대해서는 이번 강좌에서는 언급하지 않고 Validation을 위해 추가되는 코드 부분만 기술토록 하겠습니다.

혹시 "Struts를 이용한 간단한 로그인 예제”부분이 궁금하신 분들은 해당 강좌를 참고 하시기 바랍니다.

[검증 조건]
사용자 ID는 필수입력, 5~10자를 넘을 수 없으며, 시작은 알파벳으로
비밀번호는 필수입력이며 10자를 넘을 수 없음, 시작은 알파벳, 숫자로…

=========================================================

1.Struts 설정 파일에 Validator Plug-in을 Add 합니다. (메시지 리소스 설정 다음에 추가 합니다.)

<plug-in
className="org.apache.struts.validator.ValidatorPlugIn">
<set-property
property="pathnames"
value="/WEB-INF/validator-rules.xml,/WEB-INF/validation.xml"/>
</plug-in>

----------------------------------------------------------------
2.Blank Struts Web Application에서 validator-rules.xml 과 validation.xml 파일을 복사 합니다. (struts 배포판안에 struts-blank.war 파일이 포함되어 있으며 이 파일을 TOMCAT_HOME/webapps 아래에 둔 후 톰캣을 실행하면 저절로 압축이 풀리며 /WEB-INF/ 아래에 보시면 두개의 파일이 있습니다.)
-----------------------------------------------------------------

참고로 파일의 내용은 다음과 같습니다.

-------------------------------
validation.xml
-------------------------------

<?xml version="1.0" encoding="ISO-8859-1" ?>

<!DOCTYPE form-validation PUBLIC
          "-//Apache Software Foundation//DTD Commons Validator Rules Configuration 1.1.3//EN"
          "http://jakarta.apache.org/commons/dtds/validator_1_1_3.dtd">

<form-validation>

<!--
    This is a minimal Validator form file with a couple of examples.
-->

    <global>

        <!-- An example global constant
        <constant>
            <constant-name>postalCode</constant-name>
            <constant-value>^\d{5}\d*$</constant-value>
        </constant>
        end example-->

    </global>

    <formset>

        <!-- An example form -->
        <form name="logonForm">
            <field
                property="username"
                depends="required">
                    <arg key="logonForm.username"/>
            </field>
            <field
                property="password"
                depends="required,mask">
                    <arg key="logonForm.password"/>
                    <var>
                        <var-name>mask</var-name>
                        <var-value>^[0-9a-zA-Z]*$</var-value>
                    </var>
            </field>
        </form>

    </formset>

    <!-- An example formset for another locale -->
    <formset language="fr">

        <constant>
            <constant-name>postalCode</constant-name>
            <constant-value>^[0-9a-zA-Z]*$</constant-value>
        </constant>

        <!-- An example form -->
        <form name="logonForm">
            <field
                property="username"
                depends="required">
                    <arg key="logonForm.username"/>
            </field>
            <field
                property="password"
                depends="required,mask">
                    <arg key="logonForm.password"/>
                    <var>
                        <var-name>mask</var-name>
                        <var-value>^[0-9a-zA-Z]*$</var-value>
                    </var>
            </field>
        </form>

  </formset>

</form-validation>




-------------------------------
validator-rule.xml
-------------------------------

<!DOCTYPE form-validation PUBLIC
          "-//Apache Software Foundation//DTD Commons Validator Rules Configuration 1.1.3//EN"
          "http://jakarta.apache.org/commons/dtds/validator_1_1_3.dtd">
<!--
  $Id: validator-rules.xml 54932 2004-10-16 17:02:16Z germuska $

  This file contains the default Struts Validator pluggable validator
  definitions.  It should be placed somewhere under /WEB-INF and
  referenced in the struts-config.xml under the plug-in element
  for the ValidatorPlugIn.

      <plug-in className="org.apache.struts.validator.ValidatorPlugIn">
        <set-property property="pathnames" value="/WEB-INF/validator-rules.xml,
                                                  /WEB-INF/validation.xml"/>
      </plug-in>

  These are the default error messages associated with
  each validator defined in this file.  They should be
  added to your projects ApplicationResources.properties
  file or you can associate new ones by modifying the
  pluggable validators msg attributes in this file.

  # Struts Validator Error Messages
  errors.required={0} is required.
  errors.minlength={0} can not be less than {1} characters.
  errors.maxlength={0} can not be greater than {1} characters.
  errors.invalid={0} is invalid.

  errors.byte={0} must be a byte.
  errors.short={0} must be a short.
  errors.integer={0} must be an integer.
  errors.long={0} must be a long.
  errors.float={0} must be a float.
  errors.double={0} must be a double.

  errors.date={0} is not a date.
  errors.range={0} is not in the range {1} through {2}.
  errors.creditcard={0} is an invalid credit card number.
  errors.email={0} is an invalid e-mail address.

  Note: Starting in Struts 1.2.0 the default javascript definitions have
        been consolidated to commons-validator. The default can be overridden
        by supplying a <javascript> element with a CDATA section, just as
        in struts 1.1.

-->

<form-validation>

  <global>

      <validator name="required"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateRequired"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
                  msg="errors.required"/>

      <validator name="requiredif"
                classname="org.apache.struts.validator.FieldChecks"
                method="validateRequiredIf"
                methodParams="java.lang.Object,
                              org.apache.commons.validator.ValidatorAction,
                              org.apache.commons.validator.Field,
                              org.apache.struts.action.ActionMessages,
                              org.apache.commons.validator.Validator,
                              javax.servlet.http.HttpServletRequest"
                msg="errors.required"/>

      <validator name="validwhen"
          msg="errors.required"
                classname="org.apache.struts.validator.validwhen.ValidWhen"
                method="validateValidWhen"
                methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      org.apache.commons.validator.Validator,
                      javax.servlet.http.HttpServletRequest"/>


      <validator name="minlength"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateMinLength"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.minlength"
          jsFunction="org.apache.commons.validator.javascript.validateMinLength"/>


      <validator name="maxlength"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateMaxLength"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.maxlength"
          jsFunction="org.apache.commons.validator.javascript.validateMaxLength"/>



      <validator name="mask"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateMask"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.invalid"/>


      <validator name="byte"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateByte"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.byte"
      jsFunctionName="ByteValidations"/>


      <validator name="short"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateShort"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.short"
      jsFunctionName="ShortValidations"/>


      <validator name="integer"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateInteger"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.integer"
      jsFunctionName="IntegerValidations"/>



      <validator name="long"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateLong"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.long"/>


      <validator name="float"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateFloat"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.float"
      jsFunctionName="FloatValidations"/>

      <validator name="double"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateDouble"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.double"/>


      <validator name="date"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateDate"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.date"
      jsFunctionName="DateValidations"/>


      <validator name="intRange"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateIntRange"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends="integer"
                  msg="errors.range"/>


      <validator name="floatRange"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateFloatRange"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends="float"
                  msg="errors.range"/>


      <validator name="creditCard"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateCreditCard"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.creditcard"/>


      <validator name="email"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateEmail"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.email"/>

      <validator name="url"
            classname="org.apache.struts.validator.FieldChecks"
              method="validateUrl"
        methodParams="java.lang.Object,
                      org.apache.commons.validator.ValidatorAction,
                      org.apache.commons.validator.Field,
                      org.apache.struts.action.ActionMessages,
                      javax.servlet.http.HttpServletRequest"
              depends=""
                  msg="errors.url"/>

    <!--
      This simply allows struts to include the validateUtilities into a page, it should
      not be used as a validation rule.
    -->
    <validator name="includeJavaScriptUtilities"
            classname=""
              method=""
        methodParams=""
              depends=""
                  msg=""
          jsFunction="org.apache.commons.validator.javascript.validateUtilities"/>

  </global>

</form-validation>


       

--------------------------------------------------------------
3.ActionForm 클래스(LoginForm.java)를 다음과 같이 수정 합니다.  LoginForm 클래스가 ValidatorForm 클래스를 상속 받도록 구성 합니다. 물론 import org.apache.struts.validator.ValidatorForm; 문장도 추가 하구요…

혹시 validate() 메소드가 재정의 되어 있다면 주석으로 막으세요~
--------------------------------------------------------------

import org.apache.struts.validator.ValidatorForm;

public class LoginForm extends ValidatorForm {

-----------------------------------------------------------------
4.validation.xml 파일을 다음처럼 수정 합니다. minlength, maxlength인 경우 오류 출력시 매개변수가 두 개 필요 합니다.
-----------------------------------------------------------------

errors.minlength={0} can not be less than {1} characters.
  errors.maxlength={0} can not be greater than {1} characters.
 
  그래서 아래 설정에서 매개변수가 2개로 되어 있으니 참고 바랍니다.

<?xml version="1.0" encoding="ISO-8859-1" ?>

<!DOCTYPE form-validation PUBLIC
          "-//Apache Software Foundation//DTD Commons Validator Rules Configuration 1.1.3//EN"
          "http://jakarta.apache.org/commons/dtds/validator_1_1_3.dtd">

<form-validation>

<!--
    This is a minimal Validator form file with a couple of examples.
-->

    <global>

        <!-- An example global constant
        <constant>
            <constant-name>postalCode</constant-name>
            <constant-value>^\d{5}\d*$</constant-value>
        </constant>
        end example-->

    </global>

    <formset>
        <!-- An example form -->
        <form name="loginForm">
            <field
                property="id"
                depends="required, minlength, maxlength, mask">
                        <arg0 key="prompt.id"/>
                                        <arg1 key="${var:maxlength}"
                                                name="maxlength"
                                                resource="false"/>
                                        <arg1 key="${var:minlength}"
                                                name="minlength"
                                                resource="false"/>
                                        <var>
                                                <var-name>minlength</var-name>
                                                <var-value>5</var-value>
                                        </var>
                                        <var>
                                                <var-name>maxlength</var-name>
                                                <var-value>10</var-value>
                                        </var>
                                        <var>
                        <var-name>mask</var-name>
                        <var-value>^[a-zA-Z]*$</var-value>
                    </var>
            </field>
            <field
                property="pwd"
                depends="required,mask">
                    <arg0 key="prompt.password"/>
                    <arg1 key="${var:maxlength}"
                                                name="maxlength"
                                                resource="false"/>
                    <var>
                        <var-name>mask</var-name>
                        <var-value>^[0-9a-zA-Z]*$</var-value>
                    </var>
                    <var>
                                                <var-name>maxlength</var-name>
                                                <var-value>10</var-value>
                                        </var>
            </field>
        </form>

    </formset>
</form-validation>

-----------------------------------------------
5.validator-rule.xml 파일은 그대로 사용 합니다.
-----------------------------------------------

--------------------------------------------------------------------
6.application.properties 파일은 리소스 번들 파일로서 에러 메시지 출력을 위해 사용 합니다.
--------------------------------------------------------------------

login.title = LogIn
login.login = Login
login.reset = Cancel

prompt.id= User ID :
prompt.password=Password :

main.title=Main

user.depttitle=DeptNo
user.ename=ENAME
user.job=JOB

jdbc.error=SAVE FAIL!!!

# Struts Validator Error Messages
  errors.required={0} is required.
  errors.minlength={0} can not be less than {1} characters.
  errors.maxlength={0} can not be greater than {1} characters.
  errors.invalid={0} is invalid.

  errors.byte={0} must be a byte.
  errors.short={0} must be a short.
  errors.integer={0} must be an integer.
  errors.long={0} must be a long.
  errors.float={0} must be a float.
  errors.double={0} must be a double.

  errors.date={0} is not a date.
  errors.range={0} is not in the range {1} through {2}.
  errors.creditcard={0} is an invalid credit card number.
  errors.email={0} is an invalid e-mail address.



###############################################3
error.id.required=<li>ID REQUIRED.</li>
error.pwd.required=<li>PASSWORD REQUIRED</li>
error.login.invalid=<li>ID/PASSWORD NOT CORRECT</li>

errors.header=<h3><font color="red">ERROR!</font></h3>
                  You must correct the following error(s) before proceeding:<UL>
errors.footer=</ul><hr>

--------------------------
7.struts-config.xml
--------------------------

<!--  Form Bean Definitions -->
    <form-beans>
        <form-bean name="loginForm" type="login2.LoginForm"/>                 
    </form-beans>

        <!--Action Mapping Definitions  -->   
    <action-mappings>
        <!-- loginAction에 대한 정의 -->
        <action         
            path="/LoginSubmit"
            type="login2.LoginAction"
            name="loginForm"                                   
            input="/login.jsp"
        />
        </action-mappings>

2013년 8월 5일 월요일

[스트럿츠 한글, 자바프레임워크교육,자바교육,오라클자바오엔제이]Struts 에서 한글 사용하기

Struts 에서 한글 사용하기


오라클자바커뮤니티에서 설립한 오엔제이프로그래밍 실무교육센터
(오라클SQL, 튜닝, 힌트,자바프레임워크, 안드로이드, 아이폰, 닷넷 실무전문 강의) 




한글의 경우 필요한 시기에 그때그때 인코딩을 해도 되지만 Servlet2.3에서 제공하는 Filter기능을 이용하는 것처럼 스트럿츠에서도 그러한 기능의 구현이 가능 합니다.

--------------------------------------------------
우선 서블릿의  필터 기능을 이용하여 작성한 예입니다.
--------------------------------------------------

[SetCharacterEncodingFilter.java]

package filters;


import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.UnavailableException;



public class SetCharacterEncodingFilter implements Filter {

    protected String encoding = null;

    protected FilterConfig filterConfig = null;


    /**
    * Should a character encoding specified by the client be ignored?
    */
    protected boolean ignore = true;


    public void destroy() {

        this.encoding = null;
        this.filterConfig = null;

    }


    public void doFilter(ServletRequest request, ServletResponse response,
                        FilterChain chain)
        throws IOException, ServletException {

        // Conditionally select and set the character encoding to be used
        if (ignore || (request.getCharacterEncoding() == null)) {
            String encoding = selectEncoding(request);
            if (encoding != null)
                request.setCharacterEncoding(encoding);
        }

        // Pass control on to the next filter
        chain.doFilter(request, response);

    }

    public void init(FilterConfig filterConfig) throws ServletException {

        this.filterConfig = filterConfig;
        this.encoding = filterConfig.getInitParameter("encoding");
        String value = filterConfig.getInitParameter("ignore");
        if (value == null)
            this.ignore = true;
        else if (value.equalsIgnoreCase("true"))
            this.ignore = true;
        else if (value.equalsIgnoreCase("yes"))
            this.ignore = true;
        else
            this.ignore = false;

    }


    protected String selectEncoding(ServletRequest request) {

        return (this.encoding);

    }
}


[web.xml]

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/j2ee/dtds/web-app_2_3.dtd">
<web-app>

<!-- Example filter to set character encoding on each request -->
<filter>
 <filter-name>Set Character Encoding</filter-name>
 <filter-class>filters.SetCharacterEncodingFilter</filter-class>
 <init-param>
  <param-name>encoding</param-name>
  <param-value>EUC-KR</param-value>
 </init-param>
</filter>

<!-- Define filter mappings for the defined filters -->
<filter-mapping>
 <filter-name>Set Character Encoding</filter-name>
 <servlet-name>action</servlet-name>
</filter-mapping>

    <servlet>
        <servlet-name>action</servlet-name>
        <servlet-class>org.apache.struts.action.ActionServlet</servlet-class>
       
        <init-param>
            <param-name>config</param-name>
            <param-value>/WEB-INF/struts-config.xml</param-value>
        </init-param>
        <init-param>
            <param-name>debug</param-name>
            <param-value>3</param-value>
        </init-param>
        <init-param>
            <param-name>detail</param-name>
            <param-value>3</param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>
   
    <servlet-mapping>
        <servlet-name>action</servlet-name>
        <url-pattern>*.do</url-pattern>
    </servlet-mapping>
 
</web-app>


-------------------------------------------------------------
다음은 스트럿츠의 RequestProcessor 클래스를 이용한 예 입니다.
-------------------------------------------------------------

스트러츠의 RequestProcessor클래스의 processPreprocess라는 메소드를 이용하여 다른 작업이 이루어지기 전에 미리 설정할 것들을 이 메소드에서 정의할 수 있는데 …. 이곳에 한글 인코딩 부분을 기술하여 모든 스트럿츠 애플리케이션에서 적용 되도록 합니다.

아래 파일을 MyFilter.java로 저장 합니다. (WEB-INF/classes/filter에 저장)

package filter;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.struts.action.RequestProcessor;

public class MyFilter extends RequestProcessor {

        protected boolean processPreprocess(
                HttpServletRequest request,
                HttpServletResponse response) {
               
                try {
                        request.setCharacterEncoding("euc-kr");
                                               
                        return true;
                } catch (Exception e) {               
                        return false;                       
                }
        }
}


이번에 struts-config.xml 파일에 Controller 매핑을 합니다.

<!-- 한글 문제 처리를 위해 -->
    <controller
            contentType="text/html;charset=euc-kr"
            debug="3"           
            locale="true"
            nocache="true"
            processorClass="filter.MyFilter"/>

이상과 같이 설정하면 폼액션을 통해 넘어 오는 데이터에 대해 한글 변환을 일일이 하지 않아도 됩니다.

Struts의 ActionServlet

Struts의 ActionServlet


오라클자바커뮤니티에서 설립한 오엔제이프로그래밍 실무교육센터
(오라클SQL, 튜닝, 힌트,자바프레임워크, 안드로이드, 아이폰, 닷넷 실무전문 강의) 



Struts의 ActionServlet은 javax.servlet.http.HttpServlet 클래스를 상속 받았으며 HTTP 요청을 스트럿츠의 적절한 핸들러에게 보내는 역할을 수행 합니다.

스트럿츠에서 클라이언트의 요청을 처리하는 초기 진입은 ActionServlet이 담당하고 있습니다. 모델 2에서는 UserServlet이나 또 다른 서블릿을 필요할 때마다 새로 작성할 필요가 있지만 스트럿츠에서는 ActionServlet 하나만을 이용하여 모든 요청을 처리하는 하는 것이 가능 합니다.

Struts1.1 이전 버전에서는 ActionServlet이 요청을 받아 핸들러를 호출하는 유일한 클래스 였지만 1.1 버전 이후 부터는 ora.apache.struts.action.RequestProcessor 라는 클래스가 추가되어 클라이언트의 요청을 처리 할 수 있게 되었습니다.

ActionServlet은 클라이언트의 요청을 받게 되면 URI를 사용하여 어떤 Action이 주어진 요청을 처리할 것인지를 결정 하는데 이때 struts-config.xml 파일에 설정된 Action 태그에 설정된 내용을 확인 합니다. URI는 Action 설정의 path 속성과 일치해야 합니다.

아래 struts-config.xml 파일의 내용을 참고 하세요~

<action         
            path="/LoginSubmit"
            type="login2.LoginAction"
            name="loginForm"                       
            validate="true"
            input="/login.jsp"
        />

만약 login.jsp에서 HTML의 <form> tag에 action=/LoginSubmit 이라고 한 후 submit 버튼을 누르게 되면 톰캣은 struts-config.xml의 action의 path와 일치함을 알고 login2.LoginAction 클래스를 실행시킴으로써 Action을 처리하게 되는 것 입니다.


ActionServlet은 주어진 입력 값 들을 한꺼번에 자바 빈즈로 묶을 수 있는데 이 자바 빈은 스트럿츠 ActionForm 클래스의 서브클래스로 구성 됩니다. 이를 폼빈(FormBean) 이라고 하며 struts-config.xml 파일에서 <form-bean> 이라는 태그에서 설정을 하게 됩니다.

아래는 struts-config.xmml 파일에서 설정 예 입니다.

<form-beans>
        <form-bean name="loginForm" type="login2.LoginForm">                 
            <form-property name="pwd" type="java.lang.String" />
            <form-property name="id" type="java.lang.String" />           
        </form-bean>           
    </form-beans>

이때 <action> 태그의 name과 <form-bean>의 name이 일치함을 잘 보시기 바랍니다.


다른 자바 서블릿과 마찬가지로 Struts의 ActionServlet 역시 웹 애플리케이션의 배치 스크립트(web.xml)에 정의를 해야 하는데 그 내용은 아래와 같습니다.

<!-- ActionServlet Congif ================================-->
        <servlet>
                <servlet-name>action</servlet-name>
                <servlet-class>org.apache.struts.action.ActionServlet</servlet-class>
                <init-param>
                        <param-name>config</param-name>
                        <param-value>/WEB-INF/struts-config.xml</param-value>
                </init-param>
                <load-on-startup>1</load-on-startup>               
        </servlet>
       
       
        <!-- ActionServlet Mapping ====================================-->
        <servlet-mapping>
                <servlet-name>action</servlet-name>
                <url-pattern>*.do</url-pattern>
        </servlet-mapping>

확장자가 .do 로 넘어오는 요청은 ActionServlet으로 서비스 함을 의미합니다. 

2013년 8월 4일 일요일

오라클 프로세스(Oracle Process)

Oracle 프로세스 구조는 다중 작업이 발생하는 방법과 다중 작업이 수행되는 방식을 정의 하므로 
매우 중요합니다. 
A.단일 프로세스 Oracle Instance :단일 사용자 Oracle은 하나의 프로세스가 모든 Oracle Code를 
실행하는 시스템입니다. Oracle의 일부분과 Client응용프로그램을 분리하여 실행하는데 다른 
프로세스를 
사용하지 않습니다.대신 Oracle의 모든 코드와 단일사용자의 데이터베이스 응용 프로그램은 단일 
프로세스에 의해 실행됩니다. 

B.다중 프로세스 Oracle Instance : 다중사용자 Oracle은 여러 개의 프로세스가 Oracle의 각기 
다른 부분을 실행하고 접속한 각사용자들은 개별 프로세스를 실행합니다. 

C.다중 사용자 환경에서의 Oracle 프로세스 
1. 데이터베이스 기록자(DBWR) 
  -  DataBase Buffer의 내용을 데이터 파일에 기록하는 역할을 하며 DataBase Buffer Cache를 
관리하는 Oracle BackGround 프로세스임 
  -  DataBase Buffer Cache에 있는 내용이 수정되면 해당 버퍼는 'Dirty'로 표시되는데 이 
Dirty Buffer를 Disk에 기록하여 Buffer Cache를 깨끗한 상태로 유지하는 역할을 함 
  -  LRU 알고리즘에 의해 최근에 사용된 데이터블록을 메모리에 유지합니다. 
  -  Buffer Cache는 Multi Latch를 자지는데 래치는 공유 데이터구조를 보호하는 자동내부 
잠금입니다. DB_BLOCK_LRU_LATCHES라는 Parameter는 시스템에 구성되는 래치수 
    와 기본적으로 CPU에 설정되는 Latch 수를 제어합니다. 
  -  서버프로세스가 Buffer를 Dirty목록으로 이동시킨후 Dirty목록이 임계길이에 도달했음을 
발견하면 Server 프로세스는 DBWR에 신호를 보냅니다. 
  -  체크포인트가 발생하면 LGWR(로그기록자)는 DBWR에 신호를 보냅니다. 즉 Check Point가 
발생하면 LGWR는 디스크에 기록되어야 하는 수정된 Buffer 목록을 
    지정합니다. DBWR는 디스크에 지정된 Buffer의 내용을 기록합니다. 

2.  로그기록자(LGWR) 
- Redo Log Buffer를 디스크에 있는 Redo Log File에 등록합니다. 
- 사용자 프로세스가 Transaction을 Commit할때의 Commit Record및 매3초 마다의 Redo Log 
 buffer, DBWR가 수정된 버퍼를 디스크에 기록할때의 Redo Buffer의 내용등을 기록합니다. 

- 사용자가 Commit명령을 발생시키면 LGWR는 Commit Record를 즉시 Redo Log Buffer에 넣지만 
해당 데이터 버퍼의 변경은 변경사항을 데이터 파일에 기록하는 것이 
  더 효율적일때까지 지연됩니다. 

3 체크포인트(CKPT) 
- Check Point가 발생하면 Oracle은 모든 DataFile의 Header를 갱신하여 체크포인트임을 
나타내야 합니다. 보통 LGWR가 이러한 작업을 수행하지만 체크포인트가 시스템 Performance를 
저하 시킨다면 CKPT를 활성화하여 체크포인트 수행작업을 LGWR가 수행하는 다른 작업과 분리할 
수 있습니다. 
-  체크포인트가 모든 응용프로그램에 꼭 필요한 것은 아니며 DataFile의 수가 많아 Check 
Point중에 LGWR프로세스 성능이 저하되면 CKPT프로세스를 활성화 할수있다. 
- 초기화 Parameter CHECK_POINT_PROCESS는 CKPT프로세스를 활성화 하거나 비활성화 합니다. 

4. 시스템모니터(SMON) 
- 인스턴스 시작시에 인스턴스의 복구를 수행, 더 이상 사용하지 않는 임시 Segment를 삭제하며 
사용 가능한 빈영역을 더 큰 블록으로 만들기위해 인법한 빈 확장영역을 하나로 통합합니다. 

5. 프로세스모니터(PMON) 
- 사용자 프로세스에 장애가 발생하면 프로세스 복구를 수행,Cache를 비우고 해당 프로세스가 
사용한 자원을 해제하며,활성 Transaction의 Table상태를 재설정하고 Lock을 해제하며 Active 
Process목록에서 ID를 제거한다. 
- 주기적으로 Dispatch와 서버 프로세스의 상태를 점검하고 활동중이 아니면 재시작합니다.  
  
6. 복구자(RECO) 
- 분산 Transaction을 포함한 장애를 자동으로 해결하는 분산 Option과 함께 사용되는 프로세스, 
시스템이 분산 트랜잭션을 허용하고 DISTRIBUTED_TRANSACTIONS가 0보다 클때만 존재합니다. 

7. 아카이브(ARCH) 
- Online Redo Log File이 꽉차면 이를 지정된 저장장소에 저장,Redo Log가 Archive Mode에서 
운영되고 자동 Archiving기능이 활성화 되어 있을 때만 존재합니다. 

8. 잠금(LcKn) 
- 병렬서버 옵션에서 최대 10개 까지의 잠금Process(LCK0~LCK10)가 인스텀스의 Lock을 제공합니다. 
그러나 대부분의 병렬서버 시스템에서는 하나의 LCK프로세스(LCK0)만 
  으로도 충분합니다. 

9. 스냅샷갱신(SNPn) 
- 분산 옵션에서 최대 10갸 까지의 스냅샷갱신프로세스(SNP0~SNP10)가 자동으로 Table SnapShot을 
화면에 다시 표시합니다. 

10. 디스패쳐(Dnnn) 
- 디스패쳐 프로세스를 사용하면 사용자 프로세스는 제한된 수의 서버 프로세스를 공유할 수 
있습니다. 디스패쳐가 없다면 각 사용자 프로세스는 하나의 전용(Dedicated)서 
  버 프로세스를 필요로 합니다. 그러나 Multi Thread Server에서는 동일한 수의 사용자에 
대해 더 적은 수의 공유서버 프로세스를 필요로 합니다. 
- 단일 DataBase Instance에 대해 다중 Dispatcher 프로세스를 생성할 수 있습니다. Oracle을 
사용하는 시스템의 Network Protocol에 대해 적어도 하나의 디스페처가 생성되어야 합니다. DBA는 
프로세스당 접속수에 대한 OS의 한계에 따라 가장 적합한수의 디스패처를 시작해야 하며 
인스턴스가 실행되는 동안 디스패처 프로세스를 추가또는 삭제할수 있다. 

- 인스턴스가 시작되면 Listener는 사용자를 Oracle과 접속시키는 통신경로를 설정하며, 
각 디스패처는 리스너에 접속 요구사항을 수신할 수 있는 주소를 부여합니다. 
사용자 프로세스가 접속을 요구하면 리스너는 요구사항을 점검하고 사용자가 디스패처를 사용할 수 
있을지에 대해 결정합니다. 사용할 수 있을 경우 리스너는 로드할 양이 가장적은 디스패처 주소를 
돌려주며 사용자 프로세스는 디스패처에 직접 접속됩니다. 


[출처]오라클자바커뮤니티, 오엔제이프로그래밍
www.onjprogramming.co.kr

oracle instance

DataBase가 시작될때 Oracle은 시스템글로벌영역(SGA) 
이라는 메모리 영역을 할당하여 하나 이상의 Oracle Process를 시작합니다. 
SGA와 Process를 합쳐 DataBase Instance라고 합니다. 
Instance의 memory와 process들은 database의 data를 효율적으로 관리하기 위해서 사용되며 관련 
database에 대하여 하나의 user 또는 여러 user들에 의해서 이용된다. 

[SGA ---> DataBase Buffer Cache, Redo Log Buffer, Share Pool등으로 구성] 
자세한 부분별 설명은 다은 강좌에 이어 집니다. 

Instance를 시작한다음 인스턴스에 DataBase를 Mount합니다. 
다중 인스턴스가 동시에 자신의 물리적 DataBase를 Access하면서 
동일한 Computer에서 실행될수 있습니다. 
Oracle병렬서버를 사용하면 다중인스턴스에 단일 데이터베이스를 
마운트할수있습니다. 

[출처]오라클자바커뮤니티, 오엔제이프로그래밍
www.onjprogramming.co.kr

2013년 8월 3일 토요일

SQL기초에서 활용 튜닝, 힌트 그리고 오라클 기본 관리자 역할까지 ORACLE의 알아야 하는 부분을 대부분 들여다 보는 오라클 핵신 실무 과정 입니다.

SQL기초에서 활용 튜닝, 힌트 그리고 오라클 기본 관리자 역할까지 ORACLE의 알아야 하는 부분을 대부분 들여다 보는 오라클 핵신 실무 과정 입니다.


강좌명 오라클 마스터(주말주간(토/일))
교재 자체 교재 무료 제공
강좌 일정 08월10일(토) ~ 09월01일(일)((주말주간(토/일)) 10:00~18:00, 8일) 총 56시간
강의 장소 [C강의장]구로디지털단지역2번 출구-> 미니스톱끼고 우회전 -> 100m 직진 후 골목길 끝에서 이마트방향 우회전 -> 50m 직진 후 우체국 옆골목으로 길건너서 직진 -> 150미터 직진 후 JnK 타워에서 우회전 -> 50미터 직진 후 우측에 코오롱빌란트2차 803호 (구로구 구로3동 222-8 코오롱디지털타워 빌란트2차 803호)
[약도보기]
수강절차 - 강좌내용 확인
- 전화 또는 홈페이지(www.onjprogramming.co.kr)를 통한 수강지원 및 수강료 결제(무통장입금, 온라인 카드결제)
- 고용보험 가입자(재직자)인 경우 고용보험환급 관련 서류 제출
- 수강전 : 커리큘럼 및 장소에 대해 다시 한번 공지
- 교육 전 설문 작성(간단한 개발 경력, 수강 목적, 강좌진행방식 등)
- 강좌 수강
- 수강후 : 교육 후 설문 작성
수강료 - 690,000원
[고용주환급]대기업:21만원 전후,중소기업:285,254원
[개인수강지원(개인환급)]정규직:552,000원, 비정규직:전액환급

대기업(상시근로자 300인 이상 대기업)은 개인환급 불가합니다.


* 휴강 :법정공휴일
수강료
입금안내
- 온/오프라인 카드결제, 계좌이체(수강안내->입금안내 참조)
문의사항 02-851-4790 번으로 연락 부탁 드립니다.
교육개요 본 과정은 오라클 사용에 대해 불편을 느끼시는 분들을 위해 다양한 예제와 실습을 통해 오라클에 대한 내공을 키워가는 과정으로 오라클을 전체적으로 학습할 수 있으며 실무 활용능력을 높여 추후 자격증 취득이 필요한 경우 무난히 자격증을 취득할 수 있도록 기본을 다지는 과정 입니다.

처음 접하는 사람들도 수강이 가능하도록 SQL기초부터 체계적으로 교육을 진행하여 기본적인 SQL부터 PL/SQL, 기본적인 Oracle Admin 과정인 오라클 서버 구조/아키텍처를 배움으로써 오라클 서버의 작동 원리 및 구조에 대해서도 학습하며 수업 중간중간 간단한 Oracle Tip등도 소개해 드립니다.

진정한 오라클 교육을 원하시는 분들의 많은 참여 바랍니다.
교육목표 - SQL 기본함수
- PL/SQL 사용법 습득(Function, Procedure)
- Oracle Schema Object 사용법(Table, Index, View, Sequence, Synonym, DBLink)
- Oracle Basic Administration
- DataBase Structure(Logical/Physical)
- Oracle Architecture
교육대상 - 오라클 데이터베이스에 대하여 초보 수준의 학생 및 직장인
- 오라클에 관심이 있는 개발자
선수학습 - 데이터베이스 개론
 


SQL*Plus/iSQL*Plus 오라클설치
RDBMS 개념
SQL*Plus /iSQL*Plus사용법
SQL*Plus 기본
SQL 기본/고급 산술표현식 및 합성연산자
컬럼 Alias, ALL/Distinct 연산자
조건질의와 Order By
SQL연산자
집합연산자(Set Operator)
기본SQL 함수(문자/숫자/날짜관련)
Conversion Function
Analytical Function
Aggreate Function
GROUP BY, HAVING
Sub Query
Join(Outer Join, Self Join, Inner Join등)
고급SQL
- 효율적인 SQL작성 사례
- 집계용 함수사용 ( ROLLUP,CUBE,GROUPINGSET)
- 분석함수(Analytical Function) 사용
- 순환관계처리 및 병렬처리
- Oracle 정규표현식(Regular Expression)
Oracle Schema Object Table & Constraints
Index
데이터조작(Data Manipulation)
트랜잭션제어(Transaction Control)
데이터정의(Data Definition)
View, Index, Sequence, Synonym
Oracle User, Grant, Privilege, Role
오라클 구조 SGA(DataBase Buffer Cache, Redo Log Buffer, Shared Pool, Library Cache, Dictionary Cache, Large Pool, Java Pool)
PGA/UGA
Oracle Server Process(SMON, PMON, RECO, LMS, ARCH, LGWR, DBWR, CKPT, Pnnn, Dnnn등)
Shared Server & Dedicated Server
Logical Database Structure(Data Block, Extents, Segment, Tablespace)
Physical Database Structure(DataFile, Control File, Redo Log File, Parameter File, Temporary File)
고급 SQL(Hint) Oracle Hint 소개
Optimazer 소개(Rule-Based, Cost-Based)
Optimizer Mode 설정 방법
실행계획 소개 및 해석 방법
실행계획 SQL연산
Optimizer Mode를 변경하는 힌트
Access경로를 변경하는 힌트
Join순서를 변경하는 힌트
기타 Oracle Tip
 

[Oracle Hint]ACCESS 경로를 변경하는 힌트(INDEX_ASC), 오라클힌트강좌 , oracle hint, oraclejava 강좌

ACCESS 경로를 변경하는 힌트(INDEX_ASC)

구로디지털 오엔제이프로그래밍실무교육센터

앞선 강좌의 INDEX 힌트와 동일한데 보이는 그대로 인덱스를 스캔 하는데 오름차순(ASCENDING) 스캔 하라는 의미의 힌트 입니다. 이 힌트를 이용하여 데이터를 추출하게 되면 화면에 나타나는 데이터는 인덱스 키를 기준으로 오름차순 정렬된 모습으로 나타나게 된다.

만약 EMP 테이블에서 사원들의 이름과 급여를 출력하는데 이름순으로 정렬을 하라고 했을 때

다음과 같이 하면 되지만 이 경우엔 SORT를 위한 TEMP 영역을 사용하게 되므로 많은 양의 데이터를 추출하는 경우엔 상당한 OVERHEAD가 따르게 됩니다. 되도록 이런 식으로는 사용하지 않는 것이 좋다. (예제에서는 데이터가 몇 건 되지 않는 EMP 테이블의 결과이지만 대용량의 테이블 이라면 속도 차이가 상당할 것입니다.)

select ename, sal
from   emp
order  by ename asc

---------------------------------------------------------------------
Operation            Object Name      Rows     Bytes    Cost     
-------------------------------------------------------------------
SELECT STATEMENT Optimizer Mode=ALL_ROWS               15                      4
  SORT ORDER BY                        15         135       4                                   
    TABLE ACCESS FULL             SCOTT.EMP        15         135       3                                                     

이번엔 힌트를 이용해 보도록 합니다.

select
       ename, sal
from   emp
where  ename > ' '

---------------------------------------------------------------------
Operation            Object Name      Rows     Bytes    Cost     
-------------------------------------------------------------------
SELECT STATEMENT Optimizer Mode=ALL_ROWS               15                      2
  TABLE ACCESS BY INDEX ROWID            SCOTT.EMP        15         135       2            
    INDEX <st1:placetype w:st="on">RANGE</st1:placetype> SCAN   SCOTT.IDX_EMP_ENAME 15                      1                                                     

이번에는 INDEX_ASC를 사용해 보도록 하죠 INDEX 힌트와 동일한 결과를 보인다.

select
       ename, sal
from   emp
where  ename > ' '


---------------------------------------------------------------------
Operation            Object Name      Rows     Bytes    Cost     
-------------------------------------------------------------------
SELECT STATEMENT Optimizer Mode=ALL_ROWS               15                      2
  TABLE ACCESS BY INDEX ROWID            SCOTT.EMP        15         135       2            
    INDEX <st1:placetype w:st="on">RANGE</st1:placetype> SCAN   SCOTT.IDX_EMP_ENAME 15                      1





[실습]

create table myemp1
(empno number not null primary key,
 ename varchar2(100),
 deptno number,
 addr   varchar2(100),
 sal    number
 )

 create table mydept1
 (deptno number,
  dname  varchar2(100)
  )
 
 insert into mydept1 values (0, '인사팀');
 insert into mydept1 values (1, '회계팀');
 insert into mydept1 values (2, '영업팀');
 insert into mydept1 values (3, '기획팀');
 insert into mydept1 values (4, '교육팀');

 commit

-- 실습을 위해 myemp1 2000만건 만들자.
DECLARE
          v_c NUMBER := 1;
BEGIN

          WHILE (v_c <= 10000000) LOOP
                insert into myemp1 values ( v_c, '홍길동'||v_c, mod(v_c, 5), '서울'||v_c, mod(v_c, 1000000));
                v_c := v_c + 1;
                insert into myemp1 values ( v_c, '다길동'||v_c, mod(v_c, 5), '부산'||v_c, mod(v_c, 1000000));
                v_c := v_c + 1;
                insert into myemp1 values ( v_c, '나길동'||v_c, mod(v_c, 5), '대구'||v_c, mod(v_c, 1000000));
                v_c := v_c + 1;
                insert into myemp1 values ( v_c, '나길동'||v_c, mod(v_c, 5), '광주'||v_c, mod(v_c, 1000000));
                v_c := v_c + 1;
          END LOOP;
          commit;
END;


create index idx_myemp1_deptno on myemp1(deptno)

analyze table myemp1 compute statistics
analyze table mydept1 compute statistics

-------------------------------------------------- 실습데이터 생성 끝


SQL> select count(*) from myemp1;

  COUNT(*)
----------
  10000000

SQL> select count(*) from mydept1;

  COUNT(*)
----------
         5

SQL> select index_name, table_name from user_indexes
  2  where table_name = 'MYEMP1';

INDEX_NAME                     TABLE_NAME
------------------------------ ------------------------------
IDX_MYEMP1_DEPTNO              MYEMP1
SYS_C0011302                   MYEMP1


이름으로 인덱스를 만들자
SQL> create index idx_myemp1_ename on myemp1(ename);

인덱스가 생성되었습니다.

실습을 위해 인덱스를 숨기자.
SQL> alter index idx_myemp1_ename invisible;

인덱스가 변경되었습니다.


인덱스가 없는 상태에서 이름으로 조건 검색을 하니 9초 정도 걸린다.
SQL> select count(ename) from myemp1
  2  where ename = '홍길동111';

COUNT(ENAME)
------------
           0

   : 00:00:09.95

Execution Plan
----------------------------------------------------------
|   0 | SELECT STATEMENT   |        |     1 |    13 | 16956   (1)| 00:03:24 |
|   1 |  SORT AGGREGATE    |        |     1 |    13 |            |          |
|*  2 |   TABLE ACCESS FULL| MYEMP1 |     1 |    13 | 16956   (1)| 00:03:24


이번에는 인덱스를 보이도록 하고 다시 검색하자.

SQL> alter index idx_myemp1_ename visible;

바로 데이터가 보인다.


SQL>  select count(ename) from myemp1
  2   where ename = '홍길동111';

COUNT(ENAME)
------------
           0

   : 00:00:00.03

Execution Plan
----------------------------------------------------------
|   0 | SELECT STATEMENT  |                  |     1 |    13 |     3   (0)|
|   1 |  SORT AGGREGATE   |                  |     1 |    13 |            |
|*  2 |   INDEX RANGE SCAN| IDX_MYEMP1_ENAME |     1 |    13 |     3  


SQL> -- 이번에는 이름으로 오름차순으로 데이터를 가지고 와 보자.
SQL> -- oracle 11g에서 했는데 너무 느리다. 디스크 소트도 한번 하고

SQL> select empno, ename from myemp1
  2  order by ename asc
  3  /

10000000 개의 행이 선택되었습니다.

   : 00:02:16.57  //2분 넘었다.

Execution Plan
----------------------------------------------------------
|   0 | SELECT STATEMENT   |        |    10M|   171M|       | 74120   (1)|
|   1 |  SORT ORDER BY     |        |    10M|   171M|   268M| 74120   (1)|
|   2 |   TABLE ACCESS FULL| MYEMP1 |    10M|   171M|       | 16931  


Statistics
----------------------------------------------------------
        268  recursive calls
          8  db block gets
      61364  consistent gets
      95565  physical reads
       1620  redo size
  294444947  bytes sent via SQL*Net to client
    7333741  bytes received via SQL*Net from client
     666668  SQL*Net roundtrips to/from client
          0  sorts (memory)
          1  sorts (disk)
   10000000  rows processed


ename 인덱스 영역에서 가지고 오면 금방 가지고 올텐데 힌트를 써 보자.

SQL> select
      Count(ename) 
 from myemp1
where ename is not null;

6초 정도 걸린다. 인덱스 풀 스캔해서



이번에는 max(sal), min(sal)을 구해보자.

SQL> create index idx_myemp1_sal on myemp1(sal)
  2  /

인덱스가 생성되었습니다.


SQL>-- sal 인덱스를 이용하여 간단하게 가지고 온다.
SQL> select max(sal) from myemp1;

  MAX(SAL)
----------
    999999

   : 00:00:00.01

Execution Plan
|   0 | SELECT STATEMENT           |                |     1 |     4 |     3  
|   1 |  SORT AGGREGATE            |                |     1 |     4 |
|   2 |   INDEX FULL SCAN (MIN/MAX)| IDX_MYEMP1_SAL |     1 |     4 |     3  


SQL> -- 최소값도 인덱스로 가면 금방 가지고 올 수 있다.
SQL> select min(sal) from myemp1;

  MIN(SAL)
----------
         0

   : 00:00:00.03

Execution Plan
----------------------------------------------------------
|   0 | SELECT STATEMENT           |                |     1 |     4 |     3  
|   1 |  SORT AGGREGATE            |                |     1 |     4 |
|   2 |   INDEX FULL SCAN (MIN/MAX)| IDX_MYEMP1_SAL |     1 |     4 |     3  



그럼, 다음처럼 최대값에서 최소값을 뺄 때는 어떻게 ?
myemp1 FULL SCAN 한다..

SQL> select max(sal) - min(sal) from myemp1;

MAX(SAL)-MIN(SAL)
-----------------
           999999

   : 00:00:09.93



Execution Plan
----------------------------------------------------------
|   0 | SELECT STATEMENT   |        |     1 |     4 | 16961   (1)| 00:03:24 |
|   1 |  SORT AGGREGATE    |        |     1 |     4 |            |          |
|   2 |   TABLE ACCESS FULL| MYEMP1 |    10M|    38M| 16961   (1)|


다음처럼 해보자.


SQL> with a as (
  2    select max(sal) max from myemp1
  3  ) , b  as (
  4     select min(sal) min from myemp1
  5  )
  6* select a.max - b.min from a, b

A.MAX-B.MIN
-----------
     999999

   : 00:00:00.01  //바로 나온다.

Execution Plan
----------------------------------------------------------
|   0 | SELECT STATEMENT             |                |     1 |    26 |     6
|   1 |  NESTED LOOPS                |                |     1 |    26 |     6
|   2 |   VIEW                       |                |     1 |    13 |     3
|   3 |    SORT AGGREGATE            |                |     1 |     4 |
|   4 |     INDEX FULL SCAN (MIN/MAX)| IDX_MYEMP1_SAL |     1 |     4 |    
|   5 |   VIEW                       |                |     1 |    13 |     3
|   6 |    SORT AGGREGATE            |                |     1 |     4 |
|   7 |     INDEX FULL SCAN (MIN/MAX)| IDX_MYEMP1_SAL |     1 |     4 |    


물론 다음과 같이 힌트를 사용해도 된다.

with a as (
  select /*+ index(myemp1 idx_myemp1_sal) */ max(sal) max from myemp1
) , b  as (
   select /*+ index(myemp1 idx_myemp1_sal) */ min(sal) min from myemp1
)
select a.max - b.min from a, b