>백엔드 개발 >PHP 튜토리얼 >PHP의 특성 및 해당 응용 프로그램에 대한 자세한 설명

PHP의 특성 및 해당 응용 프로그램에 대한 자세한 설명

墨辰丷
墨辰丷원래의
2018-05-25 14:37:431861검색

이 글은 주로 PHP의 Traits와 그 응용 프로그램을 자세히 소개합니다. 관심 있는 친구들이 참고할 수 있습니다.

PHP는 PHP 버전 5.4.0부터 새로운 코드 재사용 개념을 제공합니다. Trait은 말 그대로 "특성"과 "기능"을 의미합니다. Trait 키워드를 사용하면 PHP의 클래스에 새로운 특성을 추가할 수 있다는 것을 알 수 있습니다.

객체 지향에 익숙한 사람이라면 소프트웨어 개발에서 일반적으로 사용되는 두 가지 코드 재사용 방법, 즉 상속과 다형성이 있다는 것을 알고 있습니다. PHP에서는 단일 상속만 가능합니다. 특성은 이것을 방지합니다. 다음은 간단한 예를 통해 비교 설명합니다.

1. 상속 VS 다형성 VS 특성

이제 Publish.php와 Answer.php라는 두 가지 클래스가 있습니다. LOG 기능을 추가하려면 클래스 내부의 작업을 기록하세요. 여러 가지 옵션이 있습니다:

Inheritance
Polymorphism
Trait

1.1 Inheritance

표시된 바와 같습니다:

코드 구조는 다음과 같습니다.

// Log.php
<?php
Class Log
{
 public function startLog()
 {
  // echo ...
 }

 public function endLog()
 {
  // echo ...
 }
}

// Publish.php
<?php
Class Publish extends Log
{

} // Answer.php
<?php
Class Answer extends Log
{

}

상속이 요구 사항을 충족하는 것을 볼 수 있습니다. 그러나 이는 객체지향 원칙에 위배됩니다. 게시, 답변 및 로그와 같은 작업 간의 관계는 하위 클래스와 상위 클래스 간의 관계가 아닙니다. 따라서 이런 방식으로 사용하는 것은 권장되지 않습니다.

1.2. 다형성

표시됨:

구현 코드:

 // Log.php
<?php
Interface Log
{
 public function startLog();
 public function endLog();
}

// Publish.php
<?php
Class Publish implements Log
{
 public function startLog()
 {
  // TODO: Implement startLog() method.
 }
 public function endLog()
 {
  // TODO: Implement endLog() method.
 }
}

// Answer.php
<?php
Class Answer implements Log
{
 public function startLog()
 {
  // TODO: Implement startLog() method.
 }
 public function endLog()
 {
  // TODO: Implement endLog() method.
 }
}

