php5 autoload 활용하기

PHP/팁앱테크 2012. 4. 3. 13:39

PHP5 autoload 함수 활용하기


PHP5 부터는 __autoload 라는 특별한 함수를 지원한다.

기존의 클래스 파일을 사용하기 위해서는

include '/classes/db/DbMysqli.class.php';
$db =  new DbMysqli();

이와 같이 include 먼저 하고나서
클래스를 선언하여 사용해야 했다.
엄청 불편하다

자바처럼 그냥 new 로 생성해서 사용하면 편한데 말이다. ^^

그래서 나온것이 바로 요놈 __autoload 이다


이놈을 사용하게 되면
그냥 include 를 수동으로 코딩하지 않아도
__autoload 이놈이 알아서 include 해준다.
그러므로 이제부터는 그냥

$db = new  DbMysqli();

이렇게만 선언해서 사용하면 된다.

아래 소스는 __autoload 를 정의 해 놓은 것이다
소스는 이렇다

@ __autoload Source

# 절대 경로
define('_ROOT_PATH_', '/www/설치경로');

# 클래스 자동 인클루드
function __autoload($class_name)
{
        # 클래스명중 경로 추출
	$tmp_args = explode(' ', preg_replace( '/([a-z0-9])([A-Z])/', "$1 $2",$class_name));
	$class_path_name = sprintf("%s/{$class_name}",strtolower($tmp_args[0]));
	
	# 체크
         if(!class_exists($class_path_name,false))
	{
		# classes 폴더
		if(file_exists(_ROOT_PATH_.'/classes/'.$class_path_name.'.class.php') !==false){
			include_once _ROOT_PATH_.'/classes/'.$class_path_name.'.class.php';
		}

		# modules 폴더
		else if(file_exists(_ROOT_PATH_.'/modules/'.$class_path_name.'.class.php') !==false){
			include_once _ROOT_PATH_.'/modules/'.$class_path_name.'.class.php';
		}

		# 기타 만든 클래스 폴더 [첫대문자만 폴더로 지원]
		# ( /my/MyTest.class.php -> MyTest.class.php)
		else if(file_exists(_ROOT_PATH_.'/'.$class_path_name.'.class.php') !==false){
			include_once _ROOT_PATH_.'/'.$class_path_name.'.class.php';
		}
	}
}


자 이젠 __autoload 함수 정의 해 놓은 것을 설명할 차례군요

먼저, 아래와 같은 폴더 구조에 클래스 파일들이 있다고 가정을 합니다.

|----------------------------------------------------------------
|       파일경로 및 클래스 파일명         |         클래스명               
|-----------------------------------+----------------------------
/classes/db/DbMysqli.class.php         // class DbMysqli{ }
/classes/req/ReqForms.class.php       // class  ReqForms { } 
/classes/tpl/TplFusion.class.php         // class  TplFusion {}

/modules/bbs/BbsModel.class.php        // class BbsModel{ }
/modules/mem/MemModel.class.php      // class MemModel{ }

/my/MyTest.class.php                           // class MyTest{ }


1. __autoload($class_name) 는 new 라는 명령만 떨어지면 자동으로 호출이 됩니다.

$db = new DbMysqli();


new -> __autoload 를 호출하고
$class_name -> DbMysqli 클래스 명이 되는거죠

2. 클래스명에서 클래스가 위치 하고 있는 경로를 추출합니다. 기준은 첫번째 대문자로 시작하고
다음 대문자가 나오기 전까지의 클래스명의 문자는 폴더를 가리킵니다.

DbMysqli    이건 폴더 db 를 가리키고
ReqForms   이건 폴더 req 를 가리키고
BbsModel   이건 폴더 bbs 를 가리키고
MemModel  이건 폴더 Mem 를 가리키고
MyTest       이건 폴더 my 를 가리킵니다.

위와 같은 기준으로 위치 (폴더) 를 알아냅니다.

