Я хочу определить отношения между таблицами, и я не знаю, где я ошибаюсь.
Это мои таблицы:
пользователи
-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()
Что я должен делать?
все еще такой же..
@MahdiAlikhani, это потому, что ваши отношения настроены неправильно. Вы должны использовать отношения belongsToMany для отношения «Ресторан-меню», как я указал в своем ответе. Если вам сложно создать правильную структуру таблицы для решения вашей проблемы, создайте отдельный вопрос с тегом database-design.






Фактически,
вы правильно определили отношения. Я думаю, что у вас есть проблема с получением этих отношений по красноречивым запросам. Если вы хотите получить отношение к меню конкретного ресторана; вы должны реализовать что-то подобное.
$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, но без возврата и эха я не получил ошибку, но он мне ничего не показывает.
@MahdiAlikhani, это просто очень простая общая ошибка PHP, которая не имеет ничего общего с Laravel. Поскольку ошибка говорит, что вы не можете преобразовать объект в строку, если вы просто хотите увидеть содержимое объекта, тогда var_dump() — это то, что вам нужно, или даже лучше функция laravels dd(), поэтому у вас также есть форматирование
Как упоминал @twigg, это общая ошибка PHP. Вы должны запустить цикл for на нем. Или вы можете получить дамп через функцию dd Laravel. dd($restaurant->menus());
Отношения, которые вы определяете, недоступны для вас в области запрос вашего экземпляра построителя так, как вы пытались их вызвать. Они доступны в контексте вашего модели. Включение ваших отношений в запросы осуществляется по-другому - вы должны проверить официальная документация по этому вопросу в первую очередь.
В вашем случае, если вы хотите выбрать все меню, принадлежащие определенному ресторану, вам нужно:
Вы можете сначала получить конкретный ресторан, а затем получить его меню через отношения:
$restaurant = Restaurant::find(2);
$menus = $restaurant->menus;
Вы можете запросить меню по модели Menu:
$menus = Menu::whereHas('restaurants', function ($query) {
$query->where('id', 2);
})-get();
Также ваши отношения настроены неправильно. Основываясь на структуре таблицы, которую вы предоставили, ваши меню и рестораны находятся в отношениях многие ко многим. Поэтому метод отношения restaurants() в классе Menu должен возвращать экземпляр BelongsToMany. Поэтому, пока вы этим занимаетесь, я настоятельно рекомендую вам просмотреть документацию отношения и посмотреть примеры, пока вы не поймете, как работают различные отношения в Laravel.
вы знаете, в этом сценарии у меня есть таблица меню, в которой я храню меню, и restaurant_menu для сводной таблицы, чтобы хранить в ней меню и цены каждого ресторана, я запутался и не знаю, что делать!
В ваших отношениях есть проблема. У вас есть отношение один ко многим между моделью ресторана и моделью меню. Однако структура вашей таблицы предполагает, что у них есть модель «многие ко многим».
Если вы считаете, что ваша модель правильная, измените таблицу меню следующим образом:
меню
-id
-name
-restaurant_id //new field
Также удалите таблицу "restaurant_menu".
Если вы считаете, что ваша база данных верна, попробуйте соответствующим образом изменить модель.
вы знаете, в этом сценарии у меня есть таблица меню, в которой я держу меню, и restaurant_menu для сводной таблицы, чтобы хранить в ней меню и цены каждого ресторана, я запутался и не знаю, что делать!
Ресторан может иметь более одного меню. Но верно ли обратное? Может ли меню быть применимо более чем к одному ресторану? Если да, то вам нужна сводная таблица. В противном случае вы должны использовать только 'restaurant_id' в таблице меню.
В пользовательской модели
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 для поворота стол, чтобы хранить в нем меню и цены каждого ресторана, я запутался и не знаю, что делать!
если вы хотите делать то, что говорите, вам нужно использовать отношение «hasManyThrough» в вашей модели. Я редактирую свой ответ. Вы найдете новое предложение в последней функции
спасибо за ваш ответ, я использовал ваш код и изменил некоторые из них и, наконец, получил результат! у меня есть еще один вопрос, как я могу хранить данные $request в таблице Ресторан_Меню? у меня есть панель для владельцев ресторанов, в которой они будут создавать свои собственные меню из списка и устанавливать цены, вся эта информация должна храниться в таблице *Меню_Ресторана.
Это полностью зависит от вас. Как вы поддерживаете свой дизайн конца шрифта. Вы можете легко использовать раскрывающиеся селекторы, которые позволяют одному пользователю выбрать restaurant_id и menu_id в текстовом поле для установки цен. В другом случае вы можете выбрать свой ресторан, а затем просто указать идентификатор меню и установить цену. Но помните одну вещь, если вы используете вторую идею, вам нужно передать идентификатор ресторана в скрытом поле. Надеюсь ты понимаешь
О, ты прав. я думаю, что первый лучше, чем второй.
Еще раз спасибо за помощь.
в сокращении вы можете использовать
$restaurant = Restaurants::find(2)->menus;