Develop

[php] 배열 관련 함수 설명 ㅎㅎ

by hooni posted Apr 23, 2003
?

단축키

Prev이전 문서

Next다음 문서

ESC닫기

크게 작게 위로 아래로 댓글로 가기 인쇄
php는 배열이 정말로 좋다... 신기하게도 모든 자료형과 객체들을 다 넣을 수 있다..

예전에 C를 배울때는 배열이 좋구나 라는 것을 느끼면서 하나씩 배우기는 시작했는데... C에서의 배열은 일단 틀(자료형과 개수)을 정해야한다는 단점이 있다.

그래서 여러가지 동적인 자료구조를 이용하지만 설정하는데 여러가지 번거로움이 존재한다.

자바에서는 그나마 배열이 동적으로 바뀌는 자료구조를 여러가지 채택하고 있지만... 연관배열 같은 구조는 바로 바로 쓰기 어렵다. 꼭 HashMap 이나 기타 다른 키값을 가지고 있는 클래스를 사용하여야 한다.

php는 스크립트 언어이다. 대부분의 스크립트 언어들이 유연한 자료구조로서 배열을 사용하는데 특히나 php는 따른 설정 없이 모든 자료형이 배열에 들어갈 수가 있다.

그래서 다른 언어에 있는 자료구조 형식을 따로 만들어 줄 필요가 없다... 그래서 php 하는 사람들이 자료구조에 약한가.. 흠.. ^^;;

모든 것을 담아라. 
      array 형은 모든 것을 담을 수 있다.
      $arr = array(1, '안녕', 234.566, new Object(), array(1,2,3) );

     여러가지 자료형을 동시에 담을 수 있다.
     연관배열은 기본...   $arr = array('aaa' => 'varchar');   와 같은 방식으로 바로 바로 설정이 된다.


2. 특별한 자료 구조를 필요로 하지 않는다.
     php에서는 배열은 동적으로 늘어나고 줄어들기 때문에 동적 자료구조를 만들 필요가 없다. 그리고 일반 자료구조처럼 조작할 수 있는 여러가지 함수들을 제공한다.

    1) statck 형태로 사용하기
        스택은 가장 나중에 들어간 데이타가 가장 먼저 나오는 구조이다. (LIFO)
        pop, push    스택의 가장 기본적인 2가지 형태의 조작 방법이다.
        pop은 가장 뒤에 있는 요소를 리턴해주고 배열에서 지워주는 것이고 push는 가장 뒤에 요소를 집어넣는다.  

       php는  array_pop($arr), array_push($arr, 데이타)   등으로 스택을 구현할 수 있다.
       array_push 같은 경우는 언어 문법 상으로도 지원을 한다.   $arr 이 배열일 경우  $arr[] = 데이타;  와 같은 형식은 array_push 와 같은 역할을 한다.


     2 ) Queue 형태로 사용하기
          Queue는 가장 먼저 들어간 데이타가 가장 먼저 나오는 구조 (FIFO)  이다.
          그래서 큐는 기본적으로 push와 enqueue의 2가지 방법을 가진다. push는 데이타를 넣는 것이고 enqueue 는 젤 앞 요소를 리턴해주고 배열에서 삭제해준다.

          php는  array_push($arr, 데이타), array_shift($arr) 등으로 구현한다.


      3)  Dequeue 형태로 사용하기
           Dequeue 는 앞뒤 쪽으로 어느 곳으로 나 데이타를 빼고 넣을 수 있는 자료구조이다. stack과 queue를 잘 조합했다고 보시면 된다.

앞으로 넣을때는   array_unshift($arr, 데이타),  뒤로 넣을 때는 array_push($arr, 데이타)
앞에서 자료를 뺄 때는    array_shift($arr), 뒤로 자료를 뺄 때는 array_pop($arr)


   4) 연결 리스트 (Linked List)  처럼 사용하기 
       연결 리스트라는 것은 배열의 단점을 보완하기 위해서 나온 것이다.
       배열은 기본적으로 자료형과 크기가 정해져있다.
       그래서 arr[100] 와 같이 100개의 요소를 가지는 배열을 선언해놓고 10개만 사용하면 메모리 사용측면에서 아주 안 좋게 된다.
       이렇듯 메모리 사용을 좀 줄이고 유연한 배열 구조를 만들어 보고자 나온 것이 연결리스트이다. C에서는 포인터, java에서는 참조를 사용하지만 

       php에서는 애초에 동적으로 배열이 변하기 때문에 php의 배열 자체가 연결리스트와 비슷 하다고 보시면 된다.

         연결리스트의 가장 좋은 점이 참조만으로 이루어진 자료형이라서 삽입과 삭제에 비용이 거의 들지 않는 다는 점이다.
         php의 배열의 경우는 삭제는 쉽지만  중간에 삽입하는 것은 다른언어와 비슷한 절차를 가져야 한다.
        삭제하는 방법은 unset($arr[10]) 과 같은 방법으로 요소를 메모리에서 지우면 된다.


      5) Hash
          Hash 라는 것은 고유키를 만들어서 그 키와 데이타를 일대일로 매칭 시키는 기법이다. 일반적으로 사전과 비슷하다고 보시면 된다.
          php는 배열 자체적으로 연관배열이라고 하는 것을 지원한다.

         $arr['key'] = 'value';       echo $arr['key'];

         아주 간단하게 hash 구조로 셋팅이 가능하다.  배열이 자체적으로 지원을 하기 때문이다. 스크립트 언어들이 대부분 이런 구조를 지원하지만
         vb 스크립트는 지원하지 않아서 asp 코딩할때 약간의 생각을 더 해야한다.

         ※ 기타 다른 구조들도 많지만 지금은 생각이 안나서 잠시 접어두도록 하겠다.