# 클래스명중 경로 추출
$tmp_args = explode(' ', preg_replace( '/([a-z0-9])([A-Z])/', "$1 $2",$class_name));
$class_path_name = sprintf("%s/{$class_name}",strtolower($tmp_args[0]));


2. 이미지 인클루드된 클래스 파일이 있는지 확인 한 후 있으면 include 안하고
없으면 하게 될겁니다.

제가 코딩 해 놓은건
일단 무조건 

1. classes 폴더 밑에 클래스 이름 폴더를 찾고 없으면
2. modules 폴더 밑에 클래스 이름 폴더를 찾고 그리고 나서
3. 선언된 클래스의 이름 폴더를 찾도록 하였습니다.

전 이유가 있어서 그랬지만  1,2번을 삭제하고 바로 3번만 선언해서 사용하셔도 됩니다.

그 역할을 하는 소스 부분이 바로 이 부분

# 체크
 if(!class_exists($class_path_name,false))
{
		# 1. classes 폴더
		if(file_exists(_ROOT_PATH_.'/classes/'.$class_path_name.'.class.php') !==false){
			include_once _ROOT_PATH_.'/classes/'.$class_path_name.'.class.php';
		}

		# 2. modules 폴더
		else if(file_exists(_ROOT_PATH_.'/modules/'.$class_path_name.'.class.php') !==false){
			include_once _ROOT_PATH_.'/modules/'.$class_path_name.'.class.php';
		}

		# 3. 기타 만든 클래스 폴더 [첫대문자만 폴더로 지원]
		# ( /my/MyTest.class.php -> MyTest.class.php)
		else if(file_exists(_ROOT_PATH_.'/'.$class_path_name.'.class.php') !==false){
			include_once _ROOT_PATH_.'/'.$class_path_name.'.class.php';
		}
}


끝까지 읽어 주셔서 감사합니다.
이 방법 말고도 php.net 에 보시면 클래스도 지원합니다.

마음에 드는 방법으로 사용하세요


php array 값중 가장 큰 값을 추출(max)하고 그 값이 가리키고 있는 키값(array_search)을 추출한다

PHP/팁앱테크 2012. 3. 29. 11:43

php array 값중 가장 큰 값을 추출(max)하고 그 값이 가리키고 있는 키값(array_search)을 추출한다


PHP의 매력 ㅎㅎㅎ
뭐 말하려니 입이 아프네요 ㅎㅎㅎ

오늘은 배열 값 중에서 가장 큰 값을 찾는 것과
그 값이 가리키는 키값을 찾는 걸 해볼까 합니다.

알면 유용하니 잘 보아 두세요.


@ 다음과 같은 배열이 있습니다.

$tmp_v_args = 

// 결과
Array
(
    [0] => 0
    [1] => 0
    [2] => 0
    [3] => 0
    [4] => 0
    [5] => 0
    [6] => 0
    [7] => 0
    [8] => 0
    [9] => 0
    [10] => 0
    [11] => 0
    [12] => 0
    [13] => 0
    [14] => 0
    [15] => 0
    [16] => 0
    [17] => 0
    [18] => 0
    [19] => 58
    [20] => 33
    [21] => 58
    [22] => 57
    [23] => 9
    [24] => 16
    [25] => 102
    [26] => 115
    [27] => 138
    [28] => 22
    [29] => 0
    [30] => 0
)



@ 배열중 가장 큰 값을 추출 하겠습니다.

$max_int_v = max($tmp_v_args); #배열중 가장 큰 값 추출

// 결과
138



@ 추출한 값이 가리키는 배열의 키를 추출하겠습니다.

$max_int_v_key = array_search($max_int_v, $tmp_v_args); #값이 가리키는 키값 추출

// 결과
27



@ 위 코드를 한번에 작성하면 다음과 같이 되겠죠

$max_int_v = max($tmp_v_args); #배열중 가장 큰 값 추출
$max_int_v_key = array_search($max_int_v, $tmp_v_args); #값이 가리키는 키값 추출



끝까지 읽어 주셔서 감사합니다.

php array 함수의 마력 array_splice, array_values 활용

