perf-tuning.html.ko.euc-kr 46.8 KB
Newer Older
powelld's avatar
powelld committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000
<?xml version="1.0" encoding="EUC-KR"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" lang="ko" xml:lang="ko"><head>
<meta content="text/html; charset=EUC-KR" http-equiv="Content-Type" />
<!--
        XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
              This file is generated from xml source: DO NOT EDIT
        XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
      -->
<title>아파치 성능향상 - Apache HTTP Server Version 2.4</title>
<link href="../style/css/manual.css" rel="stylesheet" media="all" type="text/css" title="Main stylesheet" />
<link href="../style/css/manual-loose-100pc.css" rel="alternate stylesheet" media="all" type="text/css" title="No Sidebar - Default font size" />
<link href="../style/css/manual-print.css" rel="stylesheet" media="print" type="text/css" /><link rel="stylesheet" type="text/css" href="../style/css/prettify.css" />
<script src="../style/scripts/prettify.min.js" type="text/javascript">
</script>

<link href="../images/favicon.ico" rel="shortcut icon" /></head>
<body id="manual-page"><div id="page-header">
<p class="menu"><a href="../mod/">모듈</a> | <a href="../mod/directives.html">지시어들</a> | <a href="http://wiki.apache.org/httpd/FAQ">FAQ</a> | <a href="../glossary.html">용어</a> | <a href="../sitemap.html">사이트맵</a></p>
<p class="apache">Apache HTTP Server Version 2.4</p>
<img alt="" src="../images/feather.png" /></div>
<div class="up"><a href="./"><img title="&lt;-" alt="&lt;-" src="../images/left.gif" /></a></div>
<div id="path">
<a href="http://www.apache.org/">Apache</a> &gt; <a href="http://httpd.apache.org/">HTTP Server</a> &gt; <a href="http://httpd.apache.org/docs/">Documentation</a> &gt; <a href="../">Version 2.4</a> &gt; <a href="./">Miscellaneous Documentation</a></div><div id="page-content"><div id="preamble"><h1>아파치 성능향상</h1>
<div class="toplang">
<p><span>가능한 언어: </span><a href="../en/misc/perf-tuning.html" hreflang="en" rel="alternate" title="English">&nbsp;en&nbsp;</a> |
<a href="../fr/misc/perf-tuning.html" hreflang="fr" rel="alternate" title="Fran&#231;ais">&nbsp;fr&nbsp;</a> |
<a href="../ko/misc/perf-tuning.html" title="Korean">&nbsp;ko&nbsp;</a> |
<a href="../tr/misc/perf-tuning.html" hreflang="tr" rel="alternate" title="T&#252;rk&#231;e">&nbsp;tr&nbsp;</a></p>
</div>
<div class="outofdate">이 문서는 최신판 번역이 아닙니다.
            최근에 변경된 내용은 영어 문서를 참고하세요.</div>


    <p>아파치 2.0은 기능과 포팅가능성과 성능의 균형이 맞도록
    설계한 범용 웹서버이다. 벤치마크 기록을 세우기위해 설계하지
    않았지만 아파치 2.0은 실제 많은 경우 높은 성능을 낸다.</p>

    <p>아파치 1.3과 비교해서 2.0 버전은 처리량과 확장성(scalability)을
    높이기위해 많은 최적화를 했다. 기본값으로 대부분 최적화한
    값을 사용한다. 그러나 컴파일시 혹은 실행시 설정이 성능에
    큰 영향을 줄 수 있다. 이 문서는 아파치 2.0의 성능을 향상하기위해
    서버 관리자가 설정할 수 있는 옵션을 설명한다. 어떤 설정
    옵션은 웹서버가 하드웨어와 운영체제의 기능을 더 잘 활용하도록
    하는 반면, 어떤 옵션은 속도를 위해 기능을 희생한다.</p>

  </div>
<div id="quickview"><a href="https://www.apache.org/foundation/contributing.html" class="badge"><img src="https://www.apache.org/images/SupportApache-small.png" alt="Support Apache!" /></a><ul id="toc"><li><img alt="" src="../images/down.gif" /> <a href="#hardware">하드웨어와 운영체제에 대해서</a></li>
<li><img alt="" src="../images/down.gif" /> <a href="#runtime">실행시 설정에 대해서</a></li>
<li><img alt="" src="../images/down.gif" /> <a href="#compiletime">컴파일시 설정에 대해서</a></li>
<li><img alt="" src="../images/down.gif" /> <a href="#trace">부록: 시스템호출 기록을 자세히 분석하기</a></li>
</ul><h3>참고</h3><ul class="seealso"><li><a href="#comments_section">Comments</a></li></ul></div>
<div class="top"><a href="#page-header"><img alt="top" src="../images/up.gif" /></a></div>
<div class="section">
<h2><a name="hardware" id="hardware">하드웨어와 운영체제에 대해서</a></h2>

    

    <p>웹서버 성능에 가장 큰 영향을 주는 것은 메모리다. 스왑은
    요청당 지연시간을 사용자가 "충분히 빠르다고" 생각하지 못하게
    늘리기때문에 웹서버는 스왑을 하면 안된다. 느려지면 사용자는
    정지하고 다시 접속하여 부하가 계속 증가한다. <code class="directive"><a href="../mod/mpm_common.html#maxclients">MaxClients</a></code> 지시어를 조절하여
    웹서버가 스왑을 할 정도로 많은 자식을 만들지않도록 해야
    한다. 방법은 간단하다: <code>top</code>과 같은 도구에서
    프로세스 목록을 보고 아파치 프로세스의 평균 메모리 사용량을
    알아낸후, 전체 사용가능한 메모리에서 다른 프로세스들이 사용할
    공간을 뺀 값에서 나눈다.</p>

    <p>나머지는 평범하다: 충분히 빠른 CPU, 충분히 빠른 네트웍카드,
    충분히 빠른 디스크, 여기서 "충분히 빠른"은 실험을 해서 결정해야
    한다.</p>

    <p>운영체제는 보통 각자 알아서 선택할 일이다. 그러나 일반적으로
    유용하다고 판명된 몇가지 지침이 있다:</p>

    <ul>
      <li>
        <p>선택한 운영체제의 최신 안정 버전과 패치를 실행한다.
        많은 운영체제 제작사는 최근 TCP 스택과 쓰레드 라이브러리에
        많은 속도향상을 했다.</p>
      </li>

      <li>
        <p>운영체제가 <code>sendfile(2)</code> 시스템호출을
        지원한다면, 이를 사용하기위한 버전이나 패치를 설치하였는지
        확인한다. (예를 들어, 리눅스라면 2.4 이상 버전을 뜻한다.
        Solaris 8 초기 버전은 패치가 필요하다.) 지원하는 시스템이라면
        아파치 2는 <code>sendfile</code>을 사용하여 CPU를 덜
        사용하며 정적 파일을 더 빨리 전송할 수 잇다.</p>
      </li>
    </ul>

  </div><div class="top"><a href="#page-header"><img alt="top" src="../images/up.gif" /></a></div>
