Php Yii2以与活动记录的多对多关系更新连接表
我有3个表。例如,数据库中的一个文章表、一个标签表、一个文章标签表。 文章和标签具有N-M关系 当我需要添加新文章时,使用Yii2的activerecord的link()方法将关系保存到连接表,效果很好 但是当我需要更新连接表时。如果我再次对文章调用link()方法。这是行不通的。下面是我的代码和错误信息Php Yii2以与活动记录的多对多关系更新连接表,php,mysql,yii2,Php,Mysql,Yii2,我有3个表。例如,数据库中的一个文章表、一个标签表、一个文章标签表。 文章和标签具有N-M关系 当我需要添加新文章时,使用Yii2的activerecord的link()方法将关系保存到连接表,效果很好 但是当我需要更新连接表时。如果我再次对文章调用link()方法。这是行不通的。下面是我的代码和错误信息 $tag_ids = Yii::$app->request->post('Article')['tags']; foreach ($tag_ids as $value) {
$tag_ids = Yii::$app->request->post('Article')['tags'];
foreach ($tag_ids as $value) {
$tag = Tag::findOne($value);
$model->link('tags', $tag);
}
SQLSTATE[23000]: Integrity constraint violation: 1062 Duplicate entry '13-1' for key 'PRIMARY'
The SQL being executed was: INSERT INTO `article_tag` (`article_id`, `tag_id`) VALUES (13, 1)
我是否需要删除连接表中的所有数据,然后使用link()进行更新??或者Yii2中缺少了一些功能
------------------------更新-----------------------------------------
public function syncTags($new_tag_ids){
$old_tag_ids = ArrayHelper::getColumn($this->tags, 'id');
$tag_to_delete = array_diff($old_tag_ids, $new_tag_ids);
$tag_to_add = array_diff($new_tag_ids, $old_tag_ids);
if($tag_to_delete){
//delete tags
Yii::$app->db->createCommand()
->delete('article_tag', ['article_id' => $this->id, 'tag_id' => $tag_to_delete])
->execute();
}
if($tag_to_add){
//link new tag assisoated with the article
foreach ($tag_to_add as $value) {
$tag = Tag::findOne($value);
$this->link('tags', $tag);
}
}
}
看来我需要自己用纯sql来做。我想到的最简单的方法是首先删除连接表中的数据,然后使用link()再次填充数据透视表。这很简单,但会弄乱索引。而且桌子长得很快。第二种方法是读取每条记录,然后决定保留或删除它。然后添加必要的数据。这种方法使它更复杂。需要更多的代码
----------------再次更新我写了一个函数------------------------
public function syncTags($new_tag_ids){
$old_tag_ids = ArrayHelper::getColumn($this->tags, 'id');
$tag_to_delete = array_diff($old_tag_ids, $new_tag_ids);
$tag_to_add = array_diff($new_tag_ids, $old_tag_ids);
if($tag_to_delete){
//delete tags
Yii::$app->db->createCommand()
->delete('article_tag', ['article_id' => $this->id, 'tag_id' => $tag_to_delete])
->execute();
}
if($tag_to_add){
//link new tag assisoated with the article
foreach ($tag_to_add as $value) {
$tag = Tag::findOne($value);
$this->link('tags', $tag);
}
}
}
它现在正在运行,但不适用于全球。我认为它可能会帮助人们
所以在这里发布。Yii需要扩展这类工作。大多数时候我只使用这种方法。第二个解决方案也很有效,但我更喜欢第一个。我是那种喜欢在没有别人帮助的情况下解决问题的人
/**
* Update categories with the new ones
* @param array $categories [description]
* @return null
*/
public function updateCategories($categories = [])
{
$this->unlinkAll('categories', true);
if ( ! is_array($categories))
return ;
foreach ($categories as $category_id)
{
$category = Category::findOne($category_id);
$this->link('categories', $category);
}
// alternative solution
/*
$old_categories = $this->getCategories()->asArray()->column(); // get all categories IDs
if ( ! is_array($categories))
$categories = [];
$inserted_categories = array_diff($categories, $old_categories);
$deleted_categories = array_diff($old_categories, $categories);
foreach ($inserted_categories as $category_id)
{
$category = Category::findOne($category_id);
$this->link('categories', $category);
}
foreach ($deleted_categories as $category_id)
{
$category = Category::findOne($category_id);
$this->unlink('categories', $category, true);
}
*/
}
你可以尝试使用多对多关系。或者。看起来每个人都为yii2:)编写了自己的多对多扩展。我非常想念Laravel中的sync()方法。节省了我很多时间。@Beowulfenator这对我不适用。因为我在这里使用的是复合主键。@SaidbakR,我对Laravel没有任何经验,很抱歉你的回答是.thks,但我已经有2年没有使用php了,因为我改用python了。。