>백엔드 개발 >PHP 튜토리얼 >현재 PHP를 정말로 이해하고 있나요?

현재 PHP를 정말로 이해하고 있나요?

PHPz
PHPz원래의
2017-04-04 15:59:171199검색

얼마 전 회사의 프로젝트가 PHP5.3에서 PHP7로 업그레이드되었습니다. 이제 프로젝트에서 PHP7의 새로운 구문과 기능 중 일부 PHP 버전 5.4, 5.5, 5.6을 되돌아보면 지식이 조금 부족하다고 느껴서 몇 가지 개념을 보충하기 위해 "Modern PHP"를 읽기로 했습니다.

이 글을 읽으면서. 책현재 PHP를 정말로 이해하고 있나요?
1. 특징

1.

네임스페이스

네임스페이스를 더 자주 사용하며, 자세히 기록하지 않고 몇 가지 주목할 만한 사례와 세부 사항을 기록합니다다중 가져오기

이렇게 하면 사람들이 혼란스러워지기 쉽습니다.

<?php
use Symfony\Component\HttpFoundation\Request,
    Symfony\Component\HttpFoundation\Response,
    Symfony\Component\HttpFoundation\Cookie;
한 줄에 하나씩 작성하는 것이 좋습니다:
<?php
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpFoundation\Cookie;

하나의 파일에 여러 네임스페이스 사용

이렇게 할 수 있지만 이는

“하나의 파일이 하나의 클래스를 정의합니다”

<?php
namespace Foo {
  //code
}

namespace Bar {
  //code 
}

<🎜를 위반합니다. >글로벌 네임스페이스PHP의 기본

Exception

클래스를 사용하려면 클래스 이름 앞에

<?php
namespace My\App;

class Foo
{
  public function doSomething()
  {
    $exception = new \Exception();
  }
}
기호를 추가해야 합니다. >Exception 앞에 기호가 없으면 , Exception 클래스는 MyApp 네임스페이스에 있습니다.
2. 인터페이스

사용 인터페이스를 사용하여 작성된 코드는 더 유연하며 위임만 하면 됩니다. 구현보다는 인터페이스에 신경쓰세요.

3. 특성

라라벨

프레임워크를 배우기 전에는 특성을 이해하지 못했습니다. > 이것은 PHP5.4.0에 도입된 새로운 개념으로, 클래스와 인터페이스와 비슷하지만 둘 다 아닙니다. >

특성은 하나 이상의 기존 PHP 클래스에 혼합될 수 있는 클래스의 부분 구현입니다.

트레이트를 사용하는 이유<. 🎜>예를 들어 Car와 Phone이라는 두 가지 클래스가 있는데 둘 다 이 문제를 해결하려면 다음과 같이 하세요. 첫 번째 반응은 상위 클래스를 만든 다음 이를상속

하는 것입니다. 그러나 이 조상이 해당 상속 계층에 속하지 않는 것이 분명하기 때문입니다. 두 번째 반응은 GPS 인터페이스를 생성하고 GPS 기능 인터페이스를 정의한 다음 Car 및 Phone 클래스 모두에서 이 인터페이스를 구현하도록 하는 것입니다. 그러나 이를 위해서는 두 클래스 모두에서 GPS 기능을 반복해야 합니다. 이는 DRY(반복하지 않음) 원칙을 따르지 않습니다. 세 번째 반응은 GPS 기능을 구현하는 특성을 만든 다음 이 특성을 Car 및 Phone 클래스에 혼합하는 것입니다. 기능을 실현할 수 있고 상속 구조에 영향을 주지 않으며 구현을 반복하지 않으며 완벽합니다.

특성 생성 및 사용특성 생성

<?php
trait MyTrait{
  //实现
}
특성 사용
<?php
class MyClass
{
  use MyTrait;
  // 类的实现
}

4.

생성기

PHP 생성기(generator)는 PHP5.5.0에 도입된 새로운 기능으로, 많은 PHP 개발자들이 이해하지 못하는 부분입니다. 생성기는 단순한 반복자이지만

반복자 인터페이스를 구현하는 데 꼭 필요한 것은 아닙니다. 생성기는 필요에 따라 반복할 값을 계산하고 생성합니다. 쿼리
가 없으면 생성기는 반복할 다음 값이 무엇인지 결코 알 수 없으며 생성기에서 뒤로 돌아가거나 빨리 감기할 방법이 없습니다. 구체적으로 다음 두 가지 예를 살펴보세요.

간단한 생성기

<?php
function makeRange($length) {
  for ($i = 0; $i < $length; $i++) {
    yield $i;
  }
}

foreach (makeRange(1000000) as $i) {
  echo $i, PHP_EOL;
}

특정 시나리오: 생성기를 사용하여 CSV 파일 처리

