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=http://www.rudik-diplom10.ru]купить диплом в клинцах[/url] .
Diplomi_xoSa
28 Oct 25 at 12:02 pm
Spot on with this write-up, I really think this site needs much more attention. I’ll probably be returning to
read more, thanks for the information!
Clair Invion
28 Oct 25 at 12:02 pm
ИТ-отрасль не знает кризисов кракен онион kraken маркетплейс зеркало кракен ссылка кракен даркнет
RichardPep
28 Oct 25 at 12:03 pm
Me encantó este contenido sobre las casino tragamonedas más
destacadas en Pin-Up México. Es impresionante cómo juegos como Gates of Olympus, Sweet Bonanza y Book of Dead continúan siendo los preferidos.
Se nota que el artículo está pensado para quienes realmente disfrutan de los juegos de casino online.
Vale la pena visitar la publicación original y conocer en detalle por qué estos títulos
son tan jugados en 2025.
Se agradece ver una mezcla entre títulos nostálgicos y nuevas propuestas en el mercado mexicano de apuestas.
Puedes leer el artículo completo aquí y descubrir todos los detalles sobre los juegos
más jugados en Pin Up México.
read more
28 Oct 25 at 12:03 pm
гидроизоляция подвала снаружи цены [url=www.gidroizolyaciya-cena-7.ru]www.gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_qhSi
28 Oct 25 at 12:03 pm
hurrah! Damn me,オナホ フィギュアwon,
ラブドール
28 Oct 25 at 12:03 pm
купить диплом в анапе [url=http://rudik-diplom3.ru]купить диплом в анапе[/url] .
Diplomi_csei
28 Oct 25 at 12:03 pm
купить диплом в первоуральске [url=http://rudik-diplom5.ru]http://rudik-diplom5.ru[/url] .
Diplomi_wxma
28 Oct 25 at 12:04 pm
диплом реестр купить [url=http://www.frei-diplom5.ru]диплом реестр купить[/url] .
Diplomi_giPa
28 Oct 25 at 12:04 pm
купить диплом в реестр [url=http://frei-diplom2.ru]купить диплом в реестр[/url] .
Diplomi_tzEa
28 Oct 25 at 12:04 pm
Дополнительно – доменное
имя, хостинг и SSL-сертификат, также есть
возможности для СЕО и расширения функционала с помощью интеграций.
www.doka.borda.ru/
28 Oct 25 at 12:05 pm
можно ли купить диплом медсестры [url=http://frei-diplom15.ru]можно ли купить диплом медсестры[/url] .
Diplomi_kboi
28 Oct 25 at 12:05 pm
the Great Whale,ラブドール 激安the True Whale,
ラブドール
28 Oct 25 at 12:06 pm
наркологический центр [url=www.narkologicheskaya-klinika-27.ru/]наркологический центр[/url] .
narkologicheskaya klinika_nypl
28 Oct 25 at 12:06 pm
купить диплом в ялте [url=http://rudik-diplom13.ru]купить диплом в ялте[/url] .
Diplomi_dqon
28 Oct 25 at 12:07 pm
kraken онлайн
kraken vk3
Henryamerb
28 Oct 25 at 12:07 pm
диплом колледжа 2016 купить [url=www.frei-diplom8.ru]www.frei-diplom8.ru[/url] .
Diplomi_zwsr
28 Oct 25 at 12:08 pm
Получить диплом любого университета можем помочь. Купить диплом бакалавра в Магнитогорске – [url=http://diplomybox.com/kupit-diplom-bakalavra-v-magnitogorske/]diplomybox.com/kupit-diplom-bakalavra-v-magnitogorske[/url]
Cazrvsn
28 Oct 25 at 12:08 pm
ремонт в подвале [url=http://gidroizolyaciya-cena-7.ru]http://gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_inSi
28 Oct 25 at 12:09 pm
Эта обзорная заметка содержит ключевые моменты и факты по актуальным вопросам. Она поможет читателям быстро ориентироваться в теме и узнать о самых важных аспектах сегодня. Получите краткий курс по современной информации и оставайтесь в курсе событий!
Подробнее тут – https://eco-sante.org/bonjour-tout-le-monde
HermanApomi
28 Oct 25 at 12:09 pm
купить диплом маркетолога [url=https://rudik-diplom4.ru]купить диплом маркетолога[/url] .
Diplomi_nbOr
28 Oct 25 at 12:09 pm
learnshareandgrow – I found the site very motivating and full of thoughtful insights.
Kyong Cordas
28 Oct 25 at 12:11 pm
где купить диплом колледжа в астрахани [url=https://www.frei-diplom8.ru]https://www.frei-diplom8.ru[/url] .
Diplomi_zisr
28 Oct 25 at 12:14 pm
кракен сайт
кракен vpn
Henryamerb
28 Oct 25 at 12:15 pm
купить диплом в тобольске [url=http://www.rudik-diplom11.ru]http://www.rudik-diplom11.ru[/url] .
Diplomi_jyMi
28 Oct 25 at 12:15 pm
ダッチワイフbut I am glad of it,indeed,
ラブドール
28 Oct 25 at 12:15 pm
кракен 2025
кракен vk3
Henryamerb
28 Oct 25 at 12:16 pm
гидроизоляция цена кг [url=https://www.gidroizolyaciya-cena-7.ru]https://www.gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_jcSi
28 Oct 25 at 12:16 pm
купить диплом техникума нижний новгород [url=https://www.frei-diplom8.ru]купить диплом техникума нижний новгород[/url] .
Diplomi_zmsr
28 Oct 25 at 12:19 pm
купить диплом в энгельсе [url=www.rudik-diplom10.ru/]www.rudik-diplom10.ru/[/url] .
Diplomi_qjSa
28 Oct 25 at 12:20 pm
kraken darknet market
kraken marketplace
Henryamerb
28 Oct 25 at 12:20 pm
what layshall we give this young man? ?“Thou knowest best,ラブドール 激安?was the sepulchral reply,
ラブドール
28 Oct 25 at 12:21 pm
клиники наркологические москва [url=http://narkologicheskaya-klinika-27.ru]http://narkologicheskaya-klinika-27.ru[/url] .
narkologicheskaya klinika_gwpl
28 Oct 25 at 12:22 pm
гидроизоляция подвала снаружи цены [url=gidroizolyaciya-cena-7.ru]gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_bkSi
28 Oct 25 at 12:22 pm
купить диплом [url=www.rudik-diplom11.ru/]купить диплом[/url] .
Diplomi_jkMi
28 Oct 25 at 12:23 pm
kraken market Kraken сайт – это таинственный портал в мир эксклюзивных возможностей, где анонимность и безопасность возведены в абсолют. Здесь границы стираются, а мечты обретают реальность, но помните: с большой свободой приходит и большая ответственность. Каждый шаг требует осознанности и понимания, ведь за порогом скрываются не только сокровища, но и подводные камни. Подходите к изучению платформы с мудростью и осторожностью, взвешивая каждый свой выбор.
Kevinmoots
28 Oct 25 at 12:24 pm
sebilgisayar.com – Bookmarked this immediately, planning to revisit for updates and inspiration.
Celinda Burkly
28 Oct 25 at 12:25 pm
(c) Road-beds and Open Track(1) On a curve,ラブドールtake the bolts out of the tie-plates connecting tosections of the outside rail,
ラブドール
28 Oct 25 at 12:26 pm
игровые автоматы 7к 7к казино официальный сайт гарантирует безопасность личных данных и честность азартных игр.
BrianFaw
28 Oct 25 at 12:27 pm
купить диплом в прокопьевске [url=https://rudik-diplom13.ru]купить диплом в прокопьевске[/url] .
Diplomi_cpon
28 Oct 25 at 12:27 pm
гидроизоляция подвала москва [url=www.gidroizolyaciya-cena-7.ru/]гидроизоляция подвала москва[/url] .
gidroizolyaciya cena_bvSi
28 Oct 25 at 12:28 pm
купить диплом математика [url=www.rudik-diplom11.ru/]купить диплом математика[/url] .
Diplomi_iiMi
28 Oct 25 at 12:28 pm
кракен тор
кракен vk3
Henryamerb
28 Oct 25 at 12:28 pm
alarm cd player [url=www.alarm-radio-clocks.com]www.alarm-radio-clocks.com[/url] .
Cd Player Radio Alarm Clocks_xnOa
28 Oct 25 at 12:28 pm
lived out thewinter sucking his own paw so,in his inclement,オナホ フィギュア
ラブドール
28 Oct 25 at 12:33 pm
OMT’s emphasis on fundamental abilities constructs unshakeable ѕelf-confidence, permitting Singapore trainees tο falⅼ fоr mathematics’s beauty аnd гeally feel inspired for tests.
Discover tһe benefit of 24/7 online math tuition at
OMT, ᴡһere appealing resources mɑke learning enjoyable and efficient f᧐r all levels.
As math forms tһе bedrock of abstract tһought and vital analytical іn Singapore’s education ѕystem, professional math tuition օffers thе customized assistance neеded to turn challenges into accomplishments.
primary tuition іs neceѕsary for developing durability аgainst PSLE’s tricky concerns,
sսch ɑѕ thoѕe οn possibility and simple data.
Іn Singapore’s competitive education ɑnd learning landscape, secondary math tuition ցives tһe extra edge
required t᧐ attract attention іn O Level rankings.
Ᏼy supplying considerable exercise ᴡith
past A Level examination papers, math tuition familiarizes students ѡith inquiry formats and noting schemes fоr optimum efficiency.
OMT’ѕ distinct method features a curriculum thаt complements the MOE framework ᴡith joint
aspects, encouraging peer conversations ߋn math
principles.
Flexible quizzes ɡet սsed tо уour level lah, challenging you ϳust right to
gradually elevate уouг test scores.
Wіth evolving MOE guidelines, math tuition maintains Singapore trainees upgraded օn curriculum changes for exam readiness.
Нere іs my website – math tuition primary
math tuition primary
28 Oct 25 at 12:33 pm
Hypotenuse AI — мощный инструмент для генерации текста, оптимизирован для создания
контента в областях электронной коммерции и социальных медиа.
https://www.matreshka.borda.ru/
28 Oct 25 at 12:35 pm
Hello to every body, it’s my first pay a quick visit of this web site; this web site contains amazing and truly excellent
information in favor of visitors.
cat exercise wheel review
28 Oct 25 at 12:35 pm
ремонт в подвале [url=https://gidroizolyaciya-cena-7.ru/]https://gidroizolyaciya-cena-7.ru/[/url] .
gidroizolyaciya cena_aySi
28 Oct 25 at 12:35 pm
Интернет вещей делает дома умными kraken онион тор kraken onion kraken onion ссылка kraken onion зеркала
RichardPep
28 Oct 25 at 12:36 pm