4

我正在开发一个 PHP Web 应用程序,用户可以在其中创建目的地旅行,并邀请其他用户参加这些旅行。

我有用于DestinationsTrips和的 MySql 表Users,以及一个Invites用于解决和 之间的多对多关系Users的表Trips

对于这些表中的每一个,我在我的应用程序中创建了一个类(Destinations一个类、Trips一个类等)。我为这些类中的每一个编写了一个静态函数,该函数使对象能够被实例化,并从 MySQL 查询中分配它们的属性(因此从SELECT *查询返回的每一行都会导致实例化对象并分配属性)。

我正在尝试创建一个页面,我的用户可以在其中查看他们计划的所有旅行的详细信息。此页面将显示目的地(从Destinations)、旅行详情(例如从 开始的日期Trips)和他们邀请的用户的详细信息(例如来自的用户名Users),并将所有这些数据放在一个 html 表格中,其中每次旅行都是一行。

为了实现这一点,到目前为止,我一直在实例化Trips哪些与 user's相关user_id,哪些DestinationsInvitesthose 相关TripsUsers哪些与 those 相关Invites。然后,我根据需要使用嵌套的 foreach 循环来显示这些,如下所示:

foreach($trips as $trip) {
 foreach($invites as $invite) {
   foreach($invited_users as $invited_user) {
      if($invite->trip_id == $trip->id && $invite->guest_id == $invited_user->id) {
        echo $invited_user->name;
                  } } } }

我可以看到这是可怕的,没有办法做到这一点。

通过阅读,我想我想使用 MySQL 来加入 4 个表,然后只选择与我的相关旅行相关的数据。我不确定的是:

a) 如何从这个 MySQL 结果中实例化对象并正确分配它们的属性?

b)即使完成了,我如何避免昂贵的foreach循环以在正确的行程旁边显示正确的数据?

是否有以面向对象的方式显示来自多个相关数据库表的数据的“最佳实践”方式?

4

2 回答 2

2

首先,“良好实践”是人们用来为自己的偏见披上可敬外衣的短语之一。最好询问如何通过设计实现特定目标。

话虽如此:您偶然发现了一个在将 RDBMS 映射到面向对象系统时很常见的问题。谷歌“延迟加载”进行大量讨论。问题归结为您在任何时候加载多少“关系”的问题。在您的示例中,旅行与目的地和用户相关联;加载“旅行”时,您是否还应该加载目的地和用户?如果你这样做了,你是否还应该为用户加载其他信息(例如他们正在进行的其他旅行?)。如果你这样做了,你会加载用户所有行程的行程数据吗?等等——如果你不小心,你最终会加载整个数据库。

不加载所有相关实体的缺点是,如果一次加载每个对象的数据,性能会变得非常差。

有关我认为最佳实践的详细描述,请购买并阅读 Evans 的“领域驱动设计”。

作为捷径,考虑创建一个“存储库”层,它知道如何将数据库记录映射到对象,并创建自定义方法来搜索您想要的内容。例如,您可以调用:

$repository::getTripsForUser($userID);

该方法将执行将 4 个表连接在一起的 SQL 查询,并将结果转换为对象树。

您可能想利用其他人在解决此问题时所做的工作;有许多“对象关系映射”框架可以解决这个问题。我认为 ORM 解决方案是否让生活更轻松尚无定论 - 但值得一看。

于 2012-11-11T21:25:28.703 回答
0

我认为您使事情复杂化,您不需要一次加载所有数据。

为什么要遍历所有行程,而您需要显示一个用户的行程?

OOP 的一种天真的但简单的方法:一个User具有检索用户相关旅行的方法的类。

它可以返回一个数组Trip,其中包含受邀用户的引用:

class User {
    private $id;
    public function __construct($id) {
        $this->id = $id;
    }
    public function getTrips() {
        $trips = array();
        // Run your query using $this->id to build the WHERE clause
        return $trips;
    }
}

class Trip {

    // Array of User objects invited to the trip        
    private $invited = array();

    // Other methods here to add/retrieve invited users

}

$user = new User(1);
foreach ($user->getTrips() as $trip) {
    // Do something with $trip
}

理想情况下,User实例将存储在会话中

于 2012-11-11T20:57:24.790 回答