<div class="section">
<h2><a name="runtime" id="runtime">실행시 설정에 대해서</a></h2>

    

    <table class="related"><tr><th>관련된 모듈</th><th>관련된 지시어</th></tr><tr><td><ul><li><code class="module"><a href="../mod/mod_dir.html">mod_dir</a></code></li><li><code class="module"><a href="../mod/mpm_common.html">mpm_common</a></code></li><li><code class="module"><a href="../mod/mod_status.html">mod_status</a></code></li></ul></td><td><ul><li><code class="directive"><a href="../mod/core.html#allowoverride">AllowOverride</a></code></li><li><code class="directive"><a href="../mod/mod_dir.html#directoryindex">DirectoryIndex</a></code></li><li><code class="directive"><a href="../mod/core.html#hostnamelookups">HostnameLookups</a></code></li><li><code class="directive"><a href="../mod/core.html#enablemmap">EnableMMAP</a></code></li><li><code class="directive"><a href="../mod/core.html#enablesendfile">EnableSendfile</a></code></li><li><code class="directive"><a href="../mod/core.html#keepalivetimeout">KeepAliveTimeout</a></code></li><li><code class="directive"><a href="../mod/prefork.html#maxspareservers">MaxSpareServers</a></code></li><li><code class="directive"><a href="../mod/prefork.html#minspareservers">MinSpareServers</a></code></li><li><code class="directive"><a href="../mod/core.html#options">Options</a></code></li><li><code class="directive"><a href="../mod/mpm_common.html#startservers">StartServers</a></code></li></ul></td></tr></table>

    <h3><a name="dns" id="dns">HostnameLookups와 DNS에 대해 고려할 점들</a></h3>

      

      <p>아파치 1.3 이전에 <code class="directive"><a href="../mod/core.html#hostnamelookups">HostnameLookups</a></code>의 기본값은
      <code>On</code>이였다. 요청을 마치기전에 DNS 검색이 끝나야
      하므로 요청마다 지연이 생겼다. 아파치 1.3에서 이 설정의
      기본값이 <code>Off</code>로 변경되었다. 로그파일의 주소를
      호스트명으로 변환하려면 여러 로그처리 프로그램중 하나인,
      아파치에 포함된 <a href="../programs/logresolve.html"><code>logresolve</code></a>
      프로그램을 사용하라.</p>

      <p>로그처리 작업이 서버 성능에 악영향을 미치므로 실제
      사용하는 웹서버가 아닌 다른 컴퓨터에서 로그파일을 후처리하길
      바란다.</p>

      <p><code><code class="directive"><a href="../mod/mod_access.html#allow">Allow</a></code>
      from domain</code>이나 <code><code class="directive"><a href="../mod/mod_access.html#deny">Deny</a></code> from domain</code>
      지시어를 사용한다면 (즉, IP 주소가 아닌 호스트명이나 도메인명을
      사용한다면) 부득이 중복-역 DNS 검색을 (역검색을 한후 악의로
      변경되었는지 확인하기위해 다시 검색) 해야 한다. 그러므로
      성능을 높이기위해 이런 지시어에는 가능하면 이름대신 IP
      주소를 사용한다.</p>

      <p><code>&lt;Location /server-status&gt;</code> 섹션 등으로
      지시어의 적용범위를 제한할 수 있음을 기억하라. 이 경우
      조건에 맞는 요청에만 DNS 조회를 한다. 다음은
      <code>.html</code><code>.cgi</code> 파일만 DNS 검색을
      하는 예제다:</p>

      <div class="example"><p><code>
        HostnameLookups off<br />
        &lt;Files ~ "\.(html|cgi)$"&gt;<br />
        <span class="indent">
          HostnameLookups on<br />
        </span>
        &lt;/Files&gt;
      </code></p></div>

      <p>그러나 CGI에서 DNS명이 필요할 뿐이라면, 필요한 특정
      CGI에서만 <code>gethostbyname</code> 호출을 하도록 고려해볼
      수 있다.</p>

    

    <h3><a name="symlinks" id="symlinks">FollowSymLinks와 SymLinksIfOwnerMatch</a></h3>

      

      <p>URL 공간에서 <code>Options FollowSymLinks</code>
      사용하지않고 <code>Options SymLinksIfOwnerMatch</code>
      사용하면 아파치는 심볼링크를 검사하기위해 시스템호출을
      한번 더 해야 한다. 파일명의 각 부분마다 한번씩 더 호출을
      한다. 예를 들어, 설정이 다음과 같고:</p>

      <div class="example"><p><code>
        DocumentRoot /www/htdocs<br />
        &lt;Directory /&gt;<br />
        <span class="indent">
          Options SymLinksIfOwnerMatch<br />
        </span>
        &lt;/Directory&gt;
      </code></p></div>

      <p><code>/index.html</code> URI에 대한 요청이 있다고 가정하자.
      그러면 아파치는 <code>/www</code>, <code>/www/htdocs</code>,
      <code>/www/htdocs/index.html</code> 각각에 대해
      <code>lstat(2)</code>를 호출한다. <code>lstats</code>
      결과를 캐싱하지 않기때문에 요청이 들어올 때마다 매번 같은
      작업을 한다. 진짜 심볼링크 보안 검사를 원한다면 다음과
      같이 할 수 있다:</p>

      <div class="example"><p><code>
        DocumentRoot /www/htdocs<br />
        &lt;Directory /&gt;<br />
        <span class="indent">
          Options FollowSymLinks<br />
        </span>
        &lt;/Directory&gt;<br />
        <br />
        &lt;Directory /www/htdocs&gt;<br />
        <span class="indent">
          Options -FollowSymLinks +SymLinksIfOwnerMatch<br />
        </span>
        &lt;/Directory&gt;
      </code></p></div>

      <p>이 경우 최소한 <code class="directive"><a href="../mod/core.html#documentroot">DocumentRoot</a></code> 경로는 검사하지
      않는다. DocumentRoot 밖에 있는 경로로 <code class="directive"><a href="../mod/mod_alias.html#alias">Alias</a></code><code class="directive"><a href="../mod/mod_rewrite.html#rewriterule">RewriteRule</a></code>을 사용한
      경우에도 위와 비슷한 섹션이 필요하다. 심볼링크 보안을
      고려하지 않고 최고의 성능을 얻으려면,
      <code>FollowSymLinks</code>를 설정하고,
      <code>SymLinksIfOwnerMatch</code>는 절대로 안된다.</p>

    

    <h3><a name="htaccess" id="htaccess">AllowOverride</a></h3>

      

      <p>URL 공간에서 overrides를 허용한다면 (보통
      <code>.htaccess</code> 파일) 아파치는 파일명의 각 부분마다
      <code>.htaccess</code>를 열길 시도한다. 예를 들어,</p>

      <div class="example"><p><code>
        DocumentRoot /www/htdocs<br />
        &lt;Directory /&gt;<br />
        <span class="indent">
          AllowOverride all<br />
        </span>
        &lt;/Directory&gt;
      </code></p></div>

      <p><code>/index.html</code> URI에 대한 요청이 있다고 가정하자.
      아파치는 <code>/.htaccess</code>, <code>/www/.htaccess</code>,
      <code>/www/htdocs/.htaccess</code>를 열려고 시도한다.
      해결책은 앞의 <code>Options FollowSymLinks</code> 경우와
      비슷하다. 최고의 성능을 얻으려면 파일시스템에 대해서 항상
      <code>AllowOverride None</code>을 사용한다.</p>

    

    <h3><a name="negotiation" id="negotiation">내용협상</a></h3>

      

      <p>가능하고 진짜 조금의 성능향상에도 관심이 있다면 내용협상을
      막는다. 실제로 협상의 이득은 성능저하보다 작다. 서버를
      빠르게 할 수 있다. 다음과 같이 와일드카드를 사용하는 대신:</p>

      <div class="example"><p><code>
        DirectoryIndex index
      </code></p></div>

      <p>완전한 목록을 사용한다:</p>

      <div class="example"><p><code>
        DirectoryIndex index.cgi index.pl index.shtml index.html
      </code></p></div>

      <p>가장 흔한 것을 앞에 둔다.</p>

      <p>또, 디렉토리에서 파일들을 찾는 <code>MultiViews</code>
      보다는, 한 파일만 읽으면 필요한 정보를 얻을 수 있는
      <code>type-map</code> 파일을 직접 만드는 것이 더 빠름을
      명심하라.</p>

    <p>사이트에 내용협상이 필요하다면 협상을 위해 <code>Options
    MultiViews</code> 지시어를 사용하기보다 <code>type-map</code>
    파일을 고려하라. 협상방법에 대한 자세한 설명과
    <code>type-map</code> 파일을 만드는 방법은 <a href="../content-negotiation.html">내용협상</a> 문서를 참고하라.</p>

    

    <h3>메모리대응 (memory-mapping)</h3>

      

      <p>예를 들어, server-side-include를 처리하는 등 아파치
      2.0이 전송할 파일을 읽을때 운영체제가 <code>mmap(2)</code>
      등을 지원한다면 파일을 메모리대응한다.</p>

      <p>여러 플래폼에서 메모리대응을 성능을 향상한다. 그러나
      메모리대응이 서버의 성능을 떨어트리고 심지어 안정성을
      해치는 경우가 있다:</p>

      <ul>
        <li>
          <p>어떤 운영체제에서 <code>mmap</code>은 CPU 개수가
          많아질때 <code>read(2)</code> 만큼 확장성이 좋지 않다.
          예를 들어, 다중프로세서 Solaris 서버에서 아파치 2.0은
          종종 <code>mmap</code>을 사용하지 않을때 서버가 처리한
          파일을 더 빨리 전송한다.</p>
        </li>

        <li>
          <p>NFS 마운트한 파일시스템에 있는 파일을 메모리대응하는
          도중에 다른 NFS 클라이언트에 있는 프로세스가 파일을
          지우거나 파일크기를 줄이면, 웹서버 프로세스가 다음
          번에 메모리대응한 파일내용을 읽을때 bus error가 발생할
          수 있다.</p>
        </li>
      </ul>

      <p>위의 조건에 해당하면 전송하는 파일을 메모리대응하지
      않도록 <code>EnableMMAP off</code>를 사용해야 한다. (주의:
      이 지시어는 디렉토리별로 변경할 수 있다.)</p>

    

    <h3>Sendfile</h3>

      

      <p>아파치는 운영체제가 <code>sendfile(2)</code>을 지원하면
      커널 sendfile을 사용하여 -- 예를 들어, 정적 파일을 서비스할때
      -- 전송할 파일을 직접 읽지않을 수 있다.</p>

      <p>여러 플래폼에서 sendfile을 사용하면 read와 send를 따로
      할 필요가 없어서 빨라진다. 그러나 sendfile을 사용하면
      웹서버의 안정성을 해치게되는 경우가 있다:</p>

      <ul>
        <li>
          <p>sendfile 지원이 잘못되었고 컴파일 시스템이 이점을
          발견하지 못하는 플래폼이 있다. 특히 다른 컴퓨터에서
          실행파일을 컴파일하여 sendfile 지원이 잘못된 컴퓨터로
          가져온 경우에 가능하다.</p>
        </li>
        <li>
          <p>커널은 자신의 캐쉬를 사용하여 NFS로 마운트한 파일을
          안정적으로 서비스할 수 없는 경우가 있다.</p>
        </li>
      </ul>

      <p>위의 조건에 해당하면 파일을 sendfile 전송하지 않도록
      <code>EnableSendfile off</code>를 사용해야 한다. (주의:
      이 지시어는 디렉토리별로 변경할 수 있다.)</p>

    

    <h3><a name="process" id="process">프로세스 생성</a></h3>

      

      <p>아파치 1.3 이전에는 <code class="directive"><a href="../mod/prefork.html#minspareservers">MinSpareServers</a></code>, <code class="directive"><a href="../mod/prefork.html#maxspareservers">MaxSpareServers</a></code>, <code class="directive"><a href="../mod/mpm_common.html#startservers">StartServers</a></code> 설정이 모두
      벤치마크 결과에 큰 영향을 미쳤다. 특히 아파치는 작업을
      서비스하기위해 충분한 자식수에 다다를 때까지 "도달" 기간이
      필요했다. 처음 <code class="directive"><a href="../mod/mpm_common.html#startservers">StartServers</a></code>개 자식을
      만든후, <code class="directive"><a href="../mod/prefork.html#minspareservers">MinSpareServers</a></code>
      설정값까지 초당 자식을 하나씩 만들었다. 그래서 <code class="directive"><a href="../mod/mpm_common.html#startservers">StartServers</a></code> 기본값이
      <code>5</code>인 서버에 클라이언트 100개가 동시에 접속하면
      부하를 처리하기에 충분한 자식을 만들기까지 95초가 걸렸다.
      자주 재시작하지 않는 실제 서버에서는 잘 동작하지만, 10분간만
      실행하는 벤치마크 결과는 매우 나쁘게 나온다.</p>

      <p>초당 한개 규칙은 자식을 새로 시작하면서 서버에 무리를
      주지 않으려고 정했다. 컴퓨터가 자식을 시작하느라 바쁘면
      요청을 서비스할 수 없다. 그러나 이 규칙이 아파치의 체감
      성능에 악영향을 주어 변경하였다. 아파치 1.3에서 초당 한개
      규칙은 완화되었다. 코드는 자식 한개를 만들고, 1초 쉬고,
      두개를 만들고, 1초 쉬고, 네개를 만들고, 이런 식으로 초당
      자식을 32개 만들때까지 지수로 증가한다. 자식수가 <code class="directive"><a href="../mod/prefork.html#minspareservers">MinSpareServers</a></code> 설정에 다다르면
      증가를 중단한다.</p>

      <p>이 경우 반응속도가 빨라져서 <code class="directive"><a href="../mod/prefork.html#minspareservers">MinSpareServers</a></code>, <code class="directive"><a href="../mod/prefork.html#maxspareservers">MaxSpareServers</a></code>, <code class="directive"><a href="../mod/mpm_common.html#startservers">StartServers</a></code>를 거의 설정할 필요가 없다. 일초에
      자식을 4개 이상 생성하면 <code class="directive"><a href="../mod/core.html#errorlog">ErrorLog</a></code>에 기록한다. 이런 오류문이
      많이 보이면 이 설정들을 조절하길 바란다.
      <code class="module"><a href="../mod/mod_status.html">mod_status</a></code> 결과가 도움이 될 것이다.</p>

    <p>프로세스 생성과 관련하여 <code class="directive"><a href="../mod/mpm_common.html#maxrequestsperchild">MaxRequestsPerChild</a></code> 설정은
    프로세스를 종료한다. 기본값은 자식당 처리할 요청수에 제한이
    없다는 <code>0</code>이다. 현재 설정이 <code>30</code>
    같이 매우 작은 값으로 설정되있다면, 값을 상당히 높힐 필요가
    있다. SunOS나 오래된 Solaris 버전을 사용한다면, 메모리유출때문에
    이 값을 <code>10000</code> 정도로 설정하라.</p>

    <p>연결유지(keep-alive)를 사용한다면 자식들은 이미 열린
    연결에서 추가 요청을 기다리며 아무것도 하지않기때문에 계속
    바쁘다. <code class="directive"><a href="../mod/core.html#keepalivetimeout">KeepAliveTimeout</a></code>
    기본값 <code>15</code> 초는 이런 현상을 최소화한다. 네트웍
    대역폭과 서버 자원 간의 균형이 맞게 설정한다. <a href="http://www.research.digital.com/wrl/techreports/abstracts/95.4.html">
    연결유지의 대부분의 이점이 사라지기때문에</a> 어떤 경우에도
    이 값을 <code>60</code> 초 이상으로 올리지 마라.</p>

    

  </div><div class="top"><a href="#page-header"><img alt="top" src="../images/up.gif" /></a></div>