3. 배열 제어 방법
    이제는 실제 php에서 배열을 제어 하는 방법을 알아보자..

    1) Iterator 형식
       php는 foreach라는 키워드를 이용해서 배열 요소를 바로 접근할 수 있게 해준다.
       foreach ($arr as $value) { echo $value ; }

       키값을 가지고 있는 배열도 바로 사용 할 수 있다. 어떻게? ㅋ 바로 이렇게
       foreach ($arr as $key => $value) { echo $key, $value; }

       php5 들어오면서 재미난 기능이 하나 더 생겼다. php4에서는 foreach에서 나오는 배열 값을 실제 배열 요소로서 조작이 불가능했다. 하지만 php5에서는 가능하다.
       foreach ($ass as &$value) { $value = $value.'aaa'; }

      값을 참조 형태로 받을 수 있기 때문에 그 배열 요소에 있는 값들을 바로 수정하는 것도 가능하다. 재밌는 기능이다.


     2) list, each 함수를 사용하기
        list 함수는 배열을 각각의 변수로 바꿔주는 것이고 each는 배열에서 키와 값을 배열 형태로 넘겨주는 함수이다. 예를 들어서
        $arr = array(0 => 1, 1=> 2, 2 => 3);
        while(list($key, $value) = each($arr)) {
                 echo $key, $value;
        }

        대충 이런 형태로 가능한데. 요즘은 이렇게 쓰는 사람이 잘 없다. foreach가 더 심플하니깐.. .


     3) 연관 배열은 key, value로 이루어져 있으니 그냥 셋팅만 해주면 된다.
         $arr = array( 'key' => 'value' );  형태로 셋팅하면 끝.. 사용할 때는 $arr['key'] 로 얻어오면 됨..

