- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我需要一些有关使用 PHP 访问数据库的帮助。我正在尝试以面向对象的方式做事,但我不确定我的方向是否正确。
假设我有一个类 Person,例如:
class Person {
private $id;
private $firstname;
private $lastname;
// maybe some more member variables
function __construct($id = NULL) {
if(isset($id)) {
$this->id = $id;
$this->retrieve();
}
}
// getters, setters and other member functions
private function retrieve() {
global $db;
// Retrieve user from database
$stmt = $db->prepare("SELECT firstname, lastname FROM users WHERE id = :id");
$stmt->bindParam(":id", $this->id, PDO::PARAM_INT);
$stmt->execute();
$result = $stmt->fetch();
$this->firstname = $result['firstname'];
$this->lastname = $result['lastname'];
}
function insert() {
global $db;
// Insert object into database, or update if exists
$stmt = $db->prepare("REPLACE INTO users (id, firstname, lastname) VALUES (:id, :firstname, :lastname)");
$stmt->bindParam(":id", $this->id, PDO::PARAM_INT);
$stmt->bindParam(":firstname", $this->firstname, PDO::PARAM_STR);
$stmt->bindParam(":lastname", $this->lastname, PDO::PARAM_STR);
$stmt->execute();
}
}
请注意,这只是我为描述我的问题而编写的示例,而不是我在应用程序中使用的实际代码。
现在我的第一个问题是:这是处理数据库交互的正确方法吗?我认为这是一个好方法,因为您可以实例化一个对象,对其进行操作,然后再次插入/更新它。
换句话说:在实例化/使用类的代码中,在类的内部(如我的示例)还是在类的外部处理数据库交互更好? ?
我的第二个问题是关于更新一大堆可能已修改或未修改的行。假设类 Person 有一个成员变量 $pets[],它是一个包含该人拥有的所有宠物的数组。宠物存储在数据库中的单独表中,如下所示:
+---------+-------------+---------+
| Field | Type | Key |
+---------+-------------+---------+
| pet_id | int(11) | PRI |
| user_id | int(11) | MUL |
| name | varchar(25) | |
+---------+-------------+---------+
假设我修改了 Person 对象中的一些宠物。也许我添加或删除了一些宠物,也许我只更新了一些宠物的名字。
在这种情况下更新整个人(包括他们的宠物)的最佳方法是什么?假设一个人有 50 只宠物,我是否只更新所有宠物,即使只有一只宠物发生了变化?
我希望这已经够清楚了 ;)
编辑:
更重要的是,如何同时处理删除/插入?我目前的方法是,在“编辑页面”上,我检索某个人(包括他们的宠物)并以一种形式显示/打印它们供用户编辑。然后用户可以编辑宠物、添加新宠物或删除一些宠物。当用户单击“应用”按钮时,表单将被 POST 回 PHP 脚本。
我能想到的将这些更改更新到数据库中的唯一方法是删除数据库中当前列出的所有宠物,然后插入新的一组宠物。但这有一些问题:首先,每次编辑时都会删除并重新插入 pets 表中的所有行,其次,自动增量 ID 每次都会因此而发生巨大的飞跃。
我觉得我做错了什么。是否不能让用户同时删除/添加宠物和修改现有宠物(我应该单独处理这些操作)吗?
最佳答案
您要完成的任务是称为对象关系映射(将对象映射到关系数据库中的表,反之亦然)。关于这一点已经写了整本书,但我将尝试给出一个简短的概述。
Now my first question is: is this the correct way to handle database interaction? I thought this would be a good way because you can instantiate an object, manipulate it, then insert/update it again.
这是一个有效的方法。然而,一般来说,你应该尽量遵守 Separation of Concerns .在我看来,建模一个领域实体(就像一个人,在你的例子中)和将这个对象存储在数据库中/从数据库中加载是两个(可以说是三个)不同的问题,应该在单独的类中实现(再次,个人意见!)。这使得对您的类进行单元测试变得非常困难,并增加了很多复杂性。
关于 ORM,随着时间的推移出现了多种设计模式。最突出的是:
Active Record 基本上是您已经在问题中建议的方法;它将数据和数据访问逻辑紧密耦合在一个对象中。在我看来这不是最好的方法,因为它违反了关注点分离,但可能最容易实现。
Gateways 或 Mappers :尝试创建一个单独的类来访问您的 Persons 表(类似于 PersonGateway
。这样,您的 Person
类只包含数据和各种行为,和您的 PersonGateway
各种插入/更新/删除方法。另一方面,映射器 可能是一个转换通用数据库结果对象的类(例如由PDO 查询)转换为 Person 对象(在这种情况下,Person 类不需要知道此类映射器类的存在)。
What is the best way to update the whole Person, including their pets in that case? Lets say one Person has 50 pets, do I just update them all even if only one of them has changed?
同样,有几种可能性。在任何情况下,您都应该将我们的 Pet 映射为一个单独的类。
将宠物保存在单独的表中,并为此类实现您自己的数据访问逻辑(例如使用上述模式之一)。然后,您可以随意单独更新、插入或删除它们。您可以跟踪父对象(人)中添加/删除的宠物,然后在保留父对象时级联更新操作。
Embed Persons 表中的 pets 集合。不过,根据此集合的平均大小以及您可能希望查询它们的方式,这可能不是一个好主意。
如果您的项目变得复杂,您可能还想看看为您解决这些问题的 ORM 框架(例如 Doctrine ORM)。
关于php - 在 PHP OOP 中处理数据库访问的正确方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30006653/
关闭。这个问题是opinion-based .它目前不接受答案。 想改善这个问题吗?更新问题,以便可以通过 editing this post 用事实和引文回答问题. 4年前关闭。 Improve t
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
在面向对象的编程中,“基类”是派生其他类的类(http://en.wikipedia.org/wiki/Base_class)。 但是,基类的反面是什么?换句话说,什么是没有任何子类的类? 编辑:我正
关闭。这个问题需要更多focused .它目前不接受答案。 想改善这个问题吗?更新问题,使其仅关注一个问题 editing this post . 8年前关闭。 Improve this questi
据我了解,OOP 是大型项目最常用的范式。我也知道大系统的一些较小的子集使用其他范式(例如 SQL,它是声明性的),并且我也意识到在较低级别的计算 OOP 并不真正可行。但在我看来,通常更高级别的解决
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
最近听说OOP(Java)有9条规则。我只知道四种:抽象、多态、继承和封装。 OOP 还有更多规则吗? 最佳答案 看来您正在寻找的是 Principles of Object-Oriented Des
我曾经在一次采访中被问到“OOP 的 3 个主要概念是什么?”。我回答说,我认为有4个,如下: 继承 封装 抽象 多态性 我说得对吗? 最佳答案 语言要成为面向对象有3个要求: 仅支持封装(对象)的语
我有一个关于特定 OOP 问题的组织的简单问题。 假设我有一个地形类,里面充满了瓷砖。 Tile 类有多个派生类,即 Door。 Door 类有一个名为 open() 的方法,用于打开门,还有一个名为
就目前情况而言,这个问题不太适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、民意调查或扩展讨论。如果您觉得这个问题可以改进并可能重新开放,visit
我是 Go 的新手,然后我通过示例搜索了很多如何拥有带有静态函数/变量的静态类,例如 C#。但是,我找不到任何可以很好地回答它的东西。也许这个问题看起来很愚蠢,但我不喜欢不确定或不完全理解某事。 假设
我曾尝试搜索此问题的答案,但很难用语言表达,而且许多问题要么是关于如何创建类,要么是关于如何做非常具体的事情。我需要更多的实用概述 - 我是自学成才的,我了解对象是什么(以及如何创建它们),但我从未见
在开始编码之前,我通常会尝试在没有太多分析(没有图表)的情况下进行 TDD。通常我发现自己将一个类拆分为其他类以分离关注点。我想知道更深入的分析是否会阻止这种情况。我认为大部分面向对象分析无法预测其中
在阅读单例时,我发现这个解释是使用单例的原因: since these object methods are not changing the internal class state, we can
如这里所述 https://standardofnorms.wordpress.com/2012/09/02/4-pillars-of-object-oriented-programming/ 并作为
我是这个网站的新手,所以如果我在发布问题时做错了什么,请告诉我,以便我下次修复。 我很好奇从单个基类继承多个类是否是糟糕的 OOP 实践。这可能不太合理,所以我要详细说明一下。 例如,假设您正在设计一
我对“工厂相关”设计模式及其 OOP 实现的理解一直很简单。 一个 《工厂法》是类内部的一个方法,它有一个接口(interface)(或抽象类)作为返回类型,并根据一些内部逻辑构造实现该接口(inte
C# 中的“密封”关键字,Java 中的 Final。 因为我几乎从不创建任何图表并且我只使用已经完成的类(来自框架)我多年后仍然不知道为什么有人会“锁定”一个类所以它永远不会被扩展/继承。 它是有益
我正在研究面向对象的概念,抽象概念基本上被描述为对用户隐藏实现。因此,如果类中有一个成员函数并且我们为某些任务调用该函数,抽象表示用户不应该关心事情是如何完成的,而应该只知道正在完成什么。但即使在非面
我是一名优秀的程序员,十分优秀!