<div class="section">
<h2><a name="compiletime" id="compiletime">컴파일시 설정에 대해서</a></h2>

    

    <h3>MPM 선택</h3>

      

      <p>아파치 2.x는 <a href="../mpm.html">다중처리모듈</a>
      (MPMs)이라는 교체할 수 있는 동기화 모델을 지원한다. 아파치를
      컴파일할때 MPM을 선택해야 한다. <code class="module"><a href="../mod/beos.html">beos</a></code>,
      <code class="module"><a href="../mod/mpm_netware.html">mpm_netware</a></code>, <code class="module"><a href="../mod/mpmt_os2.html">mpmt_os2</a></code>,
      <code class="module"><a href="../mod/mpm_winnt.html">mpm_winnt</a></code>와 같이 특정 플래폼에서만 사용할
      수 있는 MPM도 있다. 일반적인 유닉스류 시스템은 여러 MPM
      중에 하나를 선택할 수 있다. 웹서버의 속도와
      확장성(scalability)은 어떤 MPM을 선택했냐에 달렸다:</p>

      <ul>

        <li><code class="module"><a href="../mod/worker.html">worker</a></code> MPM은 여러 자식 프로세스가
        각각 여러 쓰레드를 사용한다. 각 쓰레드는 한번에 한 연결을
        담당한다. 일반적으로 worker는 prefork MPM 보다 적은
        메모리를 사용하므로 통신량이 많은 서버에 적절하다.</li>

        <li><code class="module"><a href="../mod/prefork.html">prefork</a></code> MPM은 쓰레드가 한개인 자식
        프로세스를 여러개 사용한다. 각 프로세스는 한번에 한
        연결을 담당한다. 여러 시스템에서 prefork의 속도는 worker와
        비슷하지만, 더 많은 메모리를 사용한다. 다음과 같은 상황에서
        쓰레드를 사용하지 않는 prefork 방식이 worker에 비해
        이점을 가진다: 쓰레드에 안전하지 (thread-safe) 않은
        제삼자가 만든 모듈을 사용할 수 있고, 쓰레드 디버깅 지원이
        빈약한 플래폼에서 쉽게 디버깅할 수 있다.</li>

      </ul>

      <p>이 MPM들과 다른 MPM에 대해 더 자세한 정보는 MPM <a href="../mpm.html">문서</a>를 참고하길 바란다.</p>

    

    <h3><a name="modules" id="modules">모듈</a></h3>

        

        <p>메모리 사용량이 성능에서 가장 중요한 요인이기때문에
        실제로 사용하지 않는 모듈을 제거해보자. 모듈을 <a href="../dso.html">DSO</a>로 컴파일했다면 간단히 그
        모듈에 대한 <code class="directive"><a href="../mod/mod_so.html#loadmodule">LoadModule</a></code> 지시어를 주석처리하면
        된다. 그래서 모듈을 제거하고 실행하여 사이트가 모듈없이도
        정상적으로 동작하는지 살펴볼 수 있다.</p>

        <p>반대로 모듈이 아파치 실행파일에 정적으로 링크되있다면
        원하지 않는 모듈을 제거하기위해 아파치를 재컴파일해야
        한다.</p>

        <p>여기서 당연히 어떤 모듈을 사용하고 사용하지 말지
        의문이 생긴다. 정답은 웹사이트마다 다르다. 그러나 아마도
        <em>최소한</em> <code class="module"><a href="../mod/mod_mime.html">mod_mime</a></code>,
        <code class="module"><a href="../mod/mod_dir.html">mod_dir</a></code>, <code class="module"><a href="../mod/mod_log_config.html">mod_log_config</a></code>
        모듈은 사용할 것이다. 물론 웹사이트에 로그파일이 필요없다면
        <code>mod_log_config</code>는 없어도 된다. 그러나 추천하지
        않는다.</p>

    

    <h3>Atomic 명령</h3>

      

      <p><code class="module"><a href="../mod/mod_cache.html">mod_cache</a></code> 같은 모듈과 최근 개발중인
      worker MPM은 APR의 atomic API를 사용한다. 이 API는 경량급
      쓰레드 동기화를 위할 atomic 명령을 제공한다.</p>

      <p>기본적으로 APR은 각 운영체제/CPU 플래폼에서 가장 효율적인
      방법을 사용하여 이 명령을 구현한다. 예를 들어, 여러 최신
      CPU에는 하드웨어로 atomic compare-and-swap (CAS) 연산을
      하는 명령어가 있다. 그러나 어떤 플래폼에서 APR은 이런
      명령어가 없는 오래된 CPU와 호환성을 위해 더 느린 mutex기반
      구현을 기본적으로 사용한다. 이런 플래폼에서 아파치를
      컴파일할때 아파치를 최신 CPU에서만 실행할 계획이라면,
      아파치를 구성할때 <code>--enable-nonportable-atomics</code>
      옵션을 사용하여 더 빠른 atomic 구현을 선택할 수 있다:</p>

      <div class="example"><p><code>
        ./buildconf<br />
        ./configure --with-mpm=worker --enable-nonportable-atomics=yes
      </code></p></div>

      <p><code>--enable-nonportable-atomics</code> 옵션은 다음과
      같은 플래폼에 영향이 있다:</p>

      <ul>

        <li>SPARC에서 Solaris<br />
            기본적으로 APR은 Solaris/SPARC에서 mutex기반 atomic을
            사용한다. 그러나 구성할때
            <code>--enable-nonportable-atomics</code>를 사용하면
            APR은 빠른 하드웨어 compare-and-swap을 위한 SPARC
            v8plus 명령어를 사용한다. 이 옵션을 사용하면 atomic
            명령이 더 효율적이지만 (CPU를 덜 사용하고 더 높은
            동기화가 가능하다), 컴파일한 실행파일은 UltraSPARC
            칩에서만 실행할 수 있다.
        </li>

        <li>Linux on x86<br />
            기본적으로 APR은 리눅스에서 mutex기반 atomic을
            사용한다. 그러나 구성할때
            <code>--enable-nonportable-atomics</code>를 사용하면
            APR은 빠른 하드웨어 compare-and-swap을 위한 486
            명령어를 사용한다. 더 효율적인 atomic 명령이 가능하지만,
            컴파일한 실행파일은 486 이상 칩에서만 (386은 안된다)
            실행할 수 있다.
        </li>

      </ul>

    

    <h3>mod_status와 ExtendedStatus On</h3>

      

      <p>아파치를 컴파일할때 <code class="module"><a href="../mod/mod_status.html">mod_status</a></code>를 포함하고
      실행할때 <code>ExtendedStatus On</code>을 설정하면 아파치는
      요청을 받을때마다 <code>gettimeofday(2)</code>(혹은 운영체제에
      따라 <code>times(2)</code>)를 두번 호출하고 (1.3 이전에는)
      <code>time(2)</code>도 추가로 여러번 호출한다. 상태 보고서에
      동작시간이 필요하기 때문이다. 최상의 성능을 얻으려면
      (기본값인) <code>ExtendedStatus off</code>를 설정한다.</p>

    

    <h3>accept 직렬화 - 여러 소켓</h3>

      

    <div class="warning"><h3>주의:</h3>
      <p> 아래 문서는 아파치 웹서버 2.0 버전에서 변경된 내용을
      담고 있지 않다. 아직도 유효한 정보가 있지만, 주의해서
      사용하길 바란다.</p>
    </div>

      <p>유닉스 소켓 API의 단점을 설명한다. 웹서버가 여러 포트
      혹은 여러 주소를 기다리기위해 여러 <code class="directive"><a href="../mod/mpm_common.html#listen">Listen</a></code>을 사용한다고 가정하자.
      연결이 가능한지 각 소켓을 검사하기위해 아파치는
      <code>select(2)</code>를 사용한다. <code>select(2)</code>
      소켓에 기다리고 있는 연결이 <em>없는지</em> 혹은 <em>최소한
      한개</em> 있는지 알려준다. 아파치에는 여러 자식이 있고,
      쉬고 있는 모든 자식은 동시에 새로운 연결을 검사한다. 원래
      구현은 다음과 비슷하다 (이 예는 코드에서 가져오지 않았다.
      단지 설명하기위한 용도로 만들었다.):</p>

      <div class="example"><p><code>
        for (;;) {<br />
        <span class="indent">
          for (;;) {<br />
          <span class="indent">
            fd_set accept_fds;<br />
            <br />
            FD_ZERO (&amp;accept_fds);<br />
            for (i = first_socket; i &lt;= last_socket; ++i) {<br />
            <span class="indent">
              FD_SET (i, &amp;accept_fds);<br />
            </span>
            }<br />
            rc = select (last_socket+1, &amp;accept_fds, NULL, NULL, NULL);<br />
            if (rc &lt; 1) continue;<br />
            new_connection = -1;<br />
            for (i = first_socket; i &lt;= last_socket; ++i) {<br />
            <span class="indent">
              if (FD_ISSET (i, &amp;accept_fds)) {<br />
              <span class="indent">
                new_connection = accept (i, NULL, NULL);<br />
                if (new_connection != -1) break;<br />
              </span>
              }<br />
            </span>
            }<br />
            if (new_connection != -1) break;<br />
          </span>
          }<br />
          process the new_connection;<br />
        </span>
        }
      </code></p></div>

      <p>그러나 위의 단순한 구현에는 심각한 고갈(starvation)
      문제가 있다. 여러 자식이 동시에 이 반복문을 실행하면,
      요청을 기다리며 모두 <code>select</code>에서 멈춘다. 이때
      어떤 소켓에 요청이 하나라도 들어오면 모든 자식이 깨어난다
      (깨어나는 자식의 개수는 운영체제와 타이밍에 따라 다르다).
      이들은 모두 연결을 <code>accept</code>하길 시도한다. 그러나
      (아직도 한 연결만 대기중이라면) 한 자식만 성공하고, 나머지는
      <code>accept</code>에서 <em>멈춘다.</em> 그러면 이 자식들은
      한 소켓의 요청만을 서비스하도록 묶여서, 그 소켓으로 새로운
      요청이 충분히 들어와서 모든 자식을 깨울때까지 정지해있다.
      이런 고갈 문제는 <a href="http://bugs.apache.org/index/full/467">PR#467</a>
      처음 보고되었다. 최소한 두가지 해결책이 있다.</p>

      <p>한가지는 소켓을 대기하지 않도록 (non-blocking) 만드는
      방법이다. 이 경우 자식이 <code>accept</code>를 해도 멈추지
      않고, 즉시 진행할 수 있다. 그러나 CPU 시간을 낭비한다.
      <code>select</code>에서 쉬는 자식이 10개 있고, 새로 연결이
      한개 들어왔다고 가정하자. 그러면 이 자식중 9개는 깨어나서
      연결을 <code>accept</code>하길 시도하고 실패하면 아무
      일도 하지 않고 다시 <code>select</code>를 반복한다. 다시
      <code>select</code>로 돌아올 때까지 어떤 자식도 다른 소켓에
      들어온 요청을 서비스하지 않는다. (다중프로세서 컴퓨터에서)
      쉬는 자식 개수만큼 CPU 개수가 있는 드문 경우가 아니라면
      이 해결책은 별로 좋아보이지 않는다.</p>

      <p>다른 방법은 아파치가 사용하는 방법으로 내부 반복문에
      한 자식만을 들여보낸다. 반복문은 다음과 같다 (차이를
      강조했음):</p>

      <div class="example"><p><code>
        for (;;) {<br />
        <span class="indent">
          <strong>accept_mutex_on ();</strong><br />
          for (;;) {<br />
          <span class="indent">
            fd_set accept_fds;<br />
            <br />
            FD_ZERO (&amp;accept_fds);<br />
            for (i = first_socket; i &lt;= last_socket; ++i) {<br />
            <span class="indent">
              FD_SET (i, &amp;accept_fds);<br />
            </span>
            }<br />
            rc = select (last_socket+1, &amp;accept_fds, NULL, NULL, NULL);<br />
            if (rc &lt; 1) continue;<br />
            new_connection = -1;<br />
            for (i = first_socket; i &lt;= last_socket; ++i) {<br />
            <span class="indent">
              if (FD_ISSET (i, &amp;accept_fds)) {<br />
              <span class="indent">
                new_connection = accept (i, NULL, NULL);<br />
                if (new_connection != -1) break;<br />
              </span>
              }<br />
            </span>
            }<br />
            if (new_connection != -1) break;<br />
          </span>
          }<br />
          <strong>accept_mutex_off ();</strong><br />
          process the new_connection;<br />
        </span>
        }
      </code></p></div>

      <p><code>accept_mutex_on</code><code>accept_mutex_off</code>
      <a id="serialize" name="serialize">함수</a>는 mutex 세마포어를
      구현한다. 한번에 오직 한 자식만이 mutex를 가질 수 있다.
      mutex를 구현하는 방법은 여러가지이다. 구현 방법은 (1.3
      이전) <code>src/conf.h</code>나 (1.3과 그 이후)
      <code>src/include/ap_config.h</code>에 정의되있다. 어떤
      아키텍쳐는 잠금(locking) 방법을 선택하지 않기때문에, 이런
      아키텍쳐에서 여러 <code class="directive"><a href="../mod/mpm_common.html#listen">Listen</a></code> 지시어를 사용하면
      위험하다.</p>

      <p>실행시 <code class="directive"><a href="../mod/mpm_common.html#acceptmutex">AcceptMutex</a></code> 지시어를 사용하여
      mutex 구현을 변경할 수 있다.</p>

      <dl>
        <dt><code>AcceptMutex flock</code></dt>

        <dd>
          <p>이 방법은 잠금파일을 잠그기위해 <code>flock(2)</code>
          시스템호출을 사용한다 (잠금파일 위치는 <code class="directive"><a href="../mod/mpm_common.html#lockfile">LockFile</a></code> 지시어로 지정).</p>
        </dd>

        <dt><code>AcceptMutex fcntl</code></dt>

        <dd>
          <p>이 방법은 잠금파일을 잠그기위해 <code>fcntl(2)</code>
          시스템호출을 사용한다 (잠금파일 위치는 <code class="directive"><a href="../mod/mpm_common.html#lockfile">LockFile</a></code> 지시어로 지정).</p>
        </dd>

        <dt><code>AcceptMutex sysvsem</code></dt>

        <dd>
          <p>(1.3과 그 이후) 이 방법을 SysV식 세마포어를 사용하여
          mutex를 구현한다. 불행히도 SysV식 세마포어는 나쁜
          부작용이 있다. 하나는 아파치가 세마포어를 정리하지
          않고 죽을 수 있는 점이다 (<code>ipcs(8)</code> manpage
          참고). 다른 하나는 웹서버와 동일한 uid로 실행하는
          CGI가 (<em>즉,</em> <code>suexec</code>
          <code>cgiwrapper</code>를 사용하지않는 한 모든 CGI)
          세마포어 API를 사용하여 서비스거부공격을 할 수 있는
          점이다. 이런 이유때문에 IRIX를 제외한 아키텍쳐에서
          이 방법을 사용하지 않는다 (대부분의 IRIX 컴퓨터에서
          앞의 두 방법은 지나치게 버겁다).</p>
        </dd>

        <dt><code>AcceptMutex pthread</code></dt>

        <dd>
          <p>(1.3과 그 이후) 이 방법은 POSIX mutex를 사용하기때문에
          POSIX 쓰레드 규약을 완전히 구현한 아키텍쳐라면 모두
          사용가능하지만, (2.5 이후) Solaris에서만 그것도 특정
          구성에서만 동작하는 듯하다. 이 방법을 시도해본다면
          서버가 멈춰서 응답을 안하는지 살펴봐야 한다. 정적
          내용만 서비스하는 서버는 잘 동작하는 것 같다.</p>
        </dd>

        <dt><code>AcceptMutex posixsem</code></dt>

        <dd>
          <p>(2.0과 그 이후) 이 방법은 POSIX 세마포어를 사용한다.
          mutex를 가진 프로세스의 쓰레드가 죽는다면(segfault)
          세마포어 소유권이 회복되지 않아서 웹서버가 멈춘다.</p>
        </dd>

      </dl>

      <p>시스템에 위 목록에 없는 직렬화(serialization) 방법이
      있다면 그 방법을 사용하는 코드를 APR에 추가할 가치가 있다.</p>

      <p>고려는 해봤지만 구현하지않은 다른 방법은 부분적으로
      반복문을 직렬화하는 방법이다. 즉, 프로세서를 몇개만 들여보내는
      것이다. 이 방법은 여러 자식을 동시에 실행할 수 있어서
      직렬화때문에 전체 대역폭을 활용하지 못하는 다중프로세서
      컴퓨터에서만 관심을 가져볼 수 있다. 앞으로 살펴볼 부분이지만,
      매우 병렬화된 웹서버가 흔하지 않아서 우선순위가 낮다.</p>

      <p>최상의 성능을 얻기위해서는 여러 <code class="directive"><a href="../mod/mpm_common.html#listen">Listen</a></code> 문을 사용하지 않는
      것이 이상적이다. 그러나 계속 설명한다.</p>

    

    <h3>accept 직렬화 - 소켓 한개</h3>

      

      <p>앞의 설명은 다중소켓 서버에는 좋지만, 소켓이 한개인
      서버는 어떤가? 연결이 도착할때까지 모든 자식이
      <code>accept(2)</code>에서 멈춰있기때문에 이론상 같은
      문제가 발생하지 않고, 고갈 문제도 없다. 그러나 실제로는
      앞에서 말한 대기하지 않는 (non-blocking) 방법에서 발생하는
      "공회전(spinning)" 현상을 감추고 있다. 대부분의 TCP 스택은
      연결이 도착하면 커널이 <code>accept</code>에서 멈춰있는
      모든 자식을 깨우도록 구현되있다. 프로세스중 한개가 연결을
      얻고 사용자영역으로 돌아가고, 나머지는 커널에서 공회전하여
      연결이 없음을 발견하면 다시 잠을 잔다. 사용자영역 코드에서는
      이런 공회전을 알 수 없지만, 분명히 존재한다. 그래서 다중소켓의
      대기하지 않는 방법과 동일하게 부하를 높이는 불필요한 행동이
      일어난다.</p>

      <p>그래서 우리는 여러 아키텍쳐에서 소켓이 한개인 경우에도
      직렬화하면 더 "잘" 동작함을 발견했다. 그래서 거의 대부분의
      경우 기본적으로 직렬화를 사용한다. 리눅스에서 (커널 2.0.30,
      128Mb 메모리에 듀얼 Pentium pro) 실험한 결과 소켓 한개를
      직렬화하면 하지 않은 경우에 비해 초당 요청이 3% 미만
      줄어들었다. 그러나 직렬화를 하지 않은 경우 요청당 100ms
      지연이 발생했다. 이 지연은 아마도 LAN에서 발생하는 긴
      연결선때문일 것이다. 소켓이 한개인 경우 직렬화를 사용하지
      않으려면 <code>SINGLE_LISTEN_UNSERIALIZED_ACCEPT</code>
      정의한다.</p>

    

    <h3>Close 지연(lingering)</h3>

      

      <p><a href="http://www.ics.uci.edu/pub/ietf/http/draft-ietf-http-connection-00.txt">
      draft-ietf-http-connection-00.txt</a> 8절에서 설명하듯이
      <strong>안정적인</strong> 웹서버가 되려면, 통신의 양 방향을
      독립적으로 닫을 수 있어야 한다 (TCP 연결은 쌍방향이고,
      방향은 서로 독립적이다). 이점을 다른 서버에서는 자주
      간과하지만, 아파치는 1.2부터 정확히 구현해왔다.</p>

      <p>이 기능을 부주의하게 아파치에 추가했을때 여러 유닉스
      버전에서 많은 문제가 발생했다. TCP 규약은
      <code>FIN_WAIT_2</code>에 타임아웃이 있다고 정하지 않았지만,
      금지하지도 않았다. 타임아웃이 없는 시스템에서 아파치 1.2는
      많은 소켓을 영원히 <code>FIN_WAIT_2</code> 상태로 만들었다.
      많은 경우 이 문제는 제작사가 제공하는 최신 TCP/IP 패치를
      적용하여 해결할 수 있다. 그러나 제작사가 패치를 발표하지
      않는 경우가 (<em>즉,</em> SunOS4 -- 소스 라이선스가 있는
      사람은 직접 패치할 수 있지만) 있기때문에 이 기능을 사용하지
      않기로 결정했다.</p>

      <p>방법은 두가지다. 하나는 소켓 옵션 <code>SO_LINGER</code>
      사용하는 방법이다. 그러나 불행히도 대부분의 TCP/IP 스택은
      이 옵션을 올바로 구현하지 않았다. 올바로 구현한 스택에서
      조차도 (<em>즉,</em> 리눅스 2.0.31) 이 방법은 다음 방법보다
      더 cpu를 잡아먹는다.</p>

      <p>아파치는 보통 (<code>http_main.c</code>에 있는)
      <code>lingering_close</code>라는 함수를 사용한다. 이 함수는
      대충 다음과 같다:</p>

      <div class="example"><p><code>
        void lingering_close (int s)<br />
        {<br />
        <span class="indent">
          char junk_buffer[2048];<br />
          <br />
          /* shutdown the sending side */<br />
          shutdown (s, 1);<br />
          <br />
          signal (SIGALRM, lingering_death);<br />
          alarm (30);<br />
          <br />
          for (;;) {<br />
          <span class="indent">
            select (s for reading, 2 second timeout);<br />
            if (error) break;<br />
            if (s is ready for reading) {<br />
            <span class="indent">
              if (read (s, junk_buffer, sizeof (junk_buffer)) &lt;= 0) {<br />
              <span class="indent">
                break;<br />
              </span>
              }<br />
              /* just toss away whatever is here */<br />
            </span>
            }<br />
          </span>
          }<br />
          <br />
          close (s);<br />
        </span>
        }
      </code></p></div>

      <p>이 코드는 연결을 닫을때 더 CPU를 사용하지만, 안정적인
      구현을 위해 필요하다. HTTP/1.1이 더 널리 퍼지고 모든 연결을
      유지한다면(persistent), 연결을 받는 비용은 여러 요청을
      처리하면서 상쇄될 것이다. 위험하게도
      <code>NO_LINGCLOSE</code>를 정의하여 이 기능을 사용하지
      않을 수 있지만, 절대로 권하지 않는다. 특히 HTTP/1.1
      파이프라인 <span class="transnote">(<em>역주;</em> 연결유지 상태에서 응답을 기다리지
      않고 여러 요청을 보내는 기술)</span> 연결유지에는
      <code>lingering_close</code>가 필수적이다 (그리고 <a href="http://www.w3.org/Protocols/HTTP/Performance/Pipeline.html">
      파이프라인 연결이 더 빠르기때문에</a> 사용하길 바랄 것이다).</p>

    

    <h3>Scoreboard 파일</h3>

      

      <p>아파치의 부모와 자식은 scoreboard라는 것을 통해 서로
      통신한다. 이상적으로는 scoreboard를 공유메모리로 구현해야
      한다. 우리 개발자가 해당 운영체제에 접근할 수 있거나 상세한
      포팅 결과를 받은 경우 보통 공유메모리를 사용하여 구현한다.
      나머지는 디스크에 있는 파일을 사용하여 구현한다. 디스크에
      있는 파일은 느리고 신뢰도가 떨어진다 (기능도 더 적다).
      <code>src/main/conf.h</code> 파일에서 사용하는 아키텍쳐를
      찾아서 <code>USE_MMAP_SCOREBOARD</code> 혹은
      <code>USE_SHMGET_SCOREBOARD</code>인지 확인한다. 둘중
      하나를 (각각 함께 사용할 <code>HAVE_MMAP</code>이나
      <code>HAVE_SHMGET</code>도 같이) 정의하면 공유메모리 코드를
      사용한다. 시스템이 다른 종류의 공유메모리를 사용한다면
      <code>src/main/http_main.c</code> 파일을 수정하여 아파치에서
      공유메모리를 사용할 수 있도록 훅(hook)을 추가하라. (또한
      패치를 우리에게 보내주길 바란다.)</p>

      <div class="note">역사적 설명: 아파치의 리눅스 버전은 아파치 1.2 버전부터
      공유메모리를 사용하기 시작했다. 리눅스에서 초기 아파치
      버전이 느리고 신뢰도가 떨어졌기 때문이다.</div>

    

    <h3>DYNAMIC_MODULE_LIMIT</h3>

      

      <p>모듈을 동적으로 읽어들이지 않는다면 (가능한 조금이라도
      성능을 높이기위해 이 글을 읽는다면 아마도 모듈을 동적으로
      읽어들이지 않을 것이다), 서버를 컴파일할때
      <code>-DDYNAMIC_MODULE_LIMIT=0</code>을 추가한다. 그러면
      모듈을 동적으로 읽어들이기위해 할당하는 메모리를 절약한다.</p>

    

  </div><div class="top"><a href="#page-header"><img alt="top" src="../images/up.gif" /></a></div>
