PHP 删除自动加载器后,Composer 无法完成作业
PHP upon removing autoloader the Composer fails the job
我正在关注 Lopez 的 "Learning PHP 7" 书和 Ch. 5 关于 MVC 我将非常感激了解这里的错误。
当在 init.php 中时,我注释掉了 - 请参阅 init.php 中的九行注释 - 自动加载器功能,不再需要它,Lopez 写道,我收到此错误:
PHP Fatal error: Uncaught Error: Class 'Bookstore\Utils\Config' not found in /home/petr/Documents/workspace/bookstore/init.php:25
Stack trace:
...
下面完整显示的init.php中的第25行是
$dbConfig = Config::getInstance()->get('db');
(在安装 PHP、MySQL、Composer 和 Twig 时,我没有使用 Vagrant [已在 [=53] 上安装了 PHP、MySQL 和 Apache =] 在开始学习这本书之前,今天通过 Synaptic 添加了 Composer w.Twig。我使用 PHP 的默认 Web 服务器开发这个项目。)
项目结构是这样的:
init.php 是这样的:
<?php
use Bookstore\Domain\Customer\Basic;
use Bookstore\Domain\Customer\Premium;
use Bookstore\Domain\Customer\CustomerFactory;
use Bookstore\Domain\Customer;
use Bookstore\Domain\Payer;
use Bookstore\Domain\Person;
use Bookstore\Domain\Book;
use Bookstore\Utils\Config;
use Bookstore\Utils\Unique;
use Bookstore\Exceptions\InvalidIdException;
use Bookstore\Exceptions\ExceededMaxAllowedException;
// function autoloader($classname) {
// $lastSlash = strpos($classname, '\') + 1;
// $classname = substr($classname, $lastSlash);
// $directory = str_replace('\', '/', $classname);
// $filename = __DIR__ . '/' . $directory . '.php';
// require_once($filename);
// }
//
// spl_autoload_register('autoloader');
$dbConfig = Config::getInstance()->get('db');
$db = new PDO(
'mysql:host=127.0.0.1;dbname=bookstore',
$dbConfig['user'],
$dbConfig['password']
);
$db->setAttribute(PDO::ATTR_DEFAULT_FETCH_MODE, PDO::FETCH_ASSOC);
function addBook(int $id, int $amount = 1): void {
$db = new PDO(
'mysql:host=127.0.0.1;dbname=bookstore',
'root',
''
);
$query = 'UPDATE book SET stock = stock + :n WHERE id = :id';
$statement = $db->prepare($query);
$statement->bindValue('id', $id);
$statement->bindValue('n', $amount);
if (!$statement->execute()) {
throw new Exception($statement->errorInfo()[2]);
}
}
function addSale(int $userId, array $bookIds): void {
$db = new PDO(
'mysql:host=127.0.0.1;dbname=bookstore',
'root',
''
);
$db->beginTransaction();
try {
$query = 'INSERT INTO sale (customer_id, date) '
. 'VALUES(:id, NOW())';
$statement = $db->prepare($query);
if (!$statement->execute(['id' => $userId])) {
throw new Exception($statement->errorInfo()[2]);
}
$saleId = $db->lastInsertId();
$query = 'INSERT INTO sale_book (book_id, sale_id) '
. 'VALUES (:book, :sale)';
$statement = $db->prepare($query);
$statement->bindValue('sale', $saleId);
foreach ($bookIds as $bookId) {
$statement->bindValue('book', $bookId);
if (!$statement->execute()) {
throw new Exception($statement->errorInfo()[2]);
}
}
$db->commit();
}
catch (Exception $e) {
$db->rollBack();
throw $e;
}
}
try {
addSale(1, [3, 7]);
}
catch (Exception $e) {
echo 'Error adding sale: ' . $e->getMessage();
}
composer.json 是这个文件:
{
"require": {
"monolog/monolog": "^1.17",
"twig/twig": "^1.23"
},
"autoload": {
"psr-4": {
"Bookstore\": "bookstore"
}
}
}
这是 Config.php:
<?php
namespace Bookstore\Utils;
use Bookstore\Exceptions\NotFoundException;
class Config {
private static $data; // private $data;
private static $instance;
private function __construct() { // public function __construct() {
$json = file_get_contents(__DIR__ . '/../config/app.json');
self::$data = json_decode($json, true); // $this->data = json_decode($json, true);
}
public static function getInstance() {
if (self::$instance == null) {
self::$instance = new Config();
}
return self::$instance;
}
public static function get($key) { // public function get($key) {
if (!isset(self::$data[$key])) { // if (!isset($this->data[$key])) {
throw new NotFoundException("Key $key not in config.");
}
return self::$data[$key]; // return $this->data[$key];
}
}
我想知道如何使项目与从 init.php 中删除的功能 autoloader
一起工作。谢谢。
您需要一个自动加载器才能神奇地加载类。由于您已使用依赖项和 PSR-4 自动加载规则定义 composer.json
,因此我建议使用 Composers 自动加载器。
您的 composer.json
配置有一个小错误:PSR-4 自动加载器尝试从名为 bookstore
的子文件夹中加载 Bookstore
命名空间中的 类 .这些 类 被放置在您的项目的根目录中。因此,自动加载器必须指向该目录,而不是将目录路径留空:
{
"require": {
"monolog/monolog": "^1.17",
"twig/twig": "^1.23"
},
"autoload": {
"psr-4": {
"Bookstore\": ""
}
}
}
我正在关注 Lopez 的 "Learning PHP 7" 书和 Ch. 5 关于 MVC 我将非常感激了解这里的错误。
当在 init.php 中时,我注释掉了 - 请参阅 init.php 中的九行注释 - 自动加载器功能,不再需要它,Lopez 写道,我收到此错误:
PHP Fatal error: Uncaught Error: Class 'Bookstore\Utils\Config' not found in /home/petr/Documents/workspace/bookstore/init.php:25 Stack trace: ...
下面完整显示的init.php中的第25行是
$dbConfig = Config::getInstance()->get('db');
(在安装 PHP、MySQL、Composer 和 Twig 时,我没有使用 Vagrant [已在 [=53] 上安装了 PHP、MySQL 和 Apache =] 在开始学习这本书之前,今天通过 Synaptic 添加了 Composer w.Twig。我使用 PHP 的默认 Web 服务器开发这个项目。)
项目结构是这样的:
init.php 是这样的:
<?php
use Bookstore\Domain\Customer\Basic;
use Bookstore\Domain\Customer\Premium;
use Bookstore\Domain\Customer\CustomerFactory;
use Bookstore\Domain\Customer;
use Bookstore\Domain\Payer;
use Bookstore\Domain\Person;
use Bookstore\Domain\Book;
use Bookstore\Utils\Config;
use Bookstore\Utils\Unique;
use Bookstore\Exceptions\InvalidIdException;
use Bookstore\Exceptions\ExceededMaxAllowedException;
// function autoloader($classname) {
// $lastSlash = strpos($classname, '\') + 1;
// $classname = substr($classname, $lastSlash);
// $directory = str_replace('\', '/', $classname);
// $filename = __DIR__ . '/' . $directory . '.php';
// require_once($filename);
// }
//
// spl_autoload_register('autoloader');
$dbConfig = Config::getInstance()->get('db');
$db = new PDO(
'mysql:host=127.0.0.1;dbname=bookstore',
$dbConfig['user'],
$dbConfig['password']
);
$db->setAttribute(PDO::ATTR_DEFAULT_FETCH_MODE, PDO::FETCH_ASSOC);
function addBook(int $id, int $amount = 1): void {
$db = new PDO(
'mysql:host=127.0.0.1;dbname=bookstore',
'root',
''
);
$query = 'UPDATE book SET stock = stock + :n WHERE id = :id';
$statement = $db->prepare($query);
$statement->bindValue('id', $id);
$statement->bindValue('n', $amount);
if (!$statement->execute()) {
throw new Exception($statement->errorInfo()[2]);
}
}
function addSale(int $userId, array $bookIds): void {
$db = new PDO(
'mysql:host=127.0.0.1;dbname=bookstore',
'root',
''
);
$db->beginTransaction();
try {
$query = 'INSERT INTO sale (customer_id, date) '
. 'VALUES(:id, NOW())';
$statement = $db->prepare($query);
if (!$statement->execute(['id' => $userId])) {
throw new Exception($statement->errorInfo()[2]);
}
$saleId = $db->lastInsertId();
$query = 'INSERT INTO sale_book (book_id, sale_id) '
. 'VALUES (:book, :sale)';
$statement = $db->prepare($query);
$statement->bindValue('sale', $saleId);
foreach ($bookIds as $bookId) {
$statement->bindValue('book', $bookId);
if (!$statement->execute()) {
throw new Exception($statement->errorInfo()[2]);
}
}
$db->commit();
}
catch (Exception $e) {
$db->rollBack();
throw $e;
}
}
try {
addSale(1, [3, 7]);
}
catch (Exception $e) {
echo 'Error adding sale: ' . $e->getMessage();
}
composer.json 是这个文件:
{
"require": {
"monolog/monolog": "^1.17",
"twig/twig": "^1.23"
},
"autoload": {
"psr-4": {
"Bookstore\": "bookstore"
}
}
}
这是 Config.php:
<?php
namespace Bookstore\Utils;
use Bookstore\Exceptions\NotFoundException;
class Config {
private static $data; // private $data;
private static $instance;
private function __construct() { // public function __construct() {
$json = file_get_contents(__DIR__ . '/../config/app.json');
self::$data = json_decode($json, true); // $this->data = json_decode($json, true);
}
public static function getInstance() {
if (self::$instance == null) {
self::$instance = new Config();
}
return self::$instance;
}
public static function get($key) { // public function get($key) {
if (!isset(self::$data[$key])) { // if (!isset($this->data[$key])) {
throw new NotFoundException("Key $key not in config.");
}
return self::$data[$key]; // return $this->data[$key];
}
}
我想知道如何使项目与从 init.php 中删除的功能 autoloader
一起工作。谢谢。
您需要一个自动加载器才能神奇地加载类。由于您已使用依赖项和 PSR-4 自动加载规则定义 composer.json
,因此我建议使用 Composers 自动加载器。
您的 composer.json
配置有一个小错误:PSR-4 自动加载器尝试从名为 bookstore
的子文件夹中加载 Bookstore
命名空间中的 类 .这些 类 被放置在您的项目的根目录中。因此,自动加载器必须指向该目录,而不是将目录路径留空:
{
"require": {
"monolog/monolog": "^1.17",
"twig/twig": "^1.23"
},
"autoload": {
"psr-4": {
"Bookstore\": ""
}
}
}