Как определить отношения и получить все записи с отношениями в Laravel?

Я хочу определить отношения между таблицами, и я не знаю, где я ошибаюсь.

Это мои таблицы:

пользователи

    -id
    -email
    -level

рестораны

    -id
    -name
    -user_id

меню

    -id
    -name

restaurant_menu

    -restaurant_id
    -menu_id
    -price

В таблице users поле level я задам двумя словами: [user] или [restaurant].

Модель пользователя

public function restaurant(){
    return $this->hasOne(Restaurant::class);
}

Модель ресторана

public function menus(){
    return $this->hasMany(Menu::class);
}

Модель меню

public function restaurants(){
    return $this->belongsTo(Restaurant::class);
}

Я ожидаю вывод этого кода:

$result = $restaurant->where('id',2)->menus()->get(); 

Чтобы были записи и отношения, но я получаю следующую ошибку.

BadMethodCallException Call to undefined method Illuminate\Database\Eloquent\Builder::menus()

Что я должен делать?

в сокращении вы можете использовать $restaurant = Restaurants::find(2)->menus;

Yasii 28.01.2019 01:21

все еще такой же..

Mahdi Alikhani 28.01.2019 14:49

@MahdiAlikhani, это потому, что ваши отношения настроены неправильно. Вы должны использовать отношения belongsToMany для отношения «Ресторан-меню», как я указал в своем ответе. Если вам сложно создать правильную структуру таблицы для решения вашей проблемы, создайте отдельный вопрос с тегом database-design.

d3jn 28.01.2019 15:29
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Symfony Station Communiqué - 7 июля 2023 г
Symfony Station Communiqué - 7 июля 2023 г
Это коммюнике первоначально появилось на Symfony Station .
Оживление вашего приложения Laravel: Понимание режима обслуживания
Оживление вашего приложения Laravel: Понимание режима обслуживания
Здравствуйте, разработчики! В сегодняшней статье мы рассмотрим важный аспект управления приложениями, который часто упускается из виду в суете...
Установка и настройка Nginx и PHP на Ubuntu-сервере
Установка и настройка Nginx и PHP на Ubuntu-сервере
В этот раз я сделаю руководство по установке и настройке nginx и php на Ubuntu OS.
Коллекции в Laravel более простым способом
Коллекции в Laravel более простым способом
Привет, читатели, сегодня мы узнаем о коллекциях. В Laravel коллекции - это способ манипулировать массивами и играть с массивами данных. Благодаря...
Как установить PHP на Mac
Как установить PHP на Mac
PHP - это популярный язык программирования, который используется для разработки веб-приложений. Если вы используете Mac и хотите разрабатывать...
1
3
10 467
4
Перейти к ответу Данный вопрос помечен как решенный

Ответы 4

Фактически,

вы правильно определили отношения. Я думаю, что у вас есть проблема с получением этих отношений по красноречивым запросам. Если вы хотите получить отношение к меню конкретного ресторана; вы должны реализовать что-то подобное.

$restaurant = Restaurants::find(2);
$restaurant->menus(); // this will return an array of related menus.

Вы должны прочитать Официальная документация Laravel. Для каждого определения отношения есть заголовки Восстановление отношений.

Я написал этот код с эхом и возвратом и получил эту ошибку: Object of class Illuminate\Database\Eloquent\Builder could not be converted to string, но без возврата и эха я не получил ошибку, но он мне ничего не показывает.

Mahdi Alikhani 28.01.2019 00:35

@MahdiAlikhani, это просто очень простая общая ошибка PHP, которая не имеет ничего общего с Laravel. Поскольку ошибка говорит, что вы не можете преобразовать объект в строку, если вы просто хотите увидеть содержимое объекта, тогда var_dump() — это то, что вам нужно, или даже лучше функция laravels dd(), поэтому у вас также есть форматирование

twigg 28.01.2019 00:51

Как упоминал @twigg, это общая ошибка PHP. Вы должны запустить цикл for на нем. Или вы можете получить дамп через функцию dd Laravel. dd($restaurant->menus());

bl4cksta 28.01.2019 11:51

Отношения, которые вы определяете, недоступны для вас в области запрос вашего экземпляра построителя так, как вы пытались их вызвать. Они доступны в контексте вашего модели. Включение ваших отношений в запросы осуществляется по-другому - вы должны проверить официальная документация по этому вопросу в первую очередь.