<div class="section">
<h2><a name="trace" id="trace">부록: 시스템호출 기록을 자세히 분석하기</a></h2>

    

    <p>다음은 Solaris 8에서 worker MPM을 사용한 아파치 2.0.38의
    시스템호출 기록(trace)이다. 아래 명령어를 사용하여 기록을
    얻었다:</p>

    <div class="example"><p><code>
      truss -l -p <var>httpd_child_pid</var>.
    </code></p></div>

    <p><code>-l</code> 옵션을 사용하면 truss는 시스템호출을
    하는 LWP (lightweight process, 경량급 프로세스--Solaris의
    커널수준 쓰레드) ID를 같이 기록한다.</p>

    <p>다른 시스템에는 <code>strace</code>, <code>ktrace</code>,
    <code>par</code> 같은 시스템호출 추적 도구가 있다. 결과는
    비슷하다.</p>

    <p>클라이언트는 웹서버에게 크기가 10KB인 정적 파일을 요청한다.
    정적인 파일을 요청하지 않거나 내용협상하는 요청을 한 경우
    기록이 매우 다르다 (때로는 매우 알아보기 힘들다).</p>

    <div class="example"><pre>/67:    accept(3, 0x00200BEC, 0x00200C0C, 1) (sleeping...)
/67:    accept(3, 0x00200BEC, 0x00200C0C, 1)            = 9</pre></div>

    <p>위에서 연결대기(listener) 쓰레드가 LWP #67에서 실행됨을
    알 수 있다.</p>

    <div class="note"><code>accept(2)</code> 직렬화를 사용하지 않음을 주목하라.
    여러 포트를 기다리지않는 경우 이 플래폼의 worker MPM은
    기본적으로 직렬화하지 않은 accept를 사용한다.</div>

    <div class="example"><pre>/65:    lwp_park(0x00000000, 0)                         = 0
