Таким образом, я практиковал arround с некоторыми oop и пытался создать базовый класс пользователей, я попробовал создать свою систему, чтобы на каждой странице я мог просто включить config.php и все, что мне нужно загрузить, но по какой-то причине, как только я попробуйте войти в систему, это бросает мне эту ошибку:
Неустранимая ошибка: вызов функции-члена LogIn () для не-объекта
config.php:
<?php session_start(); // Mysql details $username = "torlolol"; $password = "torlolol"; $hostname = "localhost"; $database = "torlolol"; // Autoloads classes when needed function __autoload($class_name) { include 'Classes/' . $class_name . '.php'; } $db = new DBConnector($hostname, $username, $password); // If there is no user in the session stored make a new one otherwise unserialize it if(empty($_SESSION['user'])) $user = new User(); else $user = unserialize($_SESSION['user']); function onExit() { //When exiting the page serialize the user object and store it in the session $_SESSION['user'] = serialize($user); } register_shutdown_function("onExit") ?>
login.php
<?php include "config.php"; global $user; $user->LogIn(); ?>
Класс пользователя:
class User { public $Logged = false; public function LogIn() { $this->Logged = true; } public function LogOut() { $this->Logged = false; } } ?>
index.php:
<?php include "config.php"; global $user; if ($user->Logged != true) echo '<form action="login.php" method="post">Username :<input type="text" name="username" /> Password :<input type="password" name="password" /> <input type="submit" value="Login" /></form>'; else echo '<form action="logout.php" method="post"><input style="submit" value="Logout" /></form>';
Так почему он выбрасывает эту ошибку? И почему это не происходит в индексном файле: S?>
Вы должны включить файлы классов перед началом сеанса, иначе сериализованные объекты не будут загружены правильно. К сожалению…
Если вы несериализуете объект, какой класс еще не объявлен, он не инициализируется в __PHP_Incomplete_Class
, который, очевидно, не имеет метода LogIn
. Поэтому вам нужно либо включить файл вручную, либо зарегистрировать unserialize_callback_func
:
ini_set('unserialize_callback_func', '__autoload');
Попробуйте удалить,
global $user;
Потому что когда вы включаете файл, $ user будет автоматически включаться в ваш файл.
в файле config.php вы регистрируете функцию выключения, которая устанавливает $ _SESSION ['Пользователь'] с экземпляром User (переменная $ user) … но из-за того, как вы управляете неустановленными классами (используя глобальную переменную), вы необходимо обновить global $user;
в функции onExit
.
function onExit() { //When exiting the page serialize the user object and store it in the session global $user; $_SESSION['user'] = serialize($user); }
Я лично рекомендую иметь основной глобальный класс, который обрабатывает экземпляры классов, которые вам нужны во всем приложении.
изменить: посмотрите мой ответ здесь
Вот класс, который позволяет отслеживать изменения глобальных переменных:
abstract class CSTReportDelegate { abstract public function emitVariableChange( $variableName, $oldValue, $newValue ); abstract public function emitVariableSetNew( $variableName, $newValue ); } class CSTSimpleReportDelegate extends CSTReportDelegate { public function emitVariableChange( $variableName, $oldValue, $newValue ) { echo '<br />[global/change] '. $variableName . ' : ' . print_r( $oldValue, true ) . ' → ' . print_r( $newValue, true ); } public function emitVariableSetNew( $variableName, $newValue ) { echo '<br />[global/init] '. $variableName . ' → ' . print_r( $newValue, TRUE ); } } class CSysTracer { static protected $reportDelegate; static private $globalState = array(); static private $traceableGlobals = array(); static private $globalTraceEnabled = FALSE; static public function setReportDelegate( CSTReportDelegate $aDelegate ) { self::$reportDelegate = $aDelegate; } static public function start( ) { register_tick_function ( array( 'CSysTracer', 'handleTick' ) ); } static public function stop() { unregister_tick_function( array( 'CSysTracer', 'handleTick' ) ); } static public function evalAndTrace( $someStatement ) { declare( ticks = 1 ); { self::start(); eval( $someStatement ); self::stop(); } } static public function addTraceableGlobal( $varName ) { if ( is_array( $varName )) { foreach( $varName as $singleName ) { self::addTraceableGlobal( $singleName ); } return; } self::$traceableGlobals[ $varName ] = $varName; } static public function removeTraceableGlobal( $varName ) { unset( self::$traceableGlobals[ $varName ] ); } /** * Main function called at each tick. Calls those functions, which * really perform the checks. * */ static public function handleTick( ) { if ( TRUE === self::$globalTraceEnabled ) { self::traceGlobalVariable(); } } static public function enableGlobalsTrace() { self::$globalTraceEnabled = TRUE; } static public function disableGlobalsTrace() { self::$globalTraceEnabled = FALSE; } static public function traceGlobalVariable( ) { foreach( self::$traceableGlobals as $aVarname ) { if ( ! isset( $GLOBALS[ $aVarname ] )) { continue; } if ( ! isset( self::$globalState[ $aVarname ] ) ) { self::$reportDelegate->emitVariableSetNew( $aVarname, $GLOBALS[ $aVarname ] ); self::$globalState[ $aVarname ] = $GLOBALS[ $aVarname ]; continue; } if ( self::$globalState[ $aVarname ] !== $GLOBALS[ $aVarname ]) { self::$reportDelegate->emitVariableChange( $aVarname, self::$globalState[ $aVarname ], $GLOBALS[ $aVarname ] ); } self::$globalState[ $aVarname ] = $GLOBALS[ $aVarname ]; } } }
Используйте его так:
CSysTracer::addTraceableGlobal( array( 'foo', 'bar' )); CSysTracer::setReportDelegate( new CSTSimpleReportDelegate() ); CSysTracer::enableGlobalsTrace(); CSysTracer::start(); declare( ticks = 1 ); $foo = 10; echo "<br>First output"; $foo *= $foo; $foo = 'bar'; echo "<br>Next output"; $otherFoo = array (1,2,3); $bar = 23; echo "<br>Finished!"; CSysTracer::stop();
Выход
[global/init] foo → 10 Hi! [global/change] foo : 10 → 100 [global/change] foo : 100 → bar Gotta run [global/change] foo : bar → Array ( [0] => 1 [1] => 2 [2] => 3 ) [global/init] bar → 23 Bye!
Хотя эта версия CSysTracer отслеживает изменения в глобальных символах, вы можете создать вариант traceGlobalVariable () для отслеживания изменений для переменных сеанса или, например, методов-вызовов.