gpt4 book ai didi

php - Yii2 更新与 Active Record 多对多关系的联结表

转载 作者:可可西里 更新时间:2023-11-01 07:08:49 26 4
gpt4 key购买 nike

我有3个表,例如数据库中的一个article表,一个tag表,一个article_tag表。文章和标签是N-M关系。

当我需要添加新文章时,使用 Yii2 的事件记录的 link() 方法将关系保存到联结表,它工作正常。

但是当我需要更新联结表时。如果我再次调用文章的 link() 方法。不起作用。下面是我的代码和错误信息。

$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 缺少某些功能?

------------------------更新-------------------- ------------------

看来我需要自己用纯sql来做。我想到的最简单的方法是先删除联结表中的数据,然后再次使用 link() 填充数据透视表。这很简单,但会弄乱索引。 table 也增长得很快。第二种方法是读取每条记录,然后决定保留或删除它。然后添加必要的数据。这种方法使它更复杂。需要更多代码。

----------------再次更新我写了一个函数------------------------

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);
}
*/
}

关于php - Yii2 更新与 Active Record 多对多关系的联结表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33753923/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com