/67:    lwp_unpark(65, 1)                               = 0</pre></div>

    <p>연결은 받아들이고(accept) 연결대기 쓰레드는
    worker 쓰레드를 깨워서 요청을 처리하게 한다. 아래 기록에서
    요청을 처리하는 worker 쓰레드가 LWP #65임을 알 수 있다.</p>

    <div class="example"><pre>/65:    getsockname(9, 0x00200BA4, 0x00200BC4, 1)       = 0</pre></div>

    <p>가상호스트를 구현하기위해 아파치는 연결을 받아들인
    지역(local) 소켓 주소를 알아야 한다. (가상호스트를 사용하지
    않거나 <code class="directive"><a href="../mod/mpm_common.html#listen">Listen</a></code>
    지시어에 와일드카드 주소를 사용하지 않은 경우 등) 많은 경우
    이 호출을 없앨 수 있다. 그러나 아직 이런 최적화 작업이
    안되있다. </p>

    <div class="example"><pre>/65:    brk(0x002170E8)                                 = 0
/65:    brk(0x002190E8)                                 = 0</pre></div>

    <p><code>brk(2)</code> 호출은 힙(heap)에서 메모리를 할당한다.
    웹서버는 대부분의 요청 처리시 자체 메모리
    할당자(<code>apr_pool</code><code>apr_bucket_alloc</code>)를
    사용하기때문에 시스템호출 기록에서 이 시스템호출을 보기가
    드물다. 이 기록에서 웹서버는 시작하자마자 자체 메모리 할당자가
    사용할 메모리블록을 얻기위해 <code>malloc(3)</code>을 호출한다.</p>

    <div class="example"><pre>/65:    fcntl(9, F_GETFL, 0x00000000)                   = 2