4.  배열을 위한 유틸리티 함수들
     php의 동적 배열을 좀 더 자유롭고 유연하게 이용할 수 있도록 여러가지 함수들을 지원한다.

    1) 키 또는 값으로만 배열 만들기 
        $keys = array_keys($arr);
        $values = array_values($arr);

     2) 배열 값을 유니크한 배열로 만들기
        $unique = array_unique($arr);       // 만약 값이 array(1, "1", 2, 3,"3") 이런식이라면 만들어 지는 배열은 array(1,2,3) 이 됩니다.

     3) 배열 값으로 합계내기
         $sum = array_sum($arr);

     4) 배열 요소에 모두 특정 함수 사용하여 변환하기
        보통 배열을 특정 함수를 이용해서 바꿀려고 하면 loop를 도는게 일반적인 방식이다.
        for ($i = 0; $i < 10; $i++) {
             $arr[$i] = strtoupper($arr[$i]);
       }

       하지만 php에서는 이런상황을 위해서 여러가지를 지원하는데 가장 간단한 구조가 array_map 이다. array_map은 주어진 함수를 가지고 변환한 값들을 새로운 배열로 만들어 낸다.  위의 제어 구분을 아주 간단하게 바꾸면 아래와 같다.

      $arr = array_map('strtoupper', $arr);  // 모든 요소 대문자로 변경
      $arr = array_map('trim', $arr);   // 모든 요소 공백 제거

      여기서 앞에 사용되어지는 함수 이름은 문자열로 들어간다.  함수는 매개변수를 하나만 가진다.

     5) 배열 2개로 연관 배열 만들기
        $keys = array('1','2','3','4');
        $values = array('a','b','c','d');

        $arr = array_combine($keys, $values) ; 하면
        $arr = array('1' => 'a' , '2' => 'b', '3' => 'c', '4' => 'd');  와 같은 형태로 변경이 된다.

     6) 배열에 값이 있는지 없는지 체크하기
        단순하게 체크 할 때는 in_array 라는 함수를 사용한다.
         $arr = array(1, 2, 3, 4, 5,);
        if (in_array(1,$arr)) { echo "1 포함"; }

        아주 쉽다. 그렇다면 배열에 배열이 들어가 있는 이중배열은 어떻게 될까? 
        $arr = array(array(1, 2,), 3, 4, 5, array(6, 7));
        if (in_array(array(1,2), $arr)) { echo "array(1, 2) 가 있음 "; }

        배열 자체적으로 체크도 가능하다.

     7) 자료형이 배열인지 알아볼려면 ?
         echo is_array($arr) ;

      8) 배열을 정렬 할려면..
          (1) 값을 기반으로 정렬  : sort($arr);      , 역순 정렬 : rsort($arr);
          (2) 키를 기반으로 정렬  : ksort($arr);     , 역순 정렬 : krsort($arr);
          (3) 사용자 정의 함수로 정렬하기 : usort($arr, callback);
               function cmp($a, $b) {
                    if ($a == $b) return 0;
                    return ($a < $b) ? -1 : 1 ;
               }
               usort($arr, "cmp");  하면 자동으로 사용자 정의 함수를 기반으로 바뀐다.
              같으면 : 0, 작으면 : -1, 크면 : 1 로 함수로 잘 만들어보삼.. 당신의 능력을 믿어요.. ^^/
              일단은 이정도.... 다르게 정렬 하는 함수들이 있지만 다음에 소개할게요...

        9) 서브 배열 가지고 오기
            가끔 배열도 중간에 있는 리스트만 가지고 오고 싶을 때가 있다... 파이썬의 경우는 배열 자료형 자체도 객체이기 때문에 arr[:10]  과 같은 슬라이스 형태로
            서브배열을 가지고 오는 것이 가능하지만 php는 아직 그런것은 안되고 함수로 지원을 한다.

           $arr = array(1, 2, 3, 4, 5);
           $sub = array_slice ($arr, 2);     //  array(3, 4, 5);
           $sub = array_slice($arr, 2, 1);   // array(3);
           $sub = array_slice($arr ,2, -1);  // array(3, 4);

           위와 같은 형태로 배열을 가지고 올 수 있습니다.
           기본 형태는 array_slice (배열, 시작지점, 길이);   정도로 된다. 길이는 생략가능하고 길이가 안정해지면 마지막 요소까지로 측정된다.

       10) 배열 요소 랜덤으로 가지고 오기
            $values = array_rand($arr, $num); 

            위와 같은 형태로 사용하는데... 내부적으로 rand() 함수를 부르기 때문에 srand() 를 미리 실행해야 정확하게 각 시간별 랜덤결과를 얻을 수 있다.
            srand(time());
            $pick = array_rand($arr, $num);

           $num 은 랜덤으로 가지올 개수인데 1 개 일때는 그냥 일반 자료형으로 리턴해주고 2개 이상일때는 해당 키값들을 배열로 리턴해준다.

            그래서 실제 사용할려면
           echo $arr[$pick[0]];  형태로 값을 출력시켜야한다.

        11)  배열 합치기 (합집합)
               $new_arr = array_merge($ar1, $ar2);   와 같은형태로 배열을 합칠 수 있다.
               연관 배열로 이루어진 배열들은 키값이 같으면 뒤에서 합쳐진 배열 값을 기준으로 값이 정해진다.
               여기서 재미난 거 하나 .

               $ar1 = array();
               $ar2 = array(1 => 'a'); 
               $new = array_merge($ar1, $ar2) ;    // array(0 => 'a') 와 같은 형태로 기존의 인덱스는 완전 무시 된다.  이것을 방지할려면 + 연산자를 이용할 수 있다.

               $new = $ar1 + $ar2;   // array(1 => 'a');  배열끼리 + 연산자를 사용하면 인덱스를 유지한채 배열을 합칠 수 있다. ㅋ

          12)   키 값 존재 여부
                if (array_key_exists('key', $arr)) {  echo 'key 값 확인'; }

          13) 교집합 구하기
                교집합은 값을 기준으로 구할 수 있다.
                $arr1 = array('a' => '안녕', '바보', '메롱');
                $arr2 = array('b' => '안녕', '메롱', '쪼다'); 
                $result = array_intersect($arr1, $arr2);       // array('a' => '안녕', 0 => '메롱');

           14) 키와 값의 위치 바꾸기   (키 <-> 값)
                $arr = array('a' => 1, 'b' => 1, 'c' => 2);
               $temp = array_flip($arr);   // array(1 => 'b', 2 => 'c');

            15) 요소 개수별로 서브 배열 만들기
                  $arr = array(1, 2, 3, 4, 5);
                  $temp = array_chunk($arr, 2);   // array(array(1,2), array(3, 4), array(5));
                  $temp = array_chunk($arr, 2, true);   // array(array(0 => 1, 1 => 2), array(2 = >3, 3 => 4), array(4 => 5));

                  마지막에 true가 붙는 것은 인덱스를 그대로 유지할 것인가를 정한다. 기본값은 false라서 인덱스가 자동 정해진다.

        ※ 이 외에도 여러가지 함수들이 있지만 다 내 머리속에 없는 관계로 이만.. ㅋㅋ