PHP/팁앱테크 2012. 3. 29. 11:10

php array 함수의 마력 array_splice, array_values 활용


PHP는 가볍고 속도가 빠른것으로 유명하다
실제 프로그래밍을 해봐도 그렇다 
확장성 또한 띄어나다

그중의 하나인 array 함수 들 중
array_splice 자르기, array_values 값만 추출하여 숫자로 인덱싱 해주는
이 놀라운 함수로 원하는 데이타를 추출해 보겠다


@ 먼저 다음과 같은 배열값이 있다고 하자


# 변수명 
$args = 
Array
(
    [uid] => 1
    [regi_ym] => 201203
    [ads_url_info_uid] => 1
    [totals] => 602
    [d1] => 0
    [d2] => 0
    [d3] => 0
    [d4] => 0
    [d5] => 0
    [d6] => 0
    [d7] => 0
    [d8] => 0
    [d9] => 0
    [d10] => 0
    [d11] => 0
    [d12] => 0
    [d13] => 0
    [d14] => 0
    [d15] => 0
    [d16] => 0
    [d17] => 0
    [d18] => 0
    [d19] => 0
    [d20] => 58
    [d21] => 33
    [d22] => 58
    [d23] => 57
    [d24] => 9
    [d25] => 16
    [d26] => 102
    [d27] => 115
    [d28] => 138
    [d29] => 16
    [d30] => 0
    [d31] => 0
)


@ array_splice 로 d1 ~ d31 까지만 값이 필요하므로 그 부분만 따로 다른 변수에 담도록 하겠다


$row_splice = array_splice($args,4);
print_r($row_splice);

// 결과
Array
(
    [d1] => 0
    [d2] => 0
    [d3] => 0
    [d4] => 0
    [d5] => 0
    [d6] => 0
    [d7] => 0
    [d8] => 0
    [d9] => 0
    [d10] => 0
    [d11] => 0
    [d12] => 0
    [d13] => 0
    [d14] => 0
    [d15] => 0
    [d16] => 0
    [d17] => 0
    [d18] => 0
    [d19] => 0
    [d20] => 58
    [d21] => 33
    [d22] => 58
    [d23] => 57
    [d24] => 9
    [d25] => 16
    [d26] => 102
    [d27] => 115
    [d28] => 138
    [d29] => 16
    [d30] => 0
    [d31] => 0
)

내가 원하는 값만 추출 했다.

d1 위치가 4번째 배열위치에 있었으므로 4번째 배열부터 끝배열까지 자르라는 의미였다.



@ 이젠 array_values 를 통해 d1,d2 키를 없애고 [0]=>1, [1]=>2 숫자 인덱스 값을 갖는 배열로 바꿔보겠다.


$row_values = array_values($row_splice);
print_r($row_values);

// 결과
Array
(
    [0] => 0
    [1] => 0
    [2] => 0
    [3] => 0
    [4] => 0
    [5] => 0
    [6] => 0
    [7] => 0
    [8] => 0
    [9] => 0
    [10] => 0
    [11] => 0
    [12] => 0
    [13] => 0
    [14] => 0
    [15] => 0
    [16] => 0
    [17] => 0
    [18] => 0
    [19] => 58
    [20] => 33
    [21] => 58
    [22] => 57
    [23] => 9
    [24] => 16
    [25] => 102
    [26] => 115
    [27] => 138
    [28] => 22
    [29] => 0
    [30] => 0
)


올 내가 딱 원하는 형태로 데이타 형태가 바뀌었다

이 놀라운 PHP의 세계를 경험해 보라


위 단계를 다음과 한줄로 코딩해도 됩니다.

그래도 같은 결과가 나옵니다.

print_r( array_values(array_splice($row,4)) );



끝까지 읽어 주셔서 감사합니다.


PHP Array 배열 함수 페이지


abcd1004 중 뒤의 숫자만 추출하기

PHP/팁앱테크 2012. 3. 21. 15:32
가끔 프로그램을 하다 보면
앞에는 문자와 뒤에는 숫자만의 조화를 만들어 내야 할때도 있고