/65:    fstat64(9, 0xFAF7B818)                          = 0
/65:    getsockopt(9, 65535, 8192, 0xFAF7B918, 0xFAF7B910, 2190656) = 0
/65:    fstat64(9, 0xFAF7B818)                          = 0
/65:    getsockopt(9, 65535, 8192, 0xFAF7B918, 0xFAF7B914, 2190656) = 0
/65:    setsockopt(9, 65535, 8192, 0xFAF7B918, 4, 2190656) = 0
/65:    fcntl(9, F_SETFL, 0x00000082)                   = 0</pre></div>

    <p>다음 worker 쓰레드는 클라이언트의 연결(파일기술자 9)을
    대기안함(non-blocking) 상태로 바꾼다. <code>setsockopt(2)</code>
    <code>getsockopt(2)</code> 호출은 Solaris의 libc가 소켓에
    대한 <code>fcntl(2)</code>을 어떻게 처리하는지 보여준다.</p>

    <div class="example"><pre>/65:    read(9, " G E T   / 1 0 k . h t m".., 8000)     = 97</pre></div>

    <p>worker 쓰레드는 클라이언트로 부터 요청을 읽는다.</p>

    <div class="example"><pre>/65:    stat("/var/httpd/apache/httpd-8999/htdocs/10k.html", 0xFAF7B978) = 0
