1
0
mirror of synced 2025-01-05 16:53:21 +03:00

Updated Aggregate Fields, Added generated SQL for all Association Mappings

This commit is contained in:
Benjamin Eberlei 2010-07-18 21:12:51 +02:00
parent 59aafe100b
commit 4c813f60f9
2 changed files with 221 additions and 82 deletions

View File

@ -1,9 +1,7 @@
# Aggregate Fields # Aggregate Fields
You will is often the requirement to display aggregate values of data that You will often come across the requirement to display aggregate values of data that
can be computed by using the MIN, MAX, COUNT or SUM SQL functions. Doctrine 2 can be computed by using the MIN, MAX, COUNT or SUM SQL functions. For any ORM this is a tricky issue traditionally. Doctrine 2 offers several ways to get access to these values and this article will describe all of them from different perspectives.
offers several ways to get access to these values and this article will
describe all of them from different perspectives.
You will see that aggregate fields can become very explicit You will see that aggregate fields can become very explicit
features in your domain model and how this potentially complex business rules features in your domain model and how this potentially complex business rules
@ -34,7 +32,7 @@ Our entities look like:
/** @Id @GeneratedValue @Column(type="integer") */ /** @Id @GeneratedValue @Column(type="integer") */
private $id; private $id;
/** @Column(type="string", unique=true) /** @Column(type="string", unique=true) */
private $no; private $no;
/** /**
@ -156,7 +154,7 @@ relation with this method:
{ {
$this->assertAcceptEntryAllowed($amount); $this->assertAcceptEntryAllowed($amount);
$e = new Entry($e, $amount); $e = new Entry($this, $amount);
$this->entries[] = $e; $this->entries[] = $e;
return $e; return $e;
} }
@ -184,7 +182,7 @@ Now look at the following test-code for our entities:
$account = new Account("123456", $maxCredit = 200); $account = new Account("123456", $maxCredit = 200);
$this->setExpectedException("Exception"); $this->setExpectedException("Exception");
$account->addEntriy(-1000); $account->addEntry(-1000);
} }
} }
@ -242,7 +240,7 @@ code in `Account::getBalance()` and `Account:addEntry()`:
{ {
$this->assertAcceptEntryAllowed($amount); $this->assertAcceptEntryAllowed($amount);
$e = new Entry($e, $amount); $e = new Entry($this, $amount);
$this->entries[] = $e; $this->entries[] = $e;
$this->balance += $amount; $this->balance += $amount;
return $e; return $e;

View File

@ -81,27 +81,29 @@ This is essentially the same as the following, more verbose, mapping:
The @JoinTable definition used for many-to-many mappings has similar defaults. As an example, consider this mapping: The @JoinTable definition used for many-to-many mappings has similar defaults. As an example, consider this mapping:
[php] [php]
class User { class User
//... {
/** @ManyToMany(targetEntity="Group") */ //...
private $groups; /** @ManyToMany(targetEntity="Group") */
//... private $groups;
//...
} }
This is essentially the same as the following, more verbose, mapping: This is essentially the same as the following, more verbose, mapping:
[php] [php]
class User { class User
//... {
/** //...
* @ManyToMany(targetEntity="Group") /**
* @JoinTable(name="User_Group", * @ManyToMany(targetEntity="Group")
* joinColumns={@JoinColumn(name="User_id", referencedColumnName="id")}, * @JoinTable(name="User_Group",
* inverseJoinColumns={@JoinColumn(name="Group_id", referencedColumnName="id")} * joinColumns={@JoinColumn(name="User_id", referencedColumnName="id")},
* ) * inverseJoinColumns={@JoinColumn(name="Group_id", referencedColumnName="id")}
*/ * )
private $groups; */
//... private $groups;
//...
} }
In that case, the name of the join table defaults to a combination of the simple, unqualified class names of the participating classes, separated by an underscore character. The names of the join columns default to the simple, unqualified class name of the targeted class followed by "_id". The referencedColumnName always defaults to "id", just as in one-to-one or many-to-one mappings. In that case, the name of the join table defaults to a combination of the simple, unqualified class names of the participating classes, separated by an underscore character. The names of the join columns default to the simple, unqualified class name of the targeted class followed by "_id". The referencedColumnName always defaults to "id", just as in one-to-one or many-to-one mappings.
@ -135,6 +137,20 @@ A unidirectional one-to-one association is very common. Here is an example of a
Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same. Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same.
Generated MySQL Schema:
[sql]
CREATE TABLE Product (
id INT AUTO_INCREMENT NOT NULL,
shipping_id INT DEFAULT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
CREATE TABLE Shipping (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE Product ADD FOREIGN KEY (shipping_id) REFERENCES Shipping(id);
++ One-To-One, Bidirectional ++ One-To-One, Bidirectional
Here is a one-to-one relationship between a `Customer` and a `Cart`. The `Cart` Here is a one-to-one relationship between a `Customer` and a `Cart`. The `Cart`
@ -170,18 +186,34 @@ has a reference back to the `Customer` so it is bidirectional.
Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same. Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same.
Generated MySQL Schema:
[sql]
CREATE TABLE Cart (
id INT AUTO_INCREMENT NOT NULL,
customer_id INT DEFAULT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
CREATE TABLE Customer (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE Cart ADD FOREIGN KEY (customer_id) REFERENCES Customer(id);
See how the foreign key is defined on the owning side of the relation, the table `Cart`.
++ One-To-One, Self-referencing ++ One-To-One, Self-referencing
You can easily have self referencing one-to-one relationships like below. You can easily have self referencing one-to-one relationships like below.
[php] [php]
/** @Entity */ /** @Entity */
class Customer class Student
{ {
// ... // ...
/** /**
* @OneToOne(targetEntity="Customer") * @OneToOne(targetEntity="Student")
* @JoinColumn(name="mentor_id", referencedColumnName="id") * @JoinColumn(name="mentor_id", referencedColumnName="id")
*/ */
private $mentor; private $mentor;
@ -191,6 +223,16 @@ You can easily have self referencing one-to-one relationships like below.
Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same. Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same.
With the generated MySQL Schema:
[sql]
CREATE TABLE Student (
id INT AUTO_INCREMENT NOT NULL,
mentor_id INT DEFAULT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE Student ADD FOREIGN KEY (mentor_id) REFERENCES Student(id);
++ One-To-Many, Unidirectional with Join Table ++ One-To-Many, Unidirectional with Join Table
A unidirectional one-to-many association can be mapped through a join table. From Doctrine's point of view, it is simply mapped as a unidirectional many-to-many whereby a unique constraint on one of the join columns enforces the one-to-many cardinality. A unidirectional one-to-many association can be mapped through a join table. From Doctrine's point of view, it is simply mapped as a unidirectional many-to-many whereby a unique constraint on one of the join columns enforces the one-to-many cardinality.
@ -200,18 +242,18 @@ The following example sets up such a unidirectional one-to-many association:
/** @Entity */ /** @Entity */
class User class User
{ {
// ... // ...
/** /**
* @ManyToMany(targetEntity="Phonenumber") * @ManyToMany(targetEntity="Phonenumber")
* @JoinTable(name="users_phonenumbers", * @JoinTable(name="users_phonenumbers",
* joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")}, * joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")},
* inverseJoinColumns={@JoinColumn(name="phonenumber_id", referencedColumnName="id", unique=true)} * inverseJoinColumns={@JoinColumn(name="phonenumber_id", referencedColumnName="id", unique=true)}
* ) * )
*/ */
private $phonenumbers; private $phonenumbers;
// ... // ...
} }
/** @Entity */ /** @Entity */
@ -223,6 +265,29 @@ The following example sets up such a unidirectional one-to-many association:
> **NOTE** > **NOTE**
> One-To-Many uni-directional relations with join-table only work using the @ManyToMany annotation and a unique-constraint. > One-To-Many uni-directional relations with join-table only work using the @ManyToMany annotation and a unique-constraint.
Generates the following MySQL Schema:
[sql]
CREATE TABLE User (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
CREATE TABLE users_phonenumbers (
user_id INT NOT NULL,
phonenumber_id INT NOT NULL,
UNIQUE INDEX users_phonenumbers_phonenumber_id_uniq (phonenumber_id),
PRIMARY KEY(user_id, phonenumber_id)
) ENGINE = InnoDB;
CREATE TABLE Phonenumber (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE users_phonenumbers ADD FOREIGN KEY (user_id) REFERENCES User(id);
ALTER TABLE users_phonenumbers ADD FOREIGN KEY (phonenumber_id) REFERENCES Phonenumber(id);
++ Many-To-One, Unidirectional ++ Many-To-One, Unidirectional
You can easily implement a many-to-one unidirectional association with the following: You can easily implement a many-to-one unidirectional association with the following:
@ -237,7 +302,7 @@ You can easily implement a many-to-one unidirectional association with the follo
* @ManyToOne(targetEntity="Address") * @ManyToOne(targetEntity="Address")
* @JoinColumn(name="address_id", referencedColumnName="id") * @JoinColumn(name="address_id", referencedColumnName="id")
*/ */
private $address private $address;
} }
/** @Entity */ /** @Entity */
@ -250,6 +315,22 @@ You can easily implement a many-to-one unidirectional association with the follo
> The above `@JoinColumn` is optional as it would default to `address_id` and `id` > The above `@JoinColumn` is optional as it would default to `address_id` and `id`
> anyways. You can omit it and let it use the defaults. > anyways. You can omit it and let it use the defaults.
Generated MySQL Schema:
[sql]
CREATE TABLE User (
id INT AUTO_INCREMENT NOT NULL,
address_id INT DEFAULT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
CREATE TABLE Address (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE User ADD FOREIGN KEY (address_id) REFERENCES Address(id);
++ One-To-Many, Bidirectional ++ One-To-Many, Bidirectional
Bidirectional one-to-many associations are very common. The following code shows an example with a Product and a Feature class: Bidirectional one-to-many associations are very common. The following code shows an example with a Product and a Feature class:
@ -280,6 +361,20 @@ Bidirectional one-to-many associations are very common. The following code shows
Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same. Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same.
Generated MySQL Schema:
[sql]
CREATE TABLE Product (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
CREATE TABLE Feature (
id INT AUTO_INCREMENT NOT NULL,
product_id INT DEFAULT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE Feature ADD FOREIGN KEY (product_id) REFERENCES Product(id);
++ One-To-Many, Self-referencing ++ One-To-Many, Self-referencing
You can also setup a one-to-many association that is self-referencing. In this example we You can also setup a one-to-many association that is self-referencing. In this example we
@ -306,6 +401,16 @@ This effectively models a hierarchy of categories and from the database perspect
Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same. Note that the @JoinColumn is not really necessary in this example, as the defaults would be the same.
Generated MySQL Schema:
[sql]
CREATE TABLE Category (
id INT AUTO_INCREMENT NOT NULL,
parent_id INT DEFAULT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE Category ADD FOREIGN KEY (parent_id) REFERENCES Category(id);
++ Many-To-Many, Unidirectional ++ Many-To-Many, Unidirectional
Real many-to-many associations are less common. The following example shows a unidirectional association between User and Group entities: Real many-to-many associations are less common. The following example shows a unidirectional association between User and Group entities:
@ -314,18 +419,18 @@ Real many-to-many associations are less common. The following example shows a un
/** @Entity */ /** @Entity */
class User class User
{ {
// ... // ...
/** /**
* @ManyToMany(targetEntity="Group") * @ManyToMany(targetEntity="Group")
* @JoinTable(name="users_groups", * @JoinTable(name="users_groups",
* joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")}, * joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")},
* inverseJoinColumns={@JoinColumn(name="group_id", referencedColumnName="id")} * inverseJoinColumns={@JoinColumn(name="group_id", referencedColumnName="id")}
* ) * )
*/ */
private $groups; private $groups;
// ... // ...
} }
/** @Entity */ /** @Entity */
@ -340,6 +445,25 @@ Real many-to-many associations are less common. The following example shows a un
> class. Consequently, the direct many-to-many association disappears and is replaced > class. Consequently, the direct many-to-many association disappears and is replaced
> by one-to-many/many-to-one associations between the 3 participating classes. > by one-to-many/many-to-one associations between the 3 participating classes.
Generated MySQL Schema:
[sql]
CREATE TABLE User (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
CREATE TABLE users_groups (
user_id INT NOT NULL,
group_id INT NOT NULL,
PRIMARY KEY(user_id, group_id)
) ENGINE = InnoDB;
CREATE TABLE Group (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
ALTER TABLE users_groups ADD FOREIGN KEY (user_id) REFERENCES User(id);
ALTER TABLE users_groups ADD FOREIGN KEY (group_id) REFERENCES Group(id);
++ Many-To-Many, Bidirectional ++ Many-To-Many, Bidirectional
Here is a similar many-to-many relationship as above except this one is bidirectional. Here is a similar many-to-many relationship as above except this one is bidirectional.
@ -348,18 +472,18 @@ Here is a similar many-to-many relationship as above except this one is bidirect
/** @Entity */ /** @Entity */
class User class User
{ {
// ... // ...
/** /**
* @ManyToMany(targetEntity="Group", inversedBy="users") * @ManyToMany(targetEntity="Group", inversedBy="users")
* @JoinTable(name="users_groups", * @JoinTable(name="users_groups",
* joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")}, * joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")},
* inverseJoinColumns={@JoinColumn(name="group_id", referencedColumnName="id")} * inverseJoinColumns={@JoinColumn(name="group_id", referencedColumnName="id")}
* ) * )
*/ */
private $groups; private $groups;
// ... // ...
} }
/** @Entity */ /** @Entity */
@ -373,6 +497,8 @@ Here is a similar many-to-many relationship as above except this one is bidirect
// ... // ...
} }
The MySQL schema is exactly the same as for the Many-To-Many uni-directional case above.
++ Many-To-Many, Self-referencing ++ Many-To-Many, Self-referencing
You can even have a self-referencing many-to-many association. A common scenario is where a `User` has friends and the target entity of that relationship is a `User` so it is self referencing. In this example it is bidirectional so `User` has a field named `$friendsWithMe` and `$myFriends`. You can even have a self-referencing many-to-many association. A common scenario is where a `User` has friends and the target entity of that relationship is a `User` so it is self referencing. In this example it is bidirectional so `User` has a field named `$friendsWithMe` and `$myFriends`.
@ -381,25 +507,40 @@ You can even have a self-referencing many-to-many association. A common scenario
/** @Entity */ /** @Entity */
class User class User
{ {
// ... // ...
/** /**
* @ManyToMany(targetEntity="User", mappedBy="myFriends") * @ManyToMany(targetEntity="User", mappedBy="myFriends")
*/ */
private $friendsWithMe; private $friendsWithMe;
/** /**
* @ManyToMany(targetEntity="User", inversedBy="friendsWithMe") * @ManyToMany(targetEntity="User", inversedBy="friendsWithMe")
* @JoinTable(name="friends", * @JoinTable(name="friends",
* joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")}, * joinColumns={@JoinColumn(name="user_id", referencedColumnName="id")},
* inverseJoinColumns={@JoinColumn(name="friend_user_id", referencedColumnName="id")} * inverseJoinColumns={@JoinColumn(name="friend_user_id", referencedColumnName="id")}
* ) * )
*/ */
private $myFriends; private $myFriends;
// ... // ...
} }
Generated MySQL Schema:
[sql]
CREATE TABLE User (
id INT AUTO_INCREMENT NOT NULL,
PRIMARY KEY(id)
) ENGINE = InnoDB;
CREATE TABLE friends (
user_id INT NOT NULL,
friend_user_id INT NOT NULL,
PRIMARY KEY(user_id, friend_user_id)
) ENGINE = InnoDB;
ALTER TABLE friends ADD FOREIGN KEY (user_id) REFERENCES User(id);
ALTER TABLE friends ADD FOREIGN KEY (friend_user_id) REFERENCES User(id);
++ Ordering To-Many Collections ++ Ordering To-Many Collections
In many use-cases you will want to sort collections when they are retrieved from the database. In many use-cases you will want to sort collections when they are retrieved from the database.
@ -415,13 +556,13 @@ in the following way:
/** @Entity */ /** @Entity */
class User class User
{ {
// ... // ...
/** /**
* @ManyToMany(targetEntity="Group") * @ManyToMany(targetEntity="Group")
* @OrderBy({"name" = "ASC"}) * @OrderBy({"name" = "ASC"})
*/ */
private $groups; private $groups;
} }
The DQL Snippet in OrderBy is only allowed to consist of unqualified, The DQL Snippet in OrderBy is only allowed to consist of unqualified,
@ -446,19 +587,19 @@ Given our previously defined example, the following would not add ORDER BY, sinc
However the following: However the following:
[sql] [sql]
SELECT u FROM User u JOIN u.groups g WHERE u.id = 10 SELECT u, g FROM User u JOIN u.groups g WHERE u.id = 10
...would internally be rewritten to: ...would internally be rewritten to:
[sql] [sql]
SELECT u FROM User u JOIN u.groups g WHERE u.id = 10 ORDER BY g.name ASC SELECT u, g FROM User u JOIN u.groups g WHERE u.id = 10 ORDER BY g.name ASC
You can't reverse the order, an explicit: You can't reverse the order with an explicit DQL ORDER BY:
[sql] [sql]
SELECT u FROM User u JOIN u.groups g WHERE u.id = 10 ORDER BY g.name DESC SELECT u, g FROM User u JOIN u.groups g WHERE u.id = 10 ORDER BY g.name DESC
...is internally rewritten to: ...is internally rewritten to:
[sql] [sql]
SELECT u FROM User u JOIN u.groups g WHERE u.id = 10 ORDER BY g.name DESC, g.name ASC SELECT u, g FROM User u JOIN u.groups g WHERE u.id = 10 ORDER BY g.name DESC, g.name ASC