그중 뒤에 숫자만 추출해야 하는 경우가 아주 가끔씩 있다

이때 사용하면 좋을 것

보통 주소를 줄여서 사용할 수 있도록 서버 셋팅이 되어 있지 않다면
우리는 스스로 주소를 줄여서 사용해야 할때도 사용하면 좋습니다.

그렇다면 서버 환경에 의존하지 않아도 되니
맘 편한듯...

< ?php
// 주소 창에 입력
//http://apmsoft.test.com?bbs1004

$sn_pos = strpos($_SERVER['REQUEST_URI'], '?');
if($sn_pos>0 && strpos($_SERVER['REQUEST_URI'], '&') === false && strpos($_SERVER['REQUEST_URI'], '=') === false)
{
	$req_id = substr($_SERVER['REQUEST_URI'], $sn_pos+1, strlen($_SERVER['REQUEST_URI']));

	echo $req_id;  // bbs1004
	echo ' 
'; // 처음으로 시작하는 숫자위치를 찾는다 preg_match("/([0-9]+)$/", $req_id, $out); // 결과 : Array ( [0] => 1004 [1] => 1004 ) print_r($out); } ? >

php5 mysqli 디비 접속 클래스 프로그램 및 사용법

PHP/PHP5 클래스 2012. 3. 20. 20:55
웹스크립트언어의 최강자 PHP 프로그램입니다.

PHP를 사용하려면 반드시 알아야  할것이 디비접속을 하여 그 데이타를 추출할 줄 알아야
좋은 프로그램을 만들 수 있지요

PHP5 부터는 많음 클래스 들을 제공하는데
그 중에 하나가 mysqli 라는 클래스 입니다.

이 클래스 mysqli 를  그대로 상속받아
좀더 사용하기 쉽도록 mysqli 관련 클래스를 만들드록 하겠습니다.

하지만, 모든 언어가 그렇듯이
PHP도 반드시 mysql 에만 접속 해서 사용하라는 법이 없습니다.

다양한 데이타베이스에 접속해서 사용을 할 수 있죠

그렇지만 php 코딩은 갖습니다.
단 디비만 바뀌는 거죠 이렇게 디비만 바뀌면 되는데
디비를 접속하고 쿼리 날리고 저장하고 하는 일들에 대해
프로그램을 수정해야 한다면 정말 일이 많고 스트레스가 상승하게 됩니다.

이러한 불편한 작업들을 하지 않기 위해서
하나의 interface를 정의해 놓고
다른 여러 프로그래머들이 데이타 베이스 프로그램을 작성 할때
동일한 메소드들을 정의하고 사용하도록 규칙을 정합니다.

가장 많이 사용하는
query, insert, modify, delete 등이죠
최소한의 것을 정의해서 사용하는 겁니다.

더 많은 것들을 정의 할 수 있지만
우선 위의 것들을 정의한 interface 를 만들고 mysqli를 상속 받아
보다 사용하기 편리한 데이타베이스 클래스 파일을 만들어 보겠습니다.

@ interface : DbSwitch 클래스를 만들겠습니다. 
< ?php
# purpose : 각종 SQL 관련 디비를 통일성있게  작성할 수 있도록 틀을 제공
interface DbSwitch
{
    public function query($query);			# 쿼리
    public function insert($table);			# 저장
    public function update($table,$where);	# 수정
    public function delete($table,$where);	# 삭제
    public function bindParams($query,$args=array());	#쿼리 문자 바인드효과
}
? >

@ 디비 접속 정보를 미리 설정해 놓겠습니다.
< ?php
define('_DB_HOST_','localhost');		#접속 경로
define('_DB_USER_','apmsoft');			#접속 아이디
define('_DB_PASSWD_','password');		#접속 비밀번호
define('_DB_NAME_','apmsoft_db');		#접속 데이타베이스 명
? >


@ DbMySqli : [클래스1개] mysqli, [인터페이서 2개] DbSwitch, ArrayAccess 를 상속 받아 구현을 합니다.
< ?php
# Parent : MySqli
# Parent : DBSwitch
# purpose : mysqli을 활용해 확장한다
class DbMySqli extends mysqli implements DbSwitch,ArrayAccess
{
	private $params = array();
	
	# dsn : host:dbname = localhost:dbname
    public function __construct($dsn='',$user='',$passwd='',$chrset='utf8')
	{
		# 데이타베이스 접속
		if(!empty($dsn)){
			$dsn_args = explode(':',$dsn);
			parent::__construct($dsn_args[0],$user,$passwd,$dsn_args[1]);
		}else{//config.inc.php --> config.db.php
			parent::__construct(_DB_HOST_,_DB_USER_,_DB_PASSWD_,_DB_NAME_);
		}
        
        if (mysqli_connect_error()){
        	throw new ErrorException(mysqli_connect_error(),mysqli_connect_errno());
        }

		# 문자셋
		$chrset_is = parent::character_set_name();
		if(strcmp($chrset_is,$chrset)) parent::set_charset($chrset);
    }
	
	#@ interface : ArrayAccess
	# 사용법 : $obj["two"] = "A value";
    public function offsetSet($offset, $value) {
        $this->params[$offset] = parent::real_escape_string($value);
    }
    
    #@ interface : ArrayAccess
    # 사용법 : isset($obj["two"]); -> bool(true)
	public function offsetExists($offset) {
        return isset($this->params[$offset]);
    }
    
    #@ interface : ArrayAccess
    # 사용법 : unset($obj["two"]); -> bool(false)
	public function offsetUnset($offset) {
        unset($this->params[$offset]);
    }
    
    #@ interface : ArrayAccess
   # 사용법 : $obj["two"]; -> string(7) "A value"
	public function offsetGet($offset) {
        return isset($this->params[$offset]) ? $this->params[$offset] : null;
    }
	
	# @ interface : DBSwitch
	# :s1 -> :[문자타입]+[변수]
	# :s[문자], :d[정수], :f[소수], :b[바이너리]
	# 변수타입, :s1,:sa,:sA, :d1, :d2, :dA 어떻게든 상관없음
	# 단 :s1, :s1 이렇게 중복 되어서는 안됨	
        # where 구문만 변경
	// ("SELECT * FROM `TABLE` WHERE name=':s1' and age=':d2'",array('php',26,'나','ㅈㄷ',22));
	public function bindParams($query,$args=array()){
		if(strpos($query,':') !==false){
			preg_match_all("/(\:[s|d|f|b])+[0-9]+/s",$query,$matches);
			if(is_array($matches))
			{
				foreach($matches[0] as $n => $s)
				{
					# 문자타입과 값이 일치하는지 체크
					$bindtype = substr($s,1,1);
					$bvmatched = false;
					switch($bindtype){
						case 's': if(is_string($args[$n])) $bvmatched = true; break;
						case 'd': if(is_int($args[$n])) $bvmatched = true; break;
						case 'f': if(is_float($args[$n])) $bvmatched = true; break;
						case 'b': if(is_binary($args[$n])) $bvmatched = true; break;
					}
					if($bvmatched){
						$query = str_replace($s,'%'.$bindtype,$query);
						$query = sprintf("{$query}",parent::real_escape_string($args[$n]));
					}else{
						$query = false;
						break;
					}
				}
			}
		}
	return $query;
	}

	#@ return int
	# 총게시물 갯수 추출
	public function get_total_record($table, $where="", $field='*'){
		$wh = ($where) ? " WHERE ".$where : '';
		if($result = parent::query('SELECT count('.$field.') FROM `'.$table.'`'.$wh)){
			$row = $result->fetch_row();
			return $row[0];
		}
	return 0;
	}

	// 하나의 레코드 값을 가져오기
	public function get_record($field, $table, $where){
		$where = ($where) ? " WHERE ".$where : '';
		$qry = "SELECT ".$field." FROM `".$table."` ".$where;
		if($result = $this->query($qry)){
			$row = $result->fetch_assoc();
			return $row;
		}
	return false;
	}

    # @ interface : DBSwitch
	public function query($query){
		$result = parent::query($query);
        if( !$result ){
        	throw new ErrorException(mysqli_error(&$this).' '.$query,mysqli_errno(&$this));
        }
    return $result;
    }

	# @ interface : DBSwitch
	# args = array(key => value)
	# args['name'] = 1, args['age'] = 2;
	public function insert($table){
		$fieldk = '';
		$datav	= '';
		if(count($this->params)<1) return false;
		foreach($this->params as $k => $v){
			$fieldk .= sprintf("`%s`,",$k);
			$datav .= sprintf("'%s',", parent::real_escape_string($v));
		}
		$fieldk	= substr($fieldk,0,-1);
		$datav	= substr($datav,0,-1);
		$this->params = array(); #변수값 초기화
		
		$query	= sprintf("INSERT INTO `%s` (%s) VALUES (%s)",$table,$fieldk,$datav);
		$this->query($query);
	}
    
	# @ interface : DBSwitch
	public function update($table,$where)
	{
		$fieldkv = '';
		
		if(count($this->params)<1) return false;
		foreach($this->params as $k => $v){
			$fieldkv .= sprintf("`%s`='%s',",$k,parent::real_escape_string($v));
		}
		$fieldkv = substr($fieldkv,0,-1);
		$this->params = array(); #변수값 초기화
		
		$query	= sprintf("UPDATE `%s` SET %s WHERE %s",$table,$fieldkv,$where);
		$this->query($query);
	}

	# @ interface : DBSwitch
    public function delete($table,$where){
    	$query = sprintf("DELETE FROM `%s` WHERE %s",$table,$where);
    	$this->query($query);
    }
    
	# 상속한 부모 프라퍼티 값 포함한 가져오기
	public function __get($propertyName){
		if(property_exists(__CLASS__,$propertyName)){
			return $this->{$propertyName};
		}
	}
    
    # db close
    public function __destruct(){
    	parent::close();
    }
}
? >

여기서 부터는
member 테이블이 있다고 가정하고
uid : 회원고유번호 userid : 아이디 name : 이름 passwd : 비밀번호 email : 이메일 정보 hp : 휴대폰

1. 데이타 입력
2. 데이타 쿼리
3. 데이타 수정
4. 데이타 삭제  예문을 보겠습니다.

1.  insert 디비 저장 /-------------------
< ?php
# 클래스 선언
$db = new DbMySqli();

# 디비 저장
$db['uid']		= 1;
$db['userid']	= 'apmsoft';
$db['passwd']	= 'pwd1004';
$db['name']		= '멋쟁이';
$db['email']	= 'apmsoft@test.com';
$db['hp']		= '010-3456-9876';

$db->insert();
? >

2. query  디비 쿼리 /---------------------------
/**
  일반적인 복수의 데이타 쿼리문
*/
< ?php
# 디비 선언
$db = new DbMySqli();

# 쿼리
$result = $db->query("SELECT * FROM `member` WHERE uid='1'");
while( $row = $result->fetch_assoc() )
{
       echo $row['uid'];
       echo $row['userid'];
       echo $row['name'];
}
? >

/**
    단순 데이타 쿼리문
*/
< ?php
# 디비 선언
$db = new DbMySqli();

# 쿼리
$row = $db->get_record('*', 'member',  "uid='1'");

echo $row['uid'];
echo $row['userid'];
echo $row['name'];
? >

3. update 디비 정보 수정 /----------------------------------
< ?php
# 디비선언
$db = new DbMySqli();

# 디비정보 수정
$db['name']		= '홍길동';
$db['email']	= 'ddd@gmail.com';

$db->update('member', "uid='1'");
? >


4. delete 디비 삭제 /----------------------------------------
< ?php
# 디비선언
$db = new DbMySqli();

# 디비정보 삭제하기
$db->delete('member', "uid='1'");
? >