/65:    open("/var/httpd/apache/httpd-8999/htdocs/10k.html", O_RDONLY) = 10</pre></div>

    <p>웹서버 설정은 <code>Options FollowSymLinks</code>
    <code>AllowOverride None</code>이다. 그래서 요청한 파일경로의
    각 디렉토리에 대해 <code>lstat(2)</code>하거나
    <code>.htaccess</code> 파일을 검사할 필요가 없다. 파일을
    검사하기위해, 1) 파일이 있는지, 2) 디렉토리가 아닌 일반파일인지,
    <code>stat(2)</code> 호출만 하면 된다.</p>

    <div class="example"><pre>/65:    sendfilev(0, 9, 0x00200F90, 2, 0xFAF7B53C)      = 10269</pre></div>

    <p>이 경우 웹서버는 한번의 <code>sendfilev(2)</code> 시스템호출로
    HTTP 응답헤더와 요청한 파일을 전송할 수 있다. Sendfile 지원여부는
    운영체제마다 다르다. 다른 시스템이라면 <code>sendfile(2)</code>
    호출하기 전에 헤더를 보내기위해 <code>write(2)</code>
    <code>writev(2)</code> 호출을 한다.</p>

    <div class="example"><pre>/65:    write(4, " 1 2 7 . 0 . 0 . 1   -  ".., 78)      = 78</pre></div>

    <p><code>write(2)</code> 호출은 접근로그(access log)에 요청을
    기록한다. 이 기록에 <code>time(2)</code> 호출이 없음을 주목하라.
    아파치 1.3과 달리 아파치 2.0은 시간을 알기위해
    <code>gettimeofday(3)</code>를 사용한다.
    <code>gettimeofday</code>를 최적화한 리눅스와 Solaris 같은
    몇몇 운영체제에서는 일반적인 시스템호출 부담이 없다.</p>

    <div class="example"><pre>/65:    shutdown(9, 1, 1)                               = 0
