우리는 JAVASCRIPT에서 배열 모드를 사용하여 객체의 속성과 메서드에 액세스할 수 있다는 것을 알고 있습니다. 그러나 일반적으로 이것은 불가능합니다.
왜 이러는 걸까요? 이렇게 하면 객체를 보다 편리하게 조작할 수 있고 클래스를 정의할 수 있기 때문입니다. 키 값 배열을 정의하는 대신. 당연히 다른 방법이 있다면 가장 간단한 방법은 강제로 배열로 변환하는 것입니다. 그러나 이렇게 하면 개체의 원래 메서드가 손실됩니다.
그러나 SPL의 ArrayObject는 배열 모드를 사용하여 속성에 액세스하는 데 도움이 될 수 있습니다. 그러나 이 방법은 아직 구현이 불가능합니다.
ArrayObject 클래스 구조는 다음과 같습니다(일부 메소드는 php5,1 또는 php5.2에 추가되었습니다):
ArrayObject implements IteratorAggregate , Traversable , ArrayAccess , Serializable , Countable { /* 常量 */ const integer STD_PROP_LIST = 1 ; const integer ARRAY_AS_PROPS = 2 ; /* 方法 */ __construct ([ mixed $input [, int $flags [, string $iterator_class ]]] ) void append ( mixed $value ) void asort ( void ) int count ( void ) array exchangeArray ( mixed $input ) array getArrayCopy ( void ) int getFlags ( void ) ArrayIterator getIterator ( void ) int getIteratorClass ( void ) void ksort ( void ) void natcasesort ( void ) void natsort ( void ) bool offsetExists ( mixed $index ) mixed offsetGet ( mixed $index ) void offsetSet ( mixed $index , mixed $newval ) void offsetUnset ( mixed $index ) public void serialize ( void ) void setFlags ( int $flags ) void setIteratorClass ( string $iterator_class ) void uasort ( callback $cmp_function ) void uksort ( callback $cmp_function ) public void unserialize ( string $serialized ) }
그 중: $obj['name'을 사용할 수 있는 이유는 무엇입니까? ] $obj->name에 대한 직접 액세스는 어떻습니까? 주로 위의 세 가지 방법:
offsetGet은 $obj['name'] 읽기 방법을 지원합니다.
offsetSet은 $obj['name'] 쓰기 방법을 지원합니다.
그러나 foreach는 ArrayAccess 함수 Current의 기본 구현 등.
예제 코드 보기:
class test extends ArrayObject{ public $name; private $age = 21; public function show(){ print_r(get_object_vars($this)); } } class test1{ public $name; private $age = 21; public function show(){ print_r(get_object_vars($this)); } } $obj=new test(); //使用数组方式读写属性 $obj['name']='hello'; $obj['nick']='mockArray'; echo $obj['nick'], '</br>'; var_dump($obj['show']);//检测是否可以访问方法: print_r($obj);//输出对象 $obj->show();//调用方法 $arr=(array)$obj; //强制转换成数组。 print_r($arr); //$arr->show(); 此行将出错,因为,原有方法全部丢失。 $obj1=new test1(); //创建普通对象 $arr1=(array)$obj1; //强制转换成数组。 print_r($arr1); //隐私完全暴光
이 코드는 다음을 출력합니다:
mockArrayNULL
test Object
(
[name] => hello
[nick] => mockArray
)
Array
(
[name] => hello
[nick] => mockArray
)
배열
(
[이름] => 안녕하세요
[닉네임] => mockArray
)
배열
(
[이름] = >
[ test1 age] => 21
)
배열 모드를 사용하여 속성에 접근할 수는 있지만 메소드(멤버 함수)에는 접근할 수 없는 것을 볼 수 있습니다.
강제 변환 후에는 배열 객체이며 멤버 함수가 없습니다.
물론 필요에 따라 offsetGet 및 offsetSet 두 가지 메소드를 추가로 다시 작성할 수도 있습니다. 왜? 매우 비정상적인 요구 사항이 있으면 확실히 유용할 것이기 때문입니다. 예를 들어, 참조를 사용하여 세 개의 배열을 하나의 객체로 래핑하고 이를 배열로 액세스하려고 합니다. 이때 이 두 함수를 다시 작성해야 합니다. 물론 ArrayAccess 인터페이스의 해당 기능도 다시 작성해야 합니다.
또한, 접근 가능한 모든 자산은 공공 자산입니다. 비공개라면 접근이 불가능합니다. 강제로 배열을 시켜도 똑같습니다. 그러나 ArrayObject를 상속받지 않으면 다릅니다. 이러한 클래스가 강제로 배열로 변환되면 해당 클래스의 개인 정보(개인 속성)가 노출됩니다.
그러나 private 속성이 배열로 변환된 후에는 원래 속성 이름이 유지되지 않는 것을 볼 수 있습니다. 대신 인쇄할 수 없는 특정 문자 + 클래스 이름 + 인쇄할 수 없는 문자 + 속성 이름의 형식을 사용합니다. 인쇄할 수 없는 문자의 ASCII 값은 확인되지 않았습니다. 관심이 있으시면 확인해 보세요!