ฉันกำลังทำโปรเจ็กต์ขนาดใหญ่โดยใช้ Laravel 4 และต้องตอบคำถามทั้งหมดที่คุณถามตอนนี้ หลังจากอ่านหนังสือ Laravel ที่มีอยู่ทั้งหมดที่ Leanpub และ Googling มากมายฉันก็คิดโครงสร้างต่อไปนี้
- หนึ่งคลาส Eloquent Model ต่อตารางที่สามารถระบุได้
- คลาส Repository หนึ่งคลาสต่อ Eloquent Model
- คลาสเซอร์วิสที่อาจสื่อสารระหว่างคลาส Repository หลายคลาส
สมมติว่าฉันกำลังสร้างฐานข้อมูลภาพยนตร์ ฉันจะมีคลาส Eloquent Model ต่อไปนี้เป็นอย่างน้อย:
- หนัง
- สตูดิโอ
- ผู้อำนวยการ
- นักแสดงชาย
- ทบทวน
คลาสที่เก็บจะห่อหุ้มคลาส Eloquent Model แต่ละคลาสและรับผิดชอบการดำเนินการ CRUD บนฐานข้อมูล คลาสที่เก็บอาจมีลักษณะดังนี้:
- MovieRepository
- StudioRepository
- DirectorRepository
- ActorRepository
- ReviewRepository
แต่ละคลาสที่เก็บจะขยายคลาส BaseRepository ซึ่งใช้อินเตอร์เฟสต่อไปนี้:
interface BaseRepositoryInterface
{
public function errors();
public function all(array $related = null);
public function get($id, array $related = null);
public function getWhere($column, $value, array $related = null);
public function getRecent($limit, array $related = null);
public function create(array $data);
public function update(array $data);
public function delete($id);
public function deleteWhere($column, $value);
}
คลาสบริการใช้เพื่อรวมที่เก็บข้อมูลหลายแห่งเข้าด้วยกันและมี "ตรรกะทางธุรกิจ" ที่แท้จริงของแอปพลิเคชัน ตัวควบคุมจะสื่อสารกับคลาสบริการสำหรับการสร้างอัปเดตและลบเท่านั้น
ดังนั้นเมื่อฉันต้องการสร้าง Movie record ใหม่ในฐานข้อมูลคลาส MovieController ของฉันอาจมีวิธีการดังต่อไปนี้:
public function __construct(MovieRepositoryInterface $movieRepository, MovieServiceInterface $movieService)
{
$this->movieRepository = $movieRepository;
$this->movieService = $movieService;
}
public function postCreate()
{
if( ! $this->movieService->create(Input::all()))
{
return Redirect::back()->withErrors($this->movieService->errors())->withInput();
}
// New movie was saved successfully. Do whatever you need to do here.
}
ขึ้นอยู่กับคุณที่จะกำหนดวิธีการโพสต์ข้อมูลไปยังคอนโทรลเลอร์ของคุณ แต่สมมติว่าข้อมูลที่ส่งคืนโดย Input :: all () ในเมธอด postCreate () จะมีลักษณะดังนี้:
$data = array(
'movie' => array(
'title' => 'Iron Eagle',
'year' => '1986',
'synopsis' => 'When Doug\'s father, an Air Force Pilot, is shot down by MiGs belonging to a radical Middle Eastern state, no one seems able to get him out. Doug finds Chappy, an Air Force Colonel who is intrigued by the idea of sending in two fighters piloted by himself and Doug to rescue Doug\'s father after bombing the MiG base.'
),
'actors' => array(
0 => 'Louis Gossett Jr.',
1 => 'Jason Gedrick',
2 => 'Larry B. Scott'
),
'director' => 'Sidney J. Furie',
'studio' => 'TriStar Pictures'
)
เนื่องจาก MovieRepository ไม่ควรรู้วิธีสร้างเร็กคอร์ด Actor, Director หรือ Studio ในฐานข้อมูลเราจึงใช้คลาส MovieService ของเราซึ่งอาจมีลักษณะดังนี้:
public function __construct(MovieRepositoryInterface $movieRepository, ActorRepositoryInterface $actorRepository, DirectorRepositoryInterface $directorRepository, StudioRepositoryInterface $studioRepository)
{
$this->movieRepository = $movieRepository;
$this->actorRepository = $actorRepository;
$this->directorRepository = $directorRepository;
$this->studioRepository = $studioRepository;
}
public function create(array $input)
{
$movieData = $input['movie'];
$actorsData = $input['actors'];
$directorData = $input['director'];
$studioData = $input['studio'];
// In a more complete example you would probably want to implement database transactions and perform input validation using the Laravel Validator class here.
// Create the new movie record
$movie = $this->movieRepository->create($movieData);
// Create the new actor records and associate them with the movie record
foreach($actors as $actor)
{
$actorModel = $this->actorRepository->create($actor);
$movie->actors()->save($actorModel);
}
// Create the director record and associate it with the movie record
$director = $this->directorRepository->create($directorData);
$director->movies()->associate($movie);
// Create the studio record and associate it with the movie record
$studio = $this->studioRepository->create($studioData);
$studio->movies()->associate($movie);
// Assume everything worked. In the real world you'll need to implement checks.
return true;
}
ดังนั้นสิ่งที่เราเหลืออยู่คือการแยกความกังวลที่ดีและสมเหตุสมผล ที่เก็บจะทราบเฉพาะโมเดล Eloquent ที่แทรกและดึงข้อมูลจากฐานข้อมูล ตัวควบคุมไม่สนใจที่เก็บข้อมูลพวกเขาเพียงแค่ส่งต่อข้อมูลที่รวบรวมจากผู้ใช้และส่งต่อไปยังบริการที่เหมาะสม บริการไม่สนใจว่าข้อมูลที่ได้รับจะบันทึกลงในฐานข้อมูลอย่างไรเพียงส่งต่อข้อมูลที่เกี่ยวข้องที่ผู้ควบคุมมอบให้ไปยังที่เก็บที่เหมาะสม