PHP hook, building hooks in your application
Introduction
One of the real challenges in building any type of framework, core or application is making it possible for the developers to hook into the business logic at specific points. Since PHP is not event based, nor it works with interrupts you have to come up an alternative. 
The test case
Lets assume we are the main developers of a webshop framework. Programmers can use our framework to build complete webshops. Programmers can manage the orders that are placed on the webshop with the order class. The order class is part of our framework and we don’t want it to be extended by any programmer. However we don’t want to limit to programmers in their possibilities to hook into the orders process. 
For example programmers should be able to send an email to the webshopowner if an order changes from one specific delivery status to another. This functionality is not part of the default behavior in our framework and is custom for the progammers webshop implementation.
Like said before, PHP doesn’t provide interrupts or real events so we need to come up with another way to implement hooks into our application. Lets take a look at the observer pattern.
Implementing the Observer pattern
The observer pattern is a design-pattern that describes a way for objects to be notified to specific state-changes in objects of the application.
For the first implementation we can use SPL. The SPL provides in two simple objects:
SPLSubject
- attach (new observer to attach)
- detach (existing observer to detach)
- notify (notify all observers)
SPLObserver
- update (Called from the subject (i.e. when it’s value has changed).
iOrderRef = $iOrderRef;
		
		// Get order information from the database or an other resources
		$this->iStatus = Order::STATUS_SHIPPED;
	}
	
	/**
	 * Attach an observer
	 * 
	 * @param SplObserver $oObserver 
	 * @return void
	 */
	public function attach(SplObserver $oObserver)
	{
		$sHash = spl_object_hash($oObserver);
		if (isset($this->aObservers[$sHash])) {
			throw new Exception('Observer is already attached');
		}
		$this->aObservers[$sHash] = $oObserver;
	}
	/**
	 * Detach observer
	 * 
	 * @param SplObserver $oObserver 
	 * @return void
	 */
	public function detach(SplObserver $oObserver)
	{
		$sHash = spl_object_hash($oObserver);
		if (!isset($this->aObservers[$sHash])) {
			throw new Exception('Observer not attached');
		}
		unset($this->aObservers[$sHash]);
	}
	/**
	 * Notify the attached observers
	 * 
	 * @param string $sEvent, name of the event
	 * @param mixed $mData, optional data that is not directly available for the observers
	 * @return void
	 */
	public function notify()
	{
		foreach ($this->aObservers as $oObserver) {
			try {
				$oObserver->update($this);
			} catch(Exception $e) {
			}
		}
	}
	/**
	 * Add an order
	 * 
	 * @param array $aOrder 
	 * @return void
	 */
	public function delete()
	{
		$this->notify();
	}
	
	/**
	 * Return the order reference number
	 * 
	 * @return int
	 */
	public function getRef()
	{
		return $this->iOrderRef;
	}
	
	/**
	 * Return the current order status
	 * 
	 * @return int
	 */
	public function getStatus()
	{
		return $this->iStatus;
	}
	
	/**
	 * Update the order status
	 */
	public function updateStatus($iStatus)
	{
		$this->notify();
		// ...
		$this->iStatus = $iStatus;
		// ...
		$this->notify();
	}
}
/**
 * Order status handler, observer that sends an email to secretary
 * if the status of an order changes from shipped to delivered, so the
 * secratary can make a phone call to our customer to ask for his opinion about the service
 * 
 * @package Shop
 */
class OrderStatusHandler implements SplObserver
{
	/**
	 * Previous orderstatus
	 * @var int
	 */
	protected $iPreviousOrderStatus;
	/**
	 * Current orderstatus
	 * @var int
	 */
	protected $iCurrentOrderStatus;
	
	/**
	 * Update, called by the observable object order
	 * 
	 * @param Observable_Interface $oSubject
	 * @param string $sEvent
	 * @param mixed $mData 
	 * @return void
	 */
	public function update(SplSubject $oSubject)
	{
		if(!$oSubject instanceof Order) {
			return;
		}
		if(is_null($this->iPreviousOrderStatus)) {
			$this->iPreviousOrderStatus = $oSubject->getStatus();
		} else {
			$this->iCurrentOrderStatus = $oSubject->getStatus();
			if($this->iPreviousOrderStatus === Order::STATUS_SHIPPED && $this->iCurrentOrderStatus === Order::STATUS_DELIVERED) {
				$sSubject = sprintf('Order number %d is shipped', $oSubject->getRef());
				//mail('secratary@example.com', 'Order number %d is shipped', 'Text');
				echo 'Mail sended to the secratary to help her remember to call our customer for a survey.';
			}
		}
	}
}
$oOrder = new Order(26012011);
$oOrder->attach(new OrderStatusHandler());
$oOrder->updateStatus(Order::STATUS_DELIVERED);
$oOrder->delete();
?>
There are several problems with the implementation above. To most important disadvantage is that we have only one update method in our observer. In this update method we don’t know when and why we are getting notified, just that something happened. We should keep track of everything that happens in the subject. (Or use debug_backtrace… just joking, don’t even think about using it that way ever!).
Taking it a step further, events
Lets take a look at the next example, we will extend the Observer implementation with some an additional parameter for the eventname that occured.
Finishing up, optional data
iOrderRef = $iOrderRef;
		
		// Get order information from the database or something else...
		$this->iStatus = Order::STATUS_SHIPPED;
	}
	
	/**
	 * Attach an observer
	 * 
	 * @param Observer_Interface $oObserver 
	 * @return void
	 */
	public function attachObserver(Observer_Interface $oObserver)
	{
		$sHash = spl_object_hash($oObserver);
		if (isset($this->aObservers[$sHash])) {
			throw new Exception('Observer is already attached');
		}
		$this->aObservers[$sHash] = $oObserver;
	}
	/**
	 * Detach observer
	 * 
	 * @param Observer_Interface $oObserver 
	 * @return void
	 */
	public function detachObserver(Observer_Interface $oObserver)
	{
		$sHash = spl_object_hash($oObserver);
		if (!isset($this->aObservers[$sHash])) {
			throw new Exception('Observer not attached');
		}
		unset($this->aObservers[$sHash]);
	}
	/**
	 * Notify the attached observers
	 * 
	 * @param string $sEvent, name of the event
	 * @param mixed $mData, optional data that is not directly available for the observers
	 * @return void
	 */
	public function notifyObserver($sEvent, $mData=null)
	{
		foreach ($this->aObservers as $oObserver) {
			try {
				$oObserver->update($this, $sEvent, $mData);
			} catch(Exception $e) {
			}
		}
	}
	/**
	 * Add an order
	 * 
	 * @param array $aOrder 
	 * @return void
	 */
	public function add($aOrder = array())
	{
		$this->notifyObserver('onAdd');
	}
	
	/**
	 * Return the order reference number
	 * 
	 * @return int
	 */
	public function getRef()
	{
		return $this->iOrderRef;
	}
	
	/**
	 * Return the current order status
	 * 
	 * @return int
	 */
	public function getStatus()
	{
		return $this->iStatus;
	}
	
	/**
	 * Update the order status
	 */
	public function updateStatus($iStatus)
	{
		$this->notifyObserver('onBeforeUpdateStatus');
		// ...
		$this->iStatus = $iStatus;
		// ...
		$this->notifyObserver('onAfterUpdateStatus');
	}
}
/**
 * Order status handler, observer that sends an email to secretary
 * if the status of an order changes from shipped to delivered, so the
 * secratary can make a phone call to our customer to ask for his opinion about the service
 * 
 * @package Shop
 */
class OrderStatusHandler implements Observer_Interface
{
	protected $iPreviousOrderStatus;
	protected $iCurrentOrderStatus;
	
	/**
	 * Update, called by the observable object order
	 * 
	 * @param Observable_Interface $oObservable
	 * @param string $sEvent
	 * @param mixed $mData 
	 * @return void
	 */
	public function update(Observable_Interface $oObservable, $sEvent, $mData=null)
	{
		if(!$oObservable instanceof Order) {
			return;
		}
		
		switch($sEvent) {
			case 'onBeforeUpdateStatus':
				$this->iPreviousOrderStatus = $oObservable->getStatus();
				return;
			case 'onAfterUpdateStatus':
				$this->iCurrentOrderStatus = $oObservable->getStatus();
				
				if($this->iPreviousOrderStatus === Order::STATUS_SHIPPED && $this->iCurrentOrderStatus === Order::STATUS_DELIVERED) {
					$sSubject = sprintf('Order number %d is shipped', $oObservable->getRef());
					//mail('secratary@example.com', 'Order number %d is shipped', 'Text');
					echo 'Mail sended to the secratary to help her remember to call our customer for a survey.';
				}
		}
	}
}
$oOrder = new Order(26012011);
$oOrder->attachObserver(new OrderStatusHandler());
$oOrder->updateStatus(Order::STATUS_DELIVERED);
$oOrder->add();
?>
Now we are able to take action on different events that occur.
Disadvantages
Although this implementation works quite well there are some drawbacks. One of those drawbacks is that we need to dispatch an event in our framework, if we don’t programmers can’t hook into our application. Triggering events everywhere give us a small performance penalty however I do think this way of working gives the programmers a nice way to hook into your application on those spots that you want them to hook in.
Just for the record
Notice that this code is just an example and can still use some improvements, for example: each observer is initialized even it will maybe never be notified, therefore I suggest to make use of lazy in some cases for loading the objects. There are other systems to hook into an application, more to follow!
электрокарнизы [url=www.elektrokarniz-kupit.ru/]электрокарнизы[/url] .
elektrokarniz kypit_geea
31 Oct 25 at 4:46 pm
Нужна ликвидация? ликвидация компании черногория: добровольная ликвидация, банкротство, реорганизация. Подготовка документов, публикации, сверка с ФНС/ПФР, закрытие счетов. Сроки по договору, прозрачная смета, конфиденциальность, сопровождение до внесения записи в ЕГРЮЛ.
zakrit-kompaniu-878
31 Oct 25 at 4:47 pm
купить электрические рулонные шторы [url=https://avtomaticheskie-rulonnye-shtory1.ru/]avtomaticheskie-rulonnye-shtory1.ru[/url] .
avtomaticheskie rylonnie shtori_mmMr
31 Oct 25 at 4:48 pm
Нужна ликвидация? закрыть компанию в черногории: добровольная ликвидация, банкротство, реорганизация. Подготовка документов, публикации, сверка с ФНС/ПФР, закрытие счетов. Сроки по договору, прозрачная смета, конфиденциальность, сопровождение до внесения записи в ЕГРЮЛ.
zakrit-kompaniu-432
31 Oct 25 at 4:48 pm
Tasfiyeye mi ihtiyac?n?z var? https://www.sirket-kapatma.me/: borc analizi, yasal prosedurun secimi (istege bagl? veya iflas), alacakl?lar?n bildirimi, tasfiye bilancosu, sicilden silme. Son teslim tarihlerini ve sabit fiyat? netlestirin.
sirket-kapatma-477
31 Oct 25 at 4:49 pm
cheap medicines online Australia [url=http://aussiemedshubau.com/#]Aussie Meds Hub Australia[/url] Aussie Meds Hub
Hermanengam
31 Oct 25 at 4:50 pm
https://rosevillesohosuite.com/registraciya-na-melbet-2025/
RobertHindy
31 Oct 25 at 4:50 pm
кожаные жалюзи с электроприводом [url=www.elektricheskie-zhalyuzi97.ru]www.elektricheskie-zhalyuzi97.ru[/url] .
elektricheskie jaluzi_aeet
31 Oct 25 at 4:50 pm
рулонные шторы жалюзи на окна [url=www.avtomaticheskie-rulonnye-shtory1.ru]www.avtomaticheskie-rulonnye-shtory1.ru[/url] .
avtomaticheskie rylonnie shtori_ccMr
31 Oct 25 at 4:50 pm
Crowngreen Casino is a leading gaming platform that delivers thrilling experiences for users. Crowngreen Casino shines in the online gaming world and has achieved trust among gamers.
Every visitor at Crowngreen casino
can enjoy top-rated tournaments and take advantage of generous bonuses. Crowngreen Casino provides reliable gameplay, seamless transactions, and round-the-clock customer support for every gamer.
With , gamers unlock a diverse variety of casino games, including classic options. The platform focuses on user satisfaction and maintains a safe gaming environment.
Whether you are a beginner or an expert, Crowngreen provides exciting gameplay for everyone. Sign up at Crowngreen Casino today and experience thrilling games, fantastic bonuses, and a safe gaming environment.
Crowngreen Warneratort
31 Oct 25 at 4:50 pm
Tasfiyeye mi ihtiyac?n?z var? https://www.sirket-kapatma.me/: borc analizi, yasal prosedurun secimi (istege bagl? veya iflas), alacakl?lar?n bildirimi, tasfiye bilancosu, sicilden silme. Son teslim tarihlerini ve sabit fiyat? netlestirin.
sirket-kapatma-24
31 Oct 25 at 4:50 pm
рулонные шторы на пластиковые окна с электроприводом [url=www.rulonnye-shtory-s-elektroprivodom7.ru/]www.rulonnye-shtory-s-elektroprivodom7.ru/[/url] .
rylonnie shtori s elektroprivodom_rxMl
31 Oct 25 at 4:51 pm
рулонные шторы на электроприводе [url=http://www.avtomaticheskie-rulonnye-shtory77.ru]рулонные шторы на электроприводе[/url] .
avtomaticheskie rylonnie shtori_itPa
31 Oct 25 at 4:51 pm
https://bics-llc.net/registraciya-v-melbet-2025/
JustinAcecy
31 Oct 25 at 4:51 pm
В Ростове-на-Дону клиника «ЧСП№1» предлагает профессиональный вывод из запоя. Услуга доступна на дому и в стационаре, а также включает капельницу от похмелья. Все процедуры проводятся анонимно и круглосуточно.
Исследовать вопрос подробнее – [url=https://vyvod-iz-zapoya-rostov11.ru/]вывод из запоя цена ростов-на-дону[/url]
RobertSloff
31 Oct 25 at 4:51 pm
legitimate pharmacy sites UK: online pharmacy – affordable medications UK
HaroldSHems
31 Oct 25 at 4:52 pm
электрические рулонные шторы купить [url=www.avtomaticheskie-rulonnye-shtory77.ru]электрические рулонные шторы купить[/url] .
avtomaticheskie rylonnie shtori_kzPa
31 Oct 25 at 4:53 pm
https://pro777slot.org/melbet-zerkalo-obzor-2025/
ThomasMuh
31 Oct 25 at 4:53 pm
https://babu88888.com/
1-gocasino
31 Oct 25 at 4:54 pm
online pharmacy [url=https://safemedsguide.shop/#]SafeMedsGuide[/url] trusted online pharmacy USA
Hermanengam
31 Oct 25 at 4:54 pm
In Crazy Time good, the wheel is divided into 54 segments Crazy Time bangladesh
Robinkanty
31 Oct 25 at 4:55 pm
https://www.wikiacademy.com.tn/skachat-melbet-sloty-2025/
RobertHindy
31 Oct 25 at 4:56 pm
гардина с электроприводом [url=https://elektrokarniz-kupit.ru]https://elektrokarniz-kupit.ru[/url] .
elektrokarniz kypit_ztea
31 Oct 25 at 4:56 pm
http://aussiemedshubau.com/# pharmacy discount codes AU
Haroldovaph
31 Oct 25 at 4:57 pm
рулонные шторы автоматические [url=http://avtomaticheskie-rulonnye-shtory1.ru/]рулонные шторы автоматические[/url] .
avtomaticheskie rylonnie shtori_hwMr
31 Oct 25 at 4:57 pm
натяжные потолки нижний новгород цена [url=www.natyazhnye-potolki-nizhniy-novgorod-1.ru]натяжные потолки нижний новгород цена[/url] .
natyajnie potolki nijnii novgorod_xwma
31 Oct 25 at 4:58 pm
https://riosantohotel.com.br/2025/10/11/kazino-melbet-registraciya-2025/
JeromeSix
31 Oct 25 at 4:58 pm
https://anime.format.style/2025/10/11/promokod-melbet-besplatno-2025/
ThomasMuh
31 Oct 25 at 4:59 pm
Hi, Neat post. There is an issue with your website in internet
explorer, could test this? IE still is the
marketplace chief and a good component of other folks will miss your fantastic writing due to this problem.
ankara kürtaj
31 Oct 25 at 4:59 pm
рольшторы с электроприводом [url=www.rulonnye-shtory-s-elektroprivodom7.ru]рольшторы с электроприводом[/url] .
rylonnie shtori s elektroprivodom_jsMl
31 Oct 25 at 5:00 pm
потолочкин ру нижний новгород [url=www.natyazhnye-potolki-nizhniy-novgorod-1.ru]www.natyazhnye-potolki-nizhniy-novgorod-1.ru[/url] .
natyajnie potolki nijnii novgorod_bhma
31 Oct 25 at 5:01 pm
рулонные шторы это [url=https://avtomaticheskie-rulonnye-shtory77.ru/]https://avtomaticheskie-rulonnye-shtory77.ru/[/url] .
avtomaticheskie rylonnie shtori_wqPa
31 Oct 25 at 5:01 pm
The site offers a variety of gaming services and over 13,000 casino games 1win casino argentina
JeremyRot
31 Oct 25 at 5:01 pm
На страницах блога представлены обзоры игровых ПК, с подробными рекомендациями по выбору видеокарт, процессоров и других комплектующих. Пользователи найдут обзоры готовых сборок и советы по апгрейду, которые помогут собрать или выбрать производительный игровой компьютер для современных игр.
game-computers-169
31 Oct 25 at 5:01 pm
рулонные шторы с направляющими купить [url=http://www.avtomaticheskie-rulonnye-shtory1.ru]http://www.avtomaticheskie-rulonnye-shtory1.ru[/url] .
avtomaticheskie rylonnie shtori_hxMr
31 Oct 25 at 5:02 pm
J’adore le dynamisme de Sugar Casino, on y trouve une energie contagieuse. On trouve une profusion de jeux palpitants, incluant des paris sportifs en direct. Il propulse votre jeu des le debut. Les agents sont toujours la pour aider. Les transactions sont fiables et efficaces, de temps a autre des offres plus importantes seraient super. En resume, Sugar Casino est un incontournable pour les joueurs. Notons egalement le site est rapide et style, apporte une touche d’excitation. Egalement super les evenements communautaires dynamiques, offre des bonus exclusifs.
Continuer ici|
vibeknightan1zef
31 Oct 25 at 5:03 pm
электрокарнизы для штор [url=https://elektrokarniz-kupit.ru/]электрокарнизы для штор[/url] .
elektrokarniz kypit_eoea
31 Oct 25 at 5:04 pm
https://www.comex-andina.com/melbet-promokody-2025-bonusy-igrat/
JustinAcecy
31 Oct 25 at 5:05 pm
Ich freue mich riesig uber Cat Spins Casino, es ist ein Hotspot fur Spielspa?. Die Auswahl ist atemberaubend vielfaltig, mit klassischen Tischspielen. Er gibt Ihnen einen Kickstart. Der Support ist zuverlassig und hilfsbereit. Der Prozess ist unkompliziert, dennoch zusatzliche Freispiele waren ein Bonus. Zusammenfassend, Cat Spins Casino ist ein absolutes Highlight. Nebenbei die Seite ist schnell und ansprechend, zum Weiterspielen animiert. Ein weiteres Highlight sind die zuverlassigen Krypto-Zahlungen, die Teilnahme fordern.
Online gehen|
brightbyteex4zef
31 Oct 25 at 5:06 pm
Есть ли курсы игры на пианино для тех, кто никогда не учился музыке? https://uroki-fortepiano.ru/
EdwardRon
31 Oct 25 at 5:07 pm
На сайте собраны обзоры игровых компьютеров, где подробно рассказывается о совместимости комплектующих, подборе оптимальных конфигураций и систем охлаждения. Материалы помогают пользователям ориентироваться в разнообразии моделей и выбрать подходящую сборку для своих игровых целей.
game-computers-961
31 Oct 25 at 5:08 pm
электрические гардины [url=https://www.elektrokarniz777.ru]электрические гардины[/url] .
elektrokarniz _sdsr
31 Oct 25 at 5:08 pm
купить электрические рулонные шторы [url=www.avtomaticheskie-rulonnye-shtory1.ru]www.avtomaticheskie-rulonnye-shtory1.ru[/url] .
avtomaticheskie rylonnie shtori_vbMr
31 Oct 25 at 5:09 pm
Доброго!
Постоянный виртуальный номер для смс решает проблему с регистрацией. Купите постоянный виртуальный номер для смс и используйте его без ограничений.
Полная информация по ссылке – [url=https://rusamara.com/21200-chto-takoe-virtualnye-nomera-i-kak-oni-rabotajut.html]купить виртуальный номер телефона навсегда[/url]
виртуальный номер, виртуальный номер навсегда, купить виртуальный номер для смс навсегда
постоянный виртуальный номер для смс, купить постоянный виртуальный номер, виртуальный номер
Удачи и комфорта в общении!!
Nomerpl
31 Oct 25 at 5:09 pm
Оптимальная сборка игровых ПК напрямую влияет на плавность игрового процесса, стабильность системы и ее долговечность. На страницах блога можно найти подробные обзоры, советы по выбору комплектующих и рекомендации, которые помогут подобрать подходящую конфигурацию под разные задачи.
game-computers-82
31 Oct 25 at 5:10 pm
https://pinocraft.com/melbet-kazino-slots-2025/
RobertHindy
31 Oct 25 at 5:10 pm
организация онлайн трансляций под ключ [url=https://zakazat-onlayn-translyaciyu5.ru/]организация онлайн трансляций под ключ[/url] .
zakazat onlain translyaciu_aomr
31 Oct 25 at 5:11 pm
рулонные жалюзи москва [url=http://www.avtomaticheskie-rulonnye-shtory1.ru]рулонные жалюзи москва[/url] .
avtomaticheskie rylonnie shtori_ptMr
31 Oct 25 at 5:12 pm
компания потолочник [url=https://natyazhnye-potolki-nizhniy-novgorod-1.ru/]https://natyazhnye-potolki-nizhniy-novgorod-1.ru/[/url] .
natyajnie potolki nijnii novgorod_plma
31 Oct 25 at 5:12 pm
https://tstzaneen.co.za/blog/melbet-kazino-vhod-2025/
ThomasMuh
31 Oct 25 at 5:13 pm