Страница 1 из 1
Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:23
theorist
Приветствую, коллеги.
Имеется 3 таблицы:
-
product (id, title),
-
category (id, title),
-
product_has_category (id, product_id, category_id)
Подразумевается, что товар может находиться сразу в нескольких категориях, что понятно из существования промежуточной таблицы.
Никак у меня не получается настроить связь у модели категории, которая бы возвращала список продуктов из промежуточной таблицы по id категории. Пробовал так (в модели категории):
Код: Выделить всё
public function getProducts() {
return $this->hasMany(Product::className(), ['id'=>'category_id'])
->viaTable('sk_product_has_category', ['category_id'=>'product_id']);
}
Коючи в связях пробовал разные, я, видимо, не до конца понимаю, как работает viaTable.
Буду очень благодарен за помощь.
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:25
andrei.obuhovski
а так?
Код: Выделить всё
public function getProducts() {
return $this->hasMany(Product::className(), ['id'=>'product_id'])
->viaTable('sk_product_has_category', ['category_id'=>'id']);
}
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:27
theorist
SQLSTATE[23000]: Integrity constraint violation: 1052 Column 'id' in where clause is ambiguous
А запрос генерится такой: SELECT `sk_category`.* FROM `sk_category` LEFT JOIN `sk_product_has_category` ON `sk_category`.`id` = `sk_product_has_category`.`category_id` LEFT JOIN `sk_product` ON `sk_product_has_category`.`product_id` = `sk_product`.`id` WHERE `id`=1
Я попробовал в условии в конце явно задать, из какой таблицы проверочный id (указал sk_category.id вместо id) - результат - возвращает 7 раз категорию, в которой есть товары (кстати, у этой категории как раз и должно быть 7 товаров).
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:37
lynicidn
алиасы нужны обязательно
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:40
theorist
Даже с ними (когда пробую прямой SQL-запрос выполнить) возвращается 7 категорий вместо 7 товаров этой категории.
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:43
lynicidn
так ты выбираешь сам категории - в запрос свой смотри - селект категори
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:46
theorist
Ага, как тогда связь прописать, чтобы продукты доставались?
В этом и был вопрос.
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:49
lynicidn
select `products`.* from `products` inner join `category` on `products`.`category_id` = `category`.`id`
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:53
theorist
Не то выбирает ваш запрос.
Можно даже не проверять. Продукт содержит категории в промежуточной таблице, а вы ее даже не упомянули.
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 11:57
lynicidn
я не пытался за вас решить проблему, я лишь показал примерно как должны выглядеть выборка
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 12:13
theorist
Запросом я может кое-как и достану, хотелось связь красиво прописать.
Но спасибо в любом случае.
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.28, 12:15
andrei.obuhovski
theorist писал(а):
Я попробовал в условии в конце явно задать, из какой таблицы проверочный id (указал sk_category.id вместо id) - результат - возвращает 7 раз категорию, в которой есть товары (кстати, у этой категории как раз и должно быть 7 товаров).
distinct ?
Re: Связь через промежуточную таблицу
Добавлено: 2016.01.29, 00:47
magicoder
Для работы со связными данными посредством Active Record вы прежде всего должны объявить связи в классе Active Record
При проектировании баз данных, когда между двумя таблицами имеется кратность связи many-to-many, обычно вводится промежуточная таблица. Например, таблицы order и item могут быть связаны посредством промежуточной таблицы с названием order_item. Один заказ будет соотносится с несколькими товарами, в то время как один товар будет также соотноситься с несколькими заказами.
При объявлении подобных связей вы можете пользоваться методом [[yii\db\ActiveQuery::via()|via()]] или методом [[yii\db\ActiveQuery::viaTable()|viaTable()]] для указания промежуточной таблицы. Разница между методами [[yii\db\ActiveQuery::via()|via()]] и [[yii\db\ActiveQuery::viaTable()|viaTable()]] заключается в том, что первый метод указывает промежуточную таблицу с помощью названия связи, в то время как второй метод непосредственно указывает промежуточную таблицу. Например:
Код: Выделить всё
class Order extends ActiveRecord
{
public function getItems()
{
return $this->hasMany(Item::className(), ['id' => 'item_id'])
->viaTable('order_item', ['order_id' => 'id']);
}
}
или по-другому:
Код: Выделить всё
class Order extends ActiveRecord
{
public function getOrderItems()
{
return $this->hasMany(OrderItem::className(), ['order_id' => 'id']);
}
public function getItems()
{
return $this->hasMany(Item::className(), ['id' => 'item_id'])
->via('orderItems');
}
}
Использовать связи, объявленные с помощью промежуточных таблиц, можно точно также, как и обычные связи. Например:
Код: Выделить всё
// SELECT * FROM `order` WHERE `id` = 100
$order = Order::findOne(100);
// SELECT * FROM `order_item` WHERE `order_id` = 100
// SELECT * FROM `item` WHERE `item_id` IN (...)
// возвращает массив объектов Item
$items = $order->items;
Re: Связь через промежуточную таблицу
Добавлено: 2018.08.29, 14:13
dymsonn
У меня сейчас похожая проблема. Как при подобном описании функции
magicoder писал(а): ↑2016.01.29, 00:47
Код: Выделить всё
class Order extends ActiveRecord
{
public function getItems()
{
return $this->hasMany(Item::className(), ['id' => 'item_id'])
->viaTable('order_item', ['order_id' => 'id']);
}
}
использовать ее на форме добавления нового элемента?
Допустим на странице order/create.php форма
Код: Выделить всё
<?= $form->field($model->items, 'name')->textInput() ?>
не работает.
Re: Связь через промежуточную таблицу
Добавлено: 2018.08.31, 12:49
andku83
Код: Выделить всё
foreach($model->items as $item) {
echo $form->field($item, 'name['.$item->id.']')->textInput();
}