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!
Поэтому лучше всего в таком случае временно сдать их на тёплый склад.
уборка квартиры сегодня
25 Oct 25 at 1:46 am
1x bet giri? [url=http://1xbet-9.com/]http://1xbet-9.com/[/url] .
1xbet_qwSn
25 Oct 25 at 1:47 am
купить диплом преподавателя [url=https://rudik-diplom2.ru/]купить диплом преподавателя[/url] .
Diplomi_pnpi
25 Oct 25 at 1:47 am
Мы одинаково внимательно относимся и к амбулаторным, и к стационарным маршрутам. Для каждого формируется индивидуальная цель (устранение абстиненции, нормализация сна, восстановление внимания и мотивации) и назначается «точка переоценки» — момент, когда команда сверяет динамику и, если нужно, меняет один параметр плана (дозу, время внесения процедур, приоритет модуля). Так достигается предсказуемость и избегается избыточная фармаконагрузка.
Подробнее тут – [url=https://narcologicheskaya-klinika-stavropol0.ru/]narcologicheskaya-klinika-stavropol0.ru/[/url]
RonaldEralf
25 Oct 25 at 1:48 am
купить диплом в костроме [url=https://www.rudik-diplom6.ru]https://www.rudik-diplom6.ru[/url] .
Diplomi_trKr
25 Oct 25 at 1:48 am
concernedaboutpollution.com – Feels trustworthy and credible—good for raising awareness about pollution.
Garry Trimbach
25 Oct 25 at 1:48 am
1xbet resmi giri? [url=http://www.1xbet-4.com]http://www.1xbet-4.com[/url] .
1xbet_blol
25 Oct 25 at 1:50 am
диплом техникума ссср купить [url=www.frei-diplom12.ru/]диплом техникума ссср купить[/url] .
Diplomi_pbPt
25 Oct 25 at 1:51 am
купить диплом электрика [url=http://www.rudik-diplom14.ru]купить диплом электрика[/url] .
Diplomi_wtea
25 Oct 25 at 1:52 am
https://t.me/bs_1xbet/44
https://t.me/s/bs_1xbet/33
25 Oct 25 at 1:52 am
где купить диплом техникума собою [url=www.frei-diplom11.ru]где купить диплом техникума собою[/url] .
Diplomi_oksa
25 Oct 25 at 1:53 am
https://t.me/bs_1xbet/16
https://t.me/bs_1xbet/29
25 Oct 25 at 1:54 am
I have fun with, result in I found just what I was taking a look for.
You have ended my 4 day lengthy hunt! God Bless you man.
Have a nice day. Bye
Trade Vector AI
25 Oct 25 at 1:54 am
I enjoy what you guys are up too. This type of clever work and coverage!
Keep up the wonderful works guys I’ve incorporated you guys to my own blogroll.
Fırat Engin bahis siteleri
25 Oct 25 at 1:55 am
신용카드현금화 – 급전이 필요할 때, 신용카드 한도를 안전하고 간편하게 현금으로 바꿔드립니다.
낮은 수수료, 신용등급 걱정 없이 즉시 입금,
모든 카드사 이용 가능
신용카드현금화
25 Oct 25 at 1:55 am
купить диплом математика [url=http://rudik-diplom6.ru/]купить диплом математика[/url] .
Diplomi_tvKr
25 Oct 25 at 1:56 am
Если к «окну оценки» улучшение ниже ожидаемого, меняем один параметр — скорость/состав инфузии, время приёма препаратов или порядок модулей — и назначаем новое окно. Так видно реальный вклад каждого шага и не возникает «снежный ком» вмешательств.
Узнать больше – [url=https://vyvod-iz-zapoya-petrozavodsk0.ru/]вывод из запоя в петрозаводске[/url]
PatrickSlode
25 Oct 25 at 1:57 am
Linking modules in OMT’s curriculum convenience ⅽhanges Ƅetween levels, supporting continuous
love fօr mathematics and exam confidence.
Ԍet ready fоr success іn upcoming exams ԝith OMT
Math Tuition’ѕ exclusive curriculum, designed tⲟ
foster іmportant thinking ɑnd sеⅼf-confidence in eѵery student.
In Singapore’s strenuous education system,
wheгe mathematics іs required and takes іn aгound 1600 hours of curriculum time in primary
аnd secondary schools, math tuition еnds uρ bеing vital to
assist students construct ɑ strong foundation for
ⅼong-lasting success.
Ultimately, primary school math tuition іs crucial fоr PSLE
quality, ɑs it gears up trainees ѡith the tools to accomplish top bands and secure favored secondary school placements.
Tuition aids secondary pupils сreate test strategies, ѕuch as
timе appropriation for tһe two O Level mathematics papers, Ƅring aЬout much better overall efficiency.
Ultimately, junior college math tuition іѕ crucial to protecting
tⲟp A Level results, opеning doors to prominent scholarships ɑnd college opportunities.
Distinctly tailored tο match the MOE curriculum, OMT’ѕ custom-made
mathematics program incorporates technology-driven devices fоr interactive understanding experiences.
Comprehensive protection ⲟf subjects ѕia, leaving no spaces іn understanding for top mathematics accomplishments.
Math tuition develops а solid portfolio of abilities, improving Singapore trainees’
resumes fоr scholarships based սpon examination outcomes.
Αlso visit mу webpage … maths tuition near me class 8
maths tuition near me class 8
25 Oct 25 at 1:57 am