将服务层添加到Laravel应用程序的效果

时间:2017-01-29 22:35:52

标签: laravel service-layer

我使用Zend Framework开发了很多年,现在正在学习Laravel。

在我之前的应用程序中,我通常有一个由控制器调用的服务层。服务层位于Mapper和域模型的顶部,负责一些应用程序逻辑,引发事件,一些输入过滤等。

为什么我不应该在Laravel中实现服务层?在我到目前为止看到的示例中,控制器直接使用域对象(或更准确地说,活动记录)。

如果我的Laravel控制器调用了我的服务层,我是否会失去Laravel的任何优势? (据我所知,我仍然可以使用Route / Model绑定)。

作为第二个问题 - 实施我的服务层的最佳方式是什么?作为服务提供商的集合,也许?

1 个答案:

答案 0 :(得分:4)

我也从Zend转到Laravel并错过了我的服务。为了安抚自己,我已经在名称空间App \ Services中实现了一个Service命名空间。在那里,我做我所有的模型/验证handeling。我没有经历任何功能或任何损失。

我的控制器布局示例:

<?php
namespace App\Http\Controllers;

use App\Services\Contact as ContactService;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Lang;

class IndexController extends Controller
{

    /**
     * Create a new controller instance.
     *
     * @param Request $request
     * @return void
     */
    public function __construct(Request $request)
    {
        $this->_request  = $request;
    }

    /**
     * Standard contact page
     * 
     * @return contact page
     */
    public function contact(ContactService $contactService)
    {
        $errors  = null;
        $success = false;
        if ($this->_request->isMethod('post')) {
            $validator            = $contactService->validator($this->_request->all());
            if ($validator->fails()) {
                $errors = $validator->errors();
            } else {
                $contactService->create($validator->getData());
                $success = true;
            }
        }
        return view('pages/contact', ['errors' => $errors, 'success' => $success]);
    }
}

服务返回验证器,处理cruds,基本上做了我不想在控制器中看到的所有东西,就像我在Zend项目中那样。

服务示例:

<?php
namespace App\Services;

use Validator;
use Mail;
use App\Models\Contact as ContactModel;

class Contact
{

    /**
     * Get a validator for a contact.
     *
     * @param  array $data
     * @return \Illuminate\Contracts\Validation\Validator
     */
    public function validator(array $data)
    {
        return Validator::make($data, [
                'email'     => 'required|email|max:255',
                'phone'     => 'max:255',
                'firstName' => 'required|max:255',
                'lastName'  => 'required|max:255',
                'message'   => 'required'
        ]);
    }

    /**
     * Create a new contact instance after a valid form.
     *
     * @param  array $data
     * @return ContactModel
     */
    public function create(array $data)
    {
        //Handle or map any data differently if needed, just for illustration
        $data = [
            'email'     => $data['email'],
            'firstName' => $data['firstName'],
            'lastName'  => $data['lastName'],
            'language'  => $data['language'],
            'phone'     => $data['phone'],
            'message'   => $data['message']
        ];

        // Send an email
        Mail::send('emails.contact', ['data' => $data], function ($m) use ($data) {
            $m->from(config('mail.from.address'), config('mail.from.name'));
            $m->to(env('MAIL_TO', 'hello@world.com'), env('MAIL_TO'))->subject('Contact form entry from: ' . $data['firstName']);
        });

        return ContactModel::create($data);
    }
}