<?php
function getRows($file) {
  $handle = fopen($file, &#39;rb&#39;);
  if ($handle === false) {
    throw new Exception();
  }
  while (feof($handle) === false) {
    yield fgetcsv($handle);
  }
}

foreach (getRows(&#39;data.csv&#39;) as $row) {
  print_r($row);
}
이 시나리오를 처리하기 위한 일반적인 방법은 먼저 파일의 모든 내용을 읽고

배열

에 넣은 다음 처리 등을 수행하는 것입니다. 이러한 처리의 문제점은 다음과 같습니다. 파일이 매우 큰 경우 한 번에 읽으려면 메모리 리소스를 많이 차지합니다. 생성기는 시스템 메모리를 거의 차지하지 않기 때문에 이 시나리오에 가장 적합합니다.

5. 클로저이론적으로 클로저와

익명함수

는 다른 개념입니다. 그러나 PHP는 이를 동일한 개념으로 취급합니다. 간단한 클로저

<?php
$closure = function ($name) {
  return sprintf(&#39;Hello %s&#39;, $name);
}

echo $closure("Beck");
// 输出 --> “Hello Beck”
참고: $closure변수를 호출할 수 있는 이유는 이 변수의 값이 클로저이고 클로저

Object

는 Invoke()

마법의 메소드를 구현합니다. 변수명
뒤에 ()가 있으면 PHP는 호출() 메서드를 찾아 호출합니다.

附加状态
使用use关键字可以把多个参数传入闭包,此时要像PHP函数或方法的参数一样,使用逗号分隔多个参数。

<?php
function enclosePerson($name) {
  return function ($doCommand) use ($name) {
    return sprintf(&#39;%s, %s&#39;, $name, $doCommand);
  };
}

// 把字符串“Clay”封装在闭包中
$clay = enclosePerson(&#39;Clay&#39;);

// 传入参数,调用闭包
echo $clay(&#39;get me sweet tea!&#39;);
// 输出 --> "Clay, get me sweet tea!"

使用bindTo()方法附加闭包的状态
PHP框架经常使用bindTo()方法把路由URL映射到匿名回调函数上,框架会把匿名函数绑定到应用对象上,这么做可以在这个匿名函数中使用$this关键字引用重要的应用对象。例子如下:

<?php
class App
{
  protected $routes = array();
  protected $responseStatus = &#39;200 OK&#39;;
  protected $responseContentType = &#39;text/html&#39;;
  protected $responseBody = &#39;Hello world&#39;;

  public function addRoute($routePath, $routeCallback)
  {
    $this->routes[$routePath] = $routeCallback->bindTo($this, CLASS);//重点
  }

  public function dispatch($currentPath)
  {
    foreach ($this->routes as $routePath => $callback) {
      if ($routePath === $currentPath) {
        $callback();
      }
    }

    header('HTTP/1.1' . $this->responseStatus);
    header('Content-type:' . $this->responseContentType);
    header('Content-length' . mb_strlen($this->responseBody));
    echo $this->responseBody;
  }
}

第11行是重点所在,把路由回调绑定到了当前的App实例上。这么做能在回调函数中处理App实例的状态:

<?php
$app = new App();
$app->addRoute('/users/josh', function () {
  $this->responseContentType = 'application/json;charset=utf8';
  $this->responseBody = '{"name": "Josh"}';
});
$app->dispatch('/users/josh');

6. Zend OPcache

字节码缓存不是PHP的新特性,很多独立的扩展可以实现缓存。从PHP5.5.0开始,PHP内置了字节码缓存功能,名为Zend OPcache。

字节码缓存是什么
PHP是解释性语言,PHP解释器执行PHP脚本时会解析PHP脚本代码,把PHP代码编译成一系列Zend操作码,然后执行字节码。每次请求PHP文件都是这样,会消耗很多资源。字节码缓存能存储预先编译好的PHP字节码。这意味着,请求PHP脚本时,PHP解释器不用每次都读取、解析和编译PHP代码。这样能极大地提升应用的性能。

7. 内置的HTTP服务器

从PHP5.4.0起,PHP内置了Web服务器,这对众多使用Apache或nginx的php开发者来说,可能是个隐藏功能。不过,这个内置的服务器功能并不完善,不应该在生产环境中使用,但对本地开发来说是个便利的工具,可以用于快速预览一些框架和应用。

启动服务器

php -S localhost:4000

配置服务器

php -S localhost:8000 -c app/config/php.ini

路由器脚本
与Apache和nginx不同,它不支持.htaccess文件。因此,这个服务器很难使用多数流行的PHP框架中常见的前端控制器。PHP内置的服务器使用路由器脚本弥补了这个遗漏的功能。处理每个HTTP请求前,会先经过这个路由器脚本,如果结果为false,返回当前HTTP请求中引用的静态资源URI。

php -S localhost:8000 route.php

是否为内置的服务器

<?php
if (php_sapi_name() === 'cli-server') {
  // php 内置的web服务器
}

       

위 내용은 현재 PHP를 정말로 이해하고 있나요?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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