로그 기록 작업은 동일하므로 게시 및 답변 작업의 로깅 구현도 동일합니다. 분명히 이는 DRY(Don't Repeat Yourself) 원칙을 위반합니다. 따라서 이 방법으로 구현하는 것은 권장되지 않습니다.

1.3. Trait

그림과 같이:

구현 코드는 다음과 같습니다:

 // Log.php
<?php
trait Log{
 public function startLog() {
  // echo ..
 }
 public function endLog() {
  // echo ..
 }
}

// Publish.php
<?php
class Publish {
 use Log;
}
$publish = new Publish();
$publish->startLog();
$publish->endLog();

// Answer.php
<?php
class Answer {
 use Log;
}
$answer = new Answer();
$answer->startLog();
$answer->endLog();

코드 복잡성을 증가시키지 않고도 코드 재사용이 가능하다는 것을 알 수 있습니다.

1.4. 결론

상속 방법도 문제를 해결할 수 있지만 그 아이디어는 객체 지향 원칙에 어긋나고 매우 조잡해 보이지만 다형성 방법도 적합하지 않습니다. 소프트웨어 개발의 DRY 원칙으로 인해 유지 관리 비용이 증가합니다. 특성 메소드는 위의 단점을 피하고 비교적 우아하게 코드 재사용을 달성합니다.

2. 특성의 범위

특성의 이점을 이해한 후에는 특성 구현의 규칙도 먼저 이해해야 합니다. 이를 증명하기 더 쉬운 구현 코드는 다음과 같습니다.

 <?php
class Publish {
 use Log;
 public function doPublish() {
  $this->publicF();
  $this->protectF();
  $this->privateF();
 }
}
$publish = new Publish();
$publish->doPublish();

위 코드를 실행한 결과는 다음과 같습니다.
public function
protected function
private function

It Trait의 범위가 이를 참조하고 있음을 알 수 있습니다. Trait 클래스의 내부가 보입니다. use 키워드는 Trait을 참조하는 클래스에 Trait의 구현 코드를 복사하는 것으로 이해될 수 있습니다.

3. Trait의 속성 우선순위

우선순위에 대해 말하자면, 여기서 참조 개체는 Trait 클래스와 해당 상위 클래스를 참조합니다.

다음 코드를 사용하여 Trait 애플리케이션에서 속성의 우선순위를 증명합니다.

 <?php
trait Log
{
 public function publicF()
 {
  echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
 }
 protected function protectF()
 {
  echo __METHOD__ . &#39; protected function&#39; . PHP_EOL;
 }
}

class Question
{
 public function publicF()
 {
  echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
 }
 protected function protectF()
 {
  echo __METHOD__ . &#39; protected function&#39; . PHP_EOL;
 }
}

class Publish extends Question
{
 use Log;

 public function publicF()
 {
  echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
 }
 public function doPublish()
 {
  $this->publicF();
  $this->protectF();
 }
}
$publish = new Publish();
$publish->doPublish();

위 코드의 출력은 다음과 같습니다.
Publish::publicF public function
Log::protectF protected function

위의 예를 통해 Trait 애플리케이션의 우선 순위는 다음과 같이 요약될 수 있습니다.
1. 현재 클래스의 멤버가 특성의 메서드를 재정의합니다.
2. Trait이 상속된 메서드를 재정의합니다.

클래스 멤버 우선 순위 : 현재 클래스 > Trait > 상위 클래스

4. 대신 및 As 키워드

클래스에서는 다음과 같이 여러 Traits를 참조할 수 있습니다. 한 클래스에서 여러 특성을 참조할 수 있습니다. 여러 개의 Trait을 참조할 때 문제가 발생하기 쉽습니다. 두 개의 Trait에 같은 이름의 속성이나 메소드가 있으면 어떻게 해야 합니까? 이때 대신of 및 as 키워드를 사용해야 합니다. 다음 구현 코드를 참조하세요.

 <?php

trait Log
{
  public function parameterCheck($parameters)
  {
    echo __METHOD__ . &#39; parameter check&#39; . $parameters . PHP_EOL;
  }

  public function startLog()
  {
    echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
  }
}

trait Check
{
  public function parameterCheck($parameters)
  {
    echo __METHOD__ . &#39; parameter check&#39; . $parameters . PHP_EOL;
  }

  public function startLog()
  {
    echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
  }
}

class Publish
{
  use Check, Log {
    Check::parameterCheck insteadof Log;
    Log::startLog insteadof Check;
    Check::startLog as csl;
  }

  public function doPublish()
  {
    $this->startLog();
    $this->parameterCheck(&#39;params&#39;);
    $this->csl();
  }
}

$publish = new Publish();
$publish->doPublish();

执行上述代码,输出结果如下:
 Log::startLog public function
Check::parameterCheck parameter checkparams
Check::startLog public function

就如字面意思一般,insteadof关键字用前者取代了后者,as 关键字给被取代的方法起了一个别名。 

在引用Trait时,使用了use关键字,use关键字也用来引用命名空间。两者的区别在于,引用Trait时是在class内部使用的。

以上就是本文的全部内容,希望对大家的学习有所帮助。


相关推荐:

关于PHP中的trait简单介绍

PHP中trait使用方法图文详解

php的Traits属性以及基本用法

위 내용은 PHP의 특성 및 해당 응용 프로그램에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.