/65:    poll(0xFAF7B980, 1, 2000)                       = 1
/65:    read(9, 0xFAF7BC20, 512)                        = 0
/65:    close(9)                                        = 0</pre></div>

    <p>worker 쓰레드는 연결을 지연닫기(lingering close)한다.</p>

    <div class="example"><pre>/65:    close(10)                                       = 0
/65:    lwp_park(0x00000000, 0)         (sleeping...)</pre></div>

    <p>마지막으로 worker 쓰레드는 방금 전송한 파일을 닫고,
    연결대기(listener) 쓰레드가 다른 연결을 할당할 때까지
    정지한다.</p>

    <div class="example"><pre>/67:    accept(3, 0x001FEB74, 0x001FEB94, 1) (sleeping...)</pre></div>

    <p>그동안 연결대기 쓰레드는 연결을 (모든 worker가 작업중이면
    연결대기 쓰레드를 멈추는 worker MPM의 흐름제어 기능에 따라)
    worker 쓰레드에 할당하자마자 다른 연결을 받아들일 수 있다.
    이 기록에는 나오지 않지만, worker 쓰레드가 방금 받은 연결을
    처리하는 동안 다음 <code>accept(2)</code>가 (요청이 매우
    많은 경우 항상) 일어날 수 있다.</p>

  </div></div>
<div class="bottomlang">
<p><span>가능한 언어: </span><a href="../en/misc/perf-tuning.html" hreflang="en" rel="alternate" title="English">&nbsp;en&nbsp;</a> |
<a href="../fr/misc/perf-tuning.html" hreflang="fr" rel="alternate" title="Fran&#231;ais">&nbsp;fr&nbsp;</a> |
<a href="../ko/misc/perf-tuning.html" title="Korean">&nbsp;ko&nbsp;</a> |
<a href="../tr/misc/perf-tuning.html" hreflang="tr" rel="alternate" title="T&#252;rk&#231;e">&nbsp;tr&nbsp;</a></p>
</div><div class="top"><a href="#page-header"><img src="../images/up.gif" alt="top" /></a></div><div class="section"><h2><a id="comments_section" name="comments_section">Comments</a></h2><div class="warning"><strong>Notice:</strong><br />This is not a Q&amp;A section. Comments placed here should be pointed towards suggestions on improving the documentation or server, and may be removed again by our moderators if they are either implemented or considered invalid/off-topic. Questions on how to manage the Apache HTTP Server should be directed at either our IRC channel, #httpd, on Freenode, or sent to our <a href="http://httpd.apache.org/lists.html">mailing lists</a>.</div>
<script type="text/javascript"><!--//--><![CDATA[//><!--
var comments_shortname = 'httpd';
var comments_identifier = 'http://httpd.apache.org/docs/2.4/misc/perf-tuning.html';
(function(w, d) {
    if (w.location.hostname.toLowerCase() == "httpd.apache.org") {
        d.write('<div id="comments_thread"><\/div>');
        var s = d.createElement('script');
        s.type = 'text/javascript';
        s.async = true;
        s.src = 'https://comments.apache.org/show_comments.lua?site=' + comments_shortname + '&page=' + comments_identifier;
        (d.getElementsByTagName('head')[0] || d.getElementsByTagName('body')[0]).appendChild(s);
    }
    else { 
        d.write('<div id="comments_thread">Comments are disabled for this page at the moment.<\/div>');
    }
})(window, document);
//--><!]]></script></div><div id="footer">
<p class="apache">Copyright 2017 The Apache Software Foundation.<br />Licensed under the <a href="http://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a>.</p>