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!
Что включено на практике
Подробнее – https://narkologicheskaya-klinika-shchyolkovo0.ru/
Richardtag
31 Oct 25 at 1:45 am
kamagra oral jelly: Vita Homme – Kamagra pas cher France
RobertJuike
31 Oct 25 at 1:46 am
differenza tra Spedra e Viagra: differenza tra Spedra e Viagra – Spedra prezzo basso Italia
ClydeExamp
31 Oct 25 at 1:46 am
Для жителей Красноярска организованы два конфиденциальных пути. Первый — немаркированный выезд бригады на дом с заранее согласованными порогами связи. Второй — «тихий вход» в клинику через отдельную дверь, где приём проходит без очередей и «коридорных» разговоров. На первичном звонке администратор уточняет только клинически значимые факторы: частоту рвоты, наличие мочеиспускания, ЧСС/АД в покое, эпизоды спутанности, актуальные лекарства и аллергии. Биографические подробности и лишние вопросы не задаются — мы придерживаемся принципа минимально необходимой информации.
Детальнее – http://narkologicheskaya-klinika-v-krasnoyarske17.ru
RobertNop
31 Oct 25 at 1:46 am
рулонные жалюзи с электроприводом [url=https://rulonnye-shtory-s-elektroprivodom7.ru/]рулонные жалюзи с электроприводом[/url] .
rylonnie shtori s elektroprivodom_wkMl
31 Oct 25 at 1:47 am
купил диплом техникума и поступил в институт [url=www.frei-diplom8.ru]купил диплом техникума и поступил в институт[/url] .
Diplomi_dysr
31 Oct 25 at 1:47 am
рольшторы заказать [url=avtomaticheskie-rulonnye-shtory1.ru]avtomaticheskie-rulonnye-shtory1.ru[/url] .
avtomaticheskie rylonnie shtori_btMr
31 Oct 25 at 1:49 am
как купить диплом с проводкой [url=https://frei-diplom2.ru]как купить диплом с проводкой[/url] .
Diplomi_vyEa
31 Oct 25 at 1:49 am
Ресторан чистый после профессиональная дератизация.
уничтожение тараканов в мебели
KennethceM
31 Oct 25 at 1:49 am
рулонные шторы с электроприводом и дистанционным управлением [url=https://rulonnye-shtory-s-elektroprivodom7.ru]https://rulonnye-shtory-s-elektroprivodom7.ru[/url] .
rylonnie shtori s elektroprivodom_diMl
31 Oct 25 at 1:50 am
Отличная уничтожение тараканов , мастера приехали вовремя.
обработка от тараканов стоимость
Wernermog
31 Oct 25 at 1:50 am
comprare medicinali online legali: differenza tra Spedra e Viagra – farmacia viva
ClydeExamp
31 Oct 25 at 1:51 am
купить диплом техникума 1997 [url=www.frei-diplom11.ru/]купить диплом техникума 1997[/url] .
Diplomi_dqsa
31 Oct 25 at 1:51 am
купить диплом университета [url=https://www.rudik-diplom10.ru]купить диплом университета[/url] .
Diplomi_xfSa
31 Oct 25 at 1:51 am
купить диплом в тобольске [url=rudik-diplom11.ru]rudik-diplom11.ru[/url] .
Diplomi_ejMi
31 Oct 25 at 1:52 am
Полная версия материала тут: https://yerkramas.org/article/198491/pechat-nakleek-dlya-e-commerce-shtrix-kody–artikuly-i-nadezhnaya-zashhita-ot-vlagi
StevenGathe
31 Oct 25 at 1:52 am
Срочно нужна санобработка, тараканы достали!
дератизация предприятия
KennethceM
31 Oct 25 at 1:53 am
как купить диплом с занесением в реестр [url=https://www.frei-diplom5.ru]как купить диплом с занесением в реестр[/url] .
Diplomi_ioPa
31 Oct 25 at 1:53 am
купить диплом в махачкале [url=https://rudik-diplom4.ru]купить диплом в махачкале[/url] .
Diplomi_ibOr
31 Oct 25 at 1:54 am
кракен маркетплейс
kraken darknet market
JamesDaync
31 Oct 25 at 1:54 am
купить диплом в якутске [url=www.rudik-diplom2.ru/]купить диплом в якутске[/url] .
Diplomi_hgpi
31 Oct 25 at 1:55 am
электрические рулонные шторы на окна [url=https://rulonnye-shtory-s-elektroprivodom7.ru/]электрические рулонные шторы на окна[/url] .
rylonnie shtori s elektroprivodom_efMl
31 Oct 25 at 1:55 am
line2048.com – Color palette felt calming, nothing distracting, just focused, thoughtful design.
Lavonia Murcia
31 Oct 25 at 1:55 am
купить диплом в гатчине [url=https://rudik-diplom10.ru]https://rudik-diplom10.ru[/url] .
Diplomi_unSa
31 Oct 25 at 1:56 am
купить диплом о высшем образовании с занесением в реестр цена [url=frei-diplom2.ru]купить диплом о высшем образовании с занесением в реестр цена[/url] .
Diplomi_ezEa
31 Oct 25 at 1:57 am
купить диплом в новомосковске [url=www.rudik-diplom5.ru]купить диплом в новомосковске[/url] .
Diplomi_dqma
31 Oct 25 at 1:57 am
купить диплом с реестром киев [url=https://frei-diplom6.ru]https://frei-diplom6.ru[/url] .
Diplomi_vvOl
31 Oct 25 at 1:58 am
купить диплом геодезиста [url=https://www.rudik-diplom11.ru]купить диплом геодезиста[/url] .
Diplomi_gmMi
31 Oct 25 at 1:59 am
Борьба с обработка от клопов в отеле удалась, фирма молодцы.
уничтожение клопов горячим туманом
KennethceM
31 Oct 25 at 2:00 am
как купить легально диплом о высшем образовании [url=http://frei-diplom5.ru/]как купить легально диплом о высшем образовании[/url] .
Diplomi_gbPa
31 Oct 25 at 2:00 am
купить диплом в волжском [url=https://rudik-diplom1.ru/]купить диплом в волжском[/url] .
Diplomi_pzer
31 Oct 25 at 2:00 am
рольшторы заказать [url=https://rulonnye-shtory-s-elektroprivodom7.ru/]rulonnye-shtory-s-elektroprivodom7.ru[/url] .
rylonnie shtori s elektroprivodom_ndMl
31 Oct 25 at 2:01 am
купить дипломы о высшем с занесением [url=www.rudik-diplom4.ru/]купить дипломы о высшем с занесением[/url] .
Diplomi_gkOr
31 Oct 25 at 2:01 am
кракен vk6
kraken сайт
JamesDaync
31 Oct 25 at 2:02 am
The best is right here: https://www2.ic.uff.br/~medialab/2025/10/08/arbitrazh-trafika-chto-jeto-s-chego-nachat-2/
Terrellinfub
31 Oct 25 at 2:02 am
You need to be a part of a contest for one of the
most useful blogs on the internet. I most certainly will recommend this website!
شماره سمساری تهران 09121351135
31 Oct 25 at 2:02 am
Спасибо за выведение клопов! Всё чисто и безопасно.
дератизация цена
KennethceM
31 Oct 25 at 2:03 am
An outstanding share! I have just forwarded this onto a colleague who
has been doing a little homework on this. And he in fact bought me dinner because
I stumbled upon it for him… lol. So let me reword this….
Thank YOU for the meal!! But yeah, thanks for spending some time to discuss this
subject here on your internet site.
سفارش سئو سایت در تهران
31 Oct 25 at 2:03 am
купить диплом в элисте [url=rudik-diplom11.ru]купить диплом в элисте[/url] .
Diplomi_mvMi
31 Oct 25 at 2:03 am
DRINKIO порадовал внимательностью к клиентам и скоростью работы. Сайт удобный, заказ оформляется за пару минут, подтверждение приходит сразу. Курьеры всегда звонят заранее и приезжают точно в срок. Радует, что можно заказать в любое время суток, без ограничений по времени. Всё доставляется в аккуратной упаковке и в идеальном состоянии. Прекрасный сервис для тех, кому нужна доставка алкоголя на дом в Москве быстро и удобно, https://drinkio105.ru/
Ronaldskada
31 Oct 25 at 2:04 am
DRINKIO — лучший вариант для тех, кто ценит комфорт и оперативность. Доставка всегда вовремя, даже в позднее время. Операторы доброжелательные, всё объясняют и уточняют детали. Курьер приехал быстро, всё аккуратно упаковано. Очень доволен обслуживанием https://drinkio105.ru/catalog/category/vodka/
Ronaldskada
31 Oct 25 at 2:04 am
купить государственный диплом с занесением в реестр [url=frei-diplom5.ru]купить государственный диплом с занесением в реестр[/url] .
Diplomi_nrPa
31 Oct 25 at 2:04 am
Kamagra 100mg prix France: VitaHomme – kamagra oral jelly
RichardImmon
31 Oct 25 at 2:05 am
купить аттестат школы [url=http://rudik-diplom4.ru/]купить аттестат школы[/url] .
Diplomi_unOr
31 Oct 25 at 2:05 am
диплом медсестры с аккредитацией купить [url=http://frei-diplom13.ru/]диплом медсестры с аккредитацией купить[/url] .
Diplomi_vfkt
31 Oct 25 at 2:06 am
Je ne me lasse pas de Sugar Casino, il offre une experience dynamique. Il y a un eventail de titres captivants, avec des slots aux graphismes modernes. Il propulse votre jeu des le debut. Les agents repondent avec rapidite. Les retraits sont fluides et rapides, mais des offres plus consequentes seraient parfaites. Au final, Sugar Casino merite une visite dynamique. A mentionner la plateforme est visuellement captivante, permet une immersion complete. Particulierement interessant les paiements en crypto rapides et surs, propose des avantages uniques.
Ouvrir le site|
Nightbyteor6zef
31 Oct 25 at 2:07 am
Спасибо за выведение клопов! Всё чисто и безопасно.
уничтожение блох
KennethceM
31 Oct 25 at 2:08 am
Je suis enthousiaste a propos de Sugar Casino, ca invite a plonger dans le fun. Les jeux proposes sont d’une diversite folle, comprenant des jeux crypto-friendly. Le bonus d’inscription est attrayant. Le service est disponible 24/7. Les retraits sont ultra-rapides, bien que plus de promotions variees ajouteraient du fun. En resume, Sugar Casino merite un detour palpitant. A mentionner l’interface est simple et engageante, amplifie le plaisir de jouer. Particulierement fun les evenements communautaires pleins d’energie, propose des privileges sur mesure.
http://www.sugarcasinoappfr.com|
StarHeartix8zef
31 Oct 25 at 2:08 am
I was able to find good information from your blog articles.
teslabahis giriş
31 Oct 25 at 2:08 am
купить диплом в салавате [url=rudik-diplom3.ru]купить диплом в салавате[/url] .
Diplomi_mfei
31 Oct 25 at 2:09 am