В вашем случае, если вы хотите выбрать все меню, принадлежащие определенному ресторану, вам нужно:

  1. Вы можете сначала получить конкретный ресторан, а затем получить его меню через отношения:

    $restaurant = Restaurant::find(2);
    $menus = $restaurant->menus;
    
  2. Вы можете запросить меню по модели Menu:

    $menus = Menu::whereHas('restaurants', function ($query) {
        $query->where('id', 2);
    })-get();
    

Также ваши отношения настроены неправильно. Основываясь на структуре таблицы, которую вы предоставили, ваши меню и рестораны находятся в отношениях многие ко многим. Поэтому метод отношения restaurants() в классе Menu должен возвращать экземпляр BelongsToMany. Поэтому, пока вы этим занимаетесь, я настоятельно рекомендую вам просмотреть документацию отношения и посмотреть примеры, пока вы не поймете, как работают различные отношения в Laravel.

вы знаете, в этом сценарии у меня есть таблица меню, в которой я храню меню, и restaurant_menu для сводной таблицы, чтобы хранить в ней меню и цены каждого ресторана, я запутался и не знаю, что делать!

Mahdi Alikhani 28.01.2019 14:44

В ваших отношениях есть проблема. У вас есть отношение один ко многим между моделью ресторана и моделью меню. Однако структура вашей таблицы предполагает, что у них есть модель «многие ко многим».

Если вы считаете, что ваша модель правильная, измените таблицу меню следующим образом:

меню

-id
-name
-restaurant_id //new field

Также удалите таблицу "restaurant_menu".

Если вы считаете, что ваша база данных верна, попробуйте соответствующим образом изменить модель.

вы знаете, в этом сценарии у меня есть таблица меню, в которой я держу меню, и restaurant_menu для сводной таблицы, чтобы хранить в ней меню и цены каждого ресторана, я запутался и не знаю, что делать!

Mahdi Alikhani 28.01.2019 14:46

Ресторан может иметь более одного меню. Но верно ли обратное? Может ли меню быть применимо более чем к одному ресторану? Если да, то вам нужна сводная таблица. В противном случае вы должны использовать только 'restaurant_id' в таблице меню.

Nabil Farhan 29.01.2019 04:35
Ответ принят как подходящий

В пользовательской модели

public function restaurant(){
   return $this->hasMany(Restaurant::class,'user_id','id');
}

Модель ресторана

public function menus(){
   return $this->hasMany(Menu::class,'id','menu_id');
}

Модель меню

public function restaurants(){
  return $this->belongsTo(Restaurant::class);
}

Когда вы аспектируете свой вывод, вам нужно написать

$result = Restaurant::with('menus')->where('id',2)->get(); 

вы получите реляционные данные. Например, в каких ресторанах какое меню.

Вы также можете использовать это для своего сценария. Используйте это в своей модели ресторана

public function menues(){
  return $this>hasManyThrough(Menu::class,restaurant_menu::class,'menu_id','id','id','restaurant_id');
} 

я сделал все это, но я должен создать поле идентификатор_ресторана в таблице меню, я думаю, что это мне не поможет, вы знаете, в этом сценарии у меня есть таблица меню, в которой я держу меню, и restaurant_menu для поворота стол, чтобы хранить в нем меню и цены каждого ресторана, я запутался и не знаю, что делать!

Mahdi Alikhani 28.01.2019 14:46

если вы хотите делать то, что говорите, вам нужно использовать отношение «hasManyThrough» в вашей модели. Я редактирую свой ответ. Вы найдете новое предложение в последней функции

sabbir chowdury 28.01.2019 17:42

спасибо за ваш ответ, я использовал ваш код и изменил некоторые из них и, наконец, получил результат! у меня есть еще один вопрос, как я могу хранить данные $request в таблице Ресторан_Меню? у меня есть панель для владельцев ресторанов, в которой они будут создавать свои собственные меню из списка и устанавливать цены, вся эта информация должна храниться в таблице *Меню_Ресторана.

Mahdi Alikhani 28.01.2019 20:59

Это полностью зависит от вас. Как вы поддерживаете свой дизайн конца шрифта. Вы можете легко использовать раскрывающиеся селекторы, которые позволяют одному пользователю выбрать restaurant_id и menu_id в текстовом поле для установки цен. В другом случае вы можете выбрать свой ресторан, а затем просто указать идентификатор меню и установить цену. Но помните одну вещь, если вы используете вторую идею, вам нужно передать идентификатор ресторана в скрытом поле. Надеюсь ты понимаешь

sabbir chowdury 29.01.2019 06:15

О, ты прав. я думаю, что первый лучше, чем второй.

Mahdi Alikhani 31.01.2019 18:19

Еще раз спасибо за помощь.

Mahdi Alikhani 31.01.2019 18:19

Другие вопросы по теме