在 Symfony3 中使用 Doctrine 实现朋友关系

mat*_*tth 2 symfony doctrine-orm

根据使用 Doctrine 在 Symfony2.1 中实现朋友列表的帖子,我实现了@Roberto Trunfio 的解决方案。

/**
 * @ORM\ManyToMany(targetEntity="User", mappedBy="friends")
 */
private $friendsWithMe;

/**
 * @ORM\ManyToMany(targetEntity="User", inversedBy="friendsWithMe")
 * @ORM\JoinTable(name="friends",
 *      joinColumns={@ORM\JoinColumn(name="user_id", referencedColumnName="id")},
 *      inverseJoinColumns={@ORM\JoinColumn(name="friend_user_id", referencedColumnName="id")}
 *      )
 */
private $friends;
Run Code Online (Sandbox Code Playgroud)

它有效,但是我想通过添加额外的字段来更进一步,例如“发送者、接收者、状态、发送日期......”,但我不知道如何集成它。有人可以帮我吗?谢谢

Lum*_*men 5

如果你想用额外的属性装饰关联,你需要一个关联类。从文档(向下滚动一点):

为什么多对多关联不太常见?因为您经常希望将附加属性与关联相关联,在这种情况下,您需要引入关联类。因此,直接的多对多关联消失了,取而代之的是 3 个参与类之间的一对多/多对一关联。

您示例中的关联类是friendship。一个用户可以有很多朋友,一个用户可以是很多用户的朋友。或者更专业地说:一个用户有很多朋友,很多朋友映射到一个朋友。在下面给出的示例中,Friendship具有附加属性$hasBeenHelpful(实际上是不对称的)。

// src/AppBundle/Entity/User.php
/**
 * @ORM\Entity
 */
class User
{
    /**
     * @ORM\Column(type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @ORM\Column(type="string")
     */
    private $name;

    /**
     * The people who I think are my friends.
     *
     * @ORM\OneToMany(targetEntity="Friendship", mappedBy="user")
     */
    private $friends;

    /**
     * The people who think that I’m their friend.
     *
     * @ORM\OneToMany(targetEntity="Friendship", mappedBy="friend")
     */
    private $friendsWithMe;

    // …
}
Run Code Online (Sandbox Code Playgroud)

和友谊协会:

// src/AppBundle/Entity/Friendship.php
/**
 * @ORM\Entity
 */
class Friendship
{
    /**
     * @ORM\ManyToOne(targetEntity="User", inversedBy="friends")
     * @ORM\Id
     */
    private $user;

    /**
     * @ORM\ManyToOne(targetEntity="User", inversedBy="friendsWithMe")
     * @ORM\Id
     */
    private $friend;

    /**
     * Example of an additional attribute.
     *
     * @ORM\Column(type="boolean")
     */
    private $hasBeenHelpful;

    // …
}
Run Code Online (Sandbox Code Playgroud)

您可能想在 User 类中添加一些函数,例如

<?php

use Doctrine\Common\Collections\ArrayCollection;

class User
{
    public function __construct()
    {
        $this->friends = new ArrayCollection();
        $this->friendsWithMe = new ArrayCollection();
    }

    public function addFriendship(Friendship $friendship)
    {
        $this->friends->add($friendship);
        $friendship->friend->addFriendshipWithMe($friendship);
    }

    public function addFriendshipWithMe(Friendship $friendship)
    {
        $this->friendsWithMe->add($friendship);
    }

    public function addFriend(User $friend)
    {
        $fs = new Friendship();
        $fs->setUser($this);
        $fs->setFriend($friend);
        // set defaults
        $fs->setHasBeenUseful(true);

        $this->addFriendship($fs);
    }
}
Run Code Online (Sandbox Code Playgroud)