در آموزش ساخت کلاس App به نوشتن بخش ابتدایی این کلاس پرداختیم به طوری که تا این مرحله به صورت زیر است:
<?php
namespace Core;
class App
{
private $controller;
private $action;
private $params = [];
private function parseUrl()
{
$request = trim($_SERVER['REQUEST_URI'], '/');
$request = filter_var($request, FILTER_SANITIZE_URL);
$request = explode('/', $request);
return $request;
}
}در پروسهٔ تکمیل این کلاس، نیاز داریم تا یک کانستراکتور تعریف کنیم که به محض ساخت آبجکتی از روی این کلاس فراخوانی شده و دستورات داخل آن اجرا گردند. به همین منظور، ابتدا کلاس فوق را به صورت زیر تکمیل نموده سپس به تفسیر کدها خواهیم پرداخت:
<?php
namespace Core;
use Base\Controllers\DefaultController;
class App
{
private $controller;
private $action;
private $params = [];
private function parseUrl()
{
$request = trim($_SERVER['REQUEST_URI'], '/');
$request = filter_var($request, FILTER_SANITIZE_URL);
$request = explode('/', $request);
return $request;
}
public function __construct()
{
$url = $this->parseUrl();
$routing = new Routing;
if ($routing->routes) {
$this->controller = new DefaultController;
$this->action = 'homepage';
$controllerPlusAction = (isset($url[0]) ? $url[0] : '') . (isset($url[1]) ? '/' . $url[1] : '');
foreach ($routing->routes as $route) {
if ($route['route'] == $controllerPlusAction) {
if (file_exists('../app/' . $route['module'] . '/Controllers/' . $route['controller'] . '.php')) {
$dynamicControllerName = "\\" . $route['module'] . "\Controllers\\" . $route['controller'];
$this->controller = new $dynamicControllerName;
$this->action = $route['action'];
unset($url[0], $url[1]);
break;
} else {
$this->action = 'homepage';
}
} else {
$this->action = 'notfound';
}
}
}
$this->params = $url ? array_values($url) : [];
call_user_func_array([$this->controller, $this->action], $this->params);
}
}نیاز به توضیح نیست که به منظور ساخت کانستراکتور در زبان پیاچپی میباید از ساختار ()construct__ استفاده نماییم. پیش از هر چیز، در بدنهٔ این کانستراکتور متغیری تحت عنوان url$ ساخته و تابع ()parseUrl که پیش این نوشته بودیم را به آن منتسب میکنیم و در ادامه متغیر دیگری به نام routing$ ساخته و مقدار آن را برابر با نمونهای از روی کلاس Routing قرار میدهیم سپس با استفاده از یک دستور شرطی چک کردهایم ببینیم که آیا آرایهٔ routes$ که داخل کلاس Routing قرار داد حاوی مقداری است یا خیر که اگر اینچنین بود بلافاصله پراپرتیهای controller$ و action$ را به ترتیب با ساخت یک آبجکت از روی کلاس DefaultController و همچنین استرینگ homepage مقداردهی میکنیم.
در خط بیستوهشتم متغیری به نام controllerPlusAction$ میسازیم که قرار است تا دربرگیرندهٔ نام کنترلر + یک اَکشن باشد. برای همین منظور، با استفاده از تابع ()isset چک کردهایم ببینیم که آیا خانهٔ صِفرم آرایهٔ url$ حاوی مقداری است یا خیر به طوری که اگر پُر بود آن را چاپ میکنیم و در غیر این صورت مقدار خالی '' را برایش در نظر میگیریم (لازم به یادآوری است که خانهٔ صفر متعلق به کنترلر و خانه یک متعلق به اَکشن است.) سپس یک علامت . قرار داده و همین روش را برای خانهٔ یکم این آرایه چک میکنیم به طوری که اگر پُر بود ابتدا یک علامت / چاپ کرده سپس خانهٔ یکم آرایهٔ url$ را چاپ میکنیم و در غیر این صورت مقدار خالی '' را در نظر میگیریم.
سپس آرایهٔ routes$ را با استفاده یک حلقه به اِلِمانهای تشکیلدهندهاش تفکیک کرده و داخل بدنهٔ foreach با استفاده از یک دستور شرطی چک میکنیم ببینیم که آیا در کلید route در اِلِمانهای آرایهٔ routes$ مقداری برابر با مقدار در نظر گرفته شده برای controllerPlusAction$ وجود دارد یا خیر که اگر چنین مقداری وجود نداشت وارد دستور else مرتبط با این دستور شرطی شده و مقدار استرینگ notfound را برای پراپرتی action$ در نظر میگیریم که مرتبط با ویوی notfound.php به منظور نمایش صفحهٔ ۴۰۴ خواهد بود اما اگر چنین مقداری وجود داشت، بلافاصله با استفاده از یک دستور شرطی دیگر و همچنین تابع ()file_exsits چک میکنیم ببینیم که آیا در ماژول مربوطه کنترلی همنام با کنترلری که در کلید controller متغیر route$ درج شده وجود دارد یا خیر که اگر نتیجه false بود وارد دستور else شده و استرینگ homepage به پراپرتی action$ اختصاص خواهد یافت و اگر نتیجه true بود، متغیری به صورت زیر میسازیم:
$dynamicControllerName = "\\" . $route['module'] . "\Controllers\\" . $route['controller'];پیش از توضیح پیرامون مقدار در نظر گرفته شده برای این متغیر، فرض کنیم که در لینک http://mvc.local/default/users قرار داریم. مقدار اختصاصیافته به متغیر فوق برای چنین یوآرالی برابر با استرینگ Base\Controllers\DefaultController\ خواهد بود. به نظر میرسد حال که میدانیم خروجی به چه شکل باید باشد، میتوانیم به شکل بهتری دست به تفسیر مقدار اختصاصیافته برای متغیر dynamicControllerName$ بزنیم.
در واقع، در ابتدای رشته دو علامت \\ قرار دادهایم که علامت \ اول به منظور اصطلاحاً Escape کردن علامت \ دوم به کار رفته است و برای همین منظور از دو علامت به صورت "\\" استفاده میکنیم به طوری که علامت \ اول منجر به بیتأثیر شدن علامت \ گردیده و صرفاً آن را چاپ میکند. سپس مقدار در نظر گرفته شده برای کلید module در متغیر route$ را هدف قرار گرفتهایم و در ادامه مجدد یک . قرار داده و استرینگ \\controllers\ را نوشتهایم که مجدداً علامت \ اول به منظور اِسکِیپ کردن علامت \ دوم به کار رفته است و در نهایت یک . قرار داده و مقدار در نظر گرفته شده برای کلید controller در آرایهٔ route$ را درج کردهایم.
در این مرحله از کار، توانستهایم به صورت دینامیک (پویا) اقدام به یافتن نِیماِسپیسی که کنترلر در آن قرار دارد نماییم و در ادامه میتوانیم به سادگی با استفاده از کیورد new نمونهای از روی این کنترلر ساخته و آن را به پراپرتی controller$ اختصاص داده سپس مقدار موجود در کلید action در آرایهٔ route$ را نیز به پراپرتی action$ منتسب میکنیم.
همچنین میبینیم که در خط سیوپنجم با استفاده از متد ()unset اِلِمانهای صفرم و یکم آرایهٔ url$ را خالی کردهایم و این بدین دلیل است که چنانچه هر گونه پارامتری به عنوان اِلِمان سوم سِت شده بود، به سادگی در خط چهلوششم به پراپرتی params$ اختصاص یابد.
آنچه در اینجا بسیار حائز اهمیت میباشد، استفادهٔ صحیح از کیورد break است. در واقع، کاری که این دستور داخل حلقهها انجا میدهد آن است که چنانچه شرطی خاص برآورده شود، میتوان در انتهای شرط مذکور دستور break را نوشته تا دیگر پروسهٔ چرخیدن داخل آرایهٔ ورودی برای حلقهٔ foreach خاتمه یابد و در ارتباط با اسکریپت فوق، این دقیقاً همان چیزی است که به آن نیاز داریم؛ به عبارتی، نیاز داریم تا به محض آنکه کنترلری در یوآرالی که توسط کاربر وارد شده با یکی از مقادیر موجود در کلید controller در آرایهٔ routes$ دقیقاً یکسان بود، این حلقه دیگر ادامه نیابد.
در ادامهٔ تفسیر این کلاس، نوبت به تفسیر دو خط آخر این کانستراکتور میرسد که عبارتند از:
$this->params = $url ? array_values($url) : [];
call_user_func_array([$this->controller, $this->action], $this->params);در خط اول دستور دادهایم که اگر آرایهٔ url$ وجود داشت، مقادیر آن آرایه با استفاده از تابع ()array_values به پراپرتی params$ اختصاص یابد و در غیر این صورت، از طریق علائم [] آرایهای خالی به این پراپرتی اختصاص پیدا کند و در خط دوم از هم از تابعی به نام ()call_user_func_array استفاده کردهایم که درک سازوکار آن نیاز به کمی توضیح دارد که در ادامه به بررسی این مسئله خواهیم پرداخت.
آشنایی با کاربرد تابع ()call_user_func_array
فانکشن یا بهتر بگوییم کانستراکت ()call_user_func_array در زبان پیاچپی دو پارامتر ورودی گرفته سپس بر آن اساس یک خروجی در معرض دیدمان ریترن میکند که برای درک بهتر این موضوع، مثال فرضی زیر را مد نظر قرار میدهیم:
<?php
class Behzad
{
public function code($arg1, $arg2)
{
echo "Behzad codes in $arg1 and $arg2";
}
}کلاسی نوشتهایم به نام Behzad که داخلش یک فانکشن تحت عنوان ()code داریم که دو پارامتر ورودی میگیرد؛ سپس داخل این فانکشن استرینگی را به همراه مقادیر منتسب به پارامترهای ورودی چاپ کردهایم. حال در ادامه میبینیم که چگونه از ()call_user_func_array در ارتباط با این کلاس میتوانیم استفاده کنیم:
<?php
class Behzad
{
public function code($arg1, $arg2)
{
echo "Behzad codes in $arg1 and $arg2";
}
}
$behzadObj = new Behzad;
call_user_func_array([$behzadObj, "code"], ["PHP", "JS"]);همانطور که ملاحظه میشود، از روی این کلاس آبجکتی تحت عنوان behzadObj$ ساختهایم سپس به عنوان پارامتر اول ()call_user_func_array آرایهای را در نظر گرفتهایم که اِلِمان اولش آبجکت ساختهشده از روی کلاس Behzad است و اِلِمان دوم نیز نام متدی میباشد تحت عنوان ()code که داخل این کلاس تعریف کردهایم. به عنوان پارامتر دوم ()call_user_func_array نیز آرایهای در نظر گرفتهایم که حاوی دو اِلِمان PHP و JS است.
اساساً کاری که ()call_user_func_array انجام میدهد آن است که در کلاس (آبجکت) مذکور به دنبال فانکشنی تحت عنوان ()code میگردد سپس مقادیر PHP و JS را به ترتیب به عنوان آرگومانهای ورودی فانکشن ()code در نظر میگیرد به طور که در خروجی خواهیم داشت:
Behzad codes in PHP and JSحال به نظر میرسد که با این توضیحات بتوانیم به تفسیر خطی از کانستراکتور کلاس App بپردازیم که در آن از ()call_user_func_array استفاده نمودهایم. در حقیقت، دو آرایه به عنوان آرگومان اول و دوم این فانکشن در نظر گرفتهایم به طوری که آرایهٔ اول حاوی کلیدهای this->controller$ و this->action$ است مضاف بر اینکه پراپرتی this->params$ را به عنوان پارامتر دوم در نظر گرفتهایم به طوری که در نهایت ()call_user_func_array در کلاس منتسب به پراپرتی this->controller$ به دنبال اَکشن منتسب به پراپرتی this->action$ گشته و مقدار متنسب به this->params$ را به آن متد اختصاص داده و آن را فراخوانی میکند.
در این آموزش علاوه بر تکمیل کانستراکتور کلاس App، با کاربرد دستور ()call_user_func_array در زبان پیاچپی آشنا شدیم به طوری که این امکان را در اختیار توسعهدهندگان این زبان برنامهنویسی میگذارد تا بتوانند اقدام به فراخوانی متدهای مختلف یک آبجکت کرده و به صورت کاملاً دینامیک مقادیر مد نظر خود را به آنها پاس دهند.
