пространства имен — PHP при удалении автозагрузчика Composer не справляется с заданием

Я слежу за книгой Лопеса «Изучение PHP 7» и в гл. 5 на MVC Я был бы более чем признателен, чтобы узнать, что здесь не так.

Когда в init.php я закомментирую — см. Комментарий из девяти строк в init.php — функция автозагрузчика, которая больше не нужна, пишет Лопес, я получаю эту ошибку:

Неустранимая ошибка PHP: необработанная ошибка: класс ‘Bookstore \ Utils \ Config’ не найден в /home/petr/Documents/workspace/bookstore/init.php:25
Трассировки стека:

Строка 25 в init.php, которая полностью показана ниже,

$dbConfig = Config::getInstance()->get('db');

(При установке PHP, MySQL, Composer и Twig я не использовал Vagrant [установил PHP, MySQL и Apache в Ubuntu перед началом работы с этой книгой и добавил Composer w. Twig через Synaptic сегодня]. Я использую веб-сервер PHP по умолчанию развивает этот проект.)

Структура проекта выглядит следующим образом:

введите описание изображения здесь

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];

}

}

Интересно, как заставить проект работать с функцией autoloader удалено из init.php. Спасибо.

0

Решение

Вам нужен автозагрузчик для волшебно загрузить классы. Как вы определили composer.json с вашими зависимостями и вашим правилом автозагрузки PSR-4, я бы порекомендовал использовать автозагрузчик Composers.

Ваш composer.json В конфигурации есть небольшая ошибка: автозагрузчик PSR-4 пытается загрузить классы внутри Bookstore пространство имен из подпапки с именем bookstore, Эти классы помещаются в корневой каталог вашего проекта. Поэтому автозагрузчик должен указывать на этот каталог, оставляя путь к каталогу пустым:

{
"require": {
"monolog/monolog": "^1.17",
"twig/twig": "^1.23"},
"autoload": {
"psr-4": {
"Bookstore\\": ""}
}
}
1

Другие решения

Других решений пока нет …

По вопросам рекламы ammmcru@yandex.ru