Merge pull request #1544 from evildmp/ticket_20920_rebase

Fixed #20920 -- Consolidated F() and Q() documentation
This commit is contained in:
Daniele Procida 2013-09-06 11:27:58 -07:00
commit 263eecc583
13 changed files with 179 additions and 64 deletions

View File

@ -65,7 +65,8 @@ manipulating the data of your Web application. Learn more about it below:
* **QuerySets:** * **QuerySets:**
:doc:`Executing queries <topics/db/queries>` | :doc:`Executing queries <topics/db/queries>` |
:doc:`QuerySet method reference <ref/models/querysets>` :doc:`QuerySet method reference <ref/models/querysets>` |
:doc:`Query-related classes <ref/models/queries>`
* **Model instances:** * **Model instances:**
:doc:`Instance methods <ref/models/instances>` | :doc:`Instance methods <ref/models/instances>` |

View File

@ -1062,11 +1062,12 @@ define the details of how the relation works.
only allows the choice of related objects with a ``pub_date`` before the only allows the choice of related objects with a ``pub_date`` before the
current date to be chosen. current date to be chosen.
Instead of a dictionary this can also be a :class:`~django.db.models.Q` Instead of a dictionary this can also be a :class:`Q object
object for more :ref:`complex queries <complex-lookups-with-q>`. However, <django.db.models.Q>` for more :ref:`complex queries
if ``limit_choices_to`` is a :class:`~django.db.models.Q` object then it <complex-lookups-with-q>`. However, if ``limit_choices_to`` is a :class:`Q
will only have an effect on the choices available in the admin when the object <django.db.models.Q>` then it will only have an effect on the
field is not listed in ``raw_id_fields`` in the ``ModelAdmin`` for the model. choices available in the admin when the field is not listed in
``raw_id_fields`` in the ``ModelAdmin`` for the model.
.. attribute:: ForeignKey.related_name .. attribute:: ForeignKey.related_name

View File

@ -12,3 +12,4 @@ Model API reference. For introductory material, see :doc:`/topics/db/models`.
options options
instances instances
querysets querysets
queries

View File

@ -342,6 +342,8 @@ only.
Using ``update_fields`` will force an update similarly to ``force_update``. Using ``update_fields`` will force an update similarly to ``force_update``.
.. _ref-models-field-updates-using-f-expressions:
Updating attributes based on existing fields Updating attributes based on existing fields
-------------------------------------------- --------------------------------------------
@ -356,35 +358,21 @@ achieve this is to do something like::
If the old ``number_sold`` value retrieved from the database was 10, then If the old ``number_sold`` value retrieved from the database was 10, then
the value of 11 will be written back to the database. the value of 11 will be written back to the database.
This sequence has a standard update problem in that it contains a race The process can be made robust, :ref:`avoiding a race condition
condition. If another thread of execution has already saved an updated value <avoiding-race-conditions-using-f>`, as well as slightly faster by expressing
after the current thread retrieved the old value, the current thread will only the update relative to the original field value, rather than as an explicit
save the old value plus one, rather than the new (current) value plus one. assignment of a new value. Django provides :class:`F expressions
<django.db.models.F>` for performing this kind of relative update. Using
The process can be made robust and slightly faster by expressing the update :class:`F expressions <django.db.models.F>`, the previous example is expressed
relative to the original field value, rather than as an explicit assignment of as::
a new value. Django provides :ref:`F() expressions <query-expressions>` for
performing this kind of relative update. Using ``F()`` expressions, the
previous example is expressed as::
>>> from django.db.models import F >>> from django.db.models import F
>>> product = Product.objects.get(name='Venezuelan Beaver Cheese') >>> product = Product.objects.get(name='Venezuelan Beaver Cheese')
>>> product.number_sold = F('number_sold') + 1 >>> product.number_sold = F('number_sold') + 1
>>> product.save() >>> product.save()
This approach doesn't use the initial value from the database. Instead, it For more details, see the documentation on :class:`F expressions
makes the database do the update based on whatever value is current at the time <django.db.models.F>` and their :ref:`use in update queries
that the :meth:`~Model.save()` is executed.
Once the object has been saved, you must reload the object in order to access
the actual value that was applied to the updated field::
>>> product = Products.objects.get(pk=product.pk)
>>> print(product.number_sold)
42
For more details, see the documentation on :ref:`F() expressions
<query-expressions>` and their :ref:`use in update queries
<topics-db-queries-update>`. <topics-db-queries-update>`.
Specifying which fields to save Specifying which fields to save

127
docs/ref/models/queries.txt Normal file
View File

@ -0,0 +1,127 @@
=====================
Query-related classes
=====================
.. currentmodule:: django.db.models
This document provides reference material for query-related tools not
documented elsewhere.
``F()`` expressions
===================
.. class:: F
An ``F()`` object represents the value of a model field. It makes it possible
to refer to model field values and perform database operations using them
without actually having to pull them out of the database into Python memory.
Instead, Django uses the ``F()`` object to generate a SQL expression that
describes the required operation at the database level.
This is easiest to understand though an example. Normally, one might do
something like this::
# Tintin filed a news story!
reporter = Reporters.objects.get(name='Tintin')
reporter.stories_filed += 1
reporter.save()
Here, we have pulled the value of ``reporter.stories_filed`` from the database
into memory and manipulated it using familiar Python operators, and then saved
the object back to the database. But instead we could also have done::
from django.db.models import F
reporter = Reporters.objects.get(name='Tintin')
reporter.stories_filed = F('stories_filed') + 1
reporter.save()
Although ``reporter.stories_filed = F('stories_filed') + 1`` looks like a
normal Python assignment of value to an instance attribute, in fact it's an SQL
construct describing an operation on the database.
When Django encounters an instance of ``F()``, it overrides the standard Python
operators to create an encapsulated SQL expression; in this case, one which
instructs the database to increment the database field represented by
``reporter.stories_filed``.
Whatever value is or was on ``reporter.stories_filed``, Python never gets to
know about it - it is dealt with entirely by the database. All Python does,
through Django's ``F()`` class, is create the SQL syntax to refer to the field
and describe the operation.
.. note::
In order to access the new value that has been saved in this way, the object
will need to be reloaded::
reporter = Reporters.objects.get(pk=reporter.pk)
As well as being used in operations on single instances as above, ``F()`` can
be used on ``QuerySets`` of object instances, with ``update()``. This reduces
the two queries we were using above - the ``get()`` and the
:meth:`~Model.save()` - to just one::
reporter = Reporters.objects.filter(name='Tintin')
reporter.update(stories_filed=F('stories_filed') + 1)
We can also use :meth:`~django.db.models.query.QuerySet.update()` to increment
the field value on multiple objects - which could be very much faster than
pulling them all into Python from the database, looping over them, incrementing
the field value of each one, and saving each one back to the database::
Reporter.objects.all().update(stories_filed=F('stories_filed) + 1)
``F()`` therefore can offer performance advantages by:
* getting the database, rather than Python, to do work
* reducing the number of queries some operations require
.. _avoiding-race-conditions-using-f:
Avoiding race conditions using ``F()``
--------------------------------------
Another useful benefit of ``F()`` is that having the database - rather than
Python - update a field's value avoids a *race condition*.
If two Python threads execute the code in the first example above, one thread
could retrieve, increment, and save a field's value after the other has
retrieved it from the database. The value that the second thread saves will be
based on the original value; the work of the first thread will simply be lost.
If the database is responsible for updating the field, the process is more
robust: it will only ever update the field based on the value of the field in
the database when the :meth:`~Model.save()` or ``update()`` is executed, rather
than based on its value when the instance was retrieved.
Using ``F()`` in filters
------------------------
``F()`` is also very useful in ``QuerySet`` filters, where they make it
possible to filter a set of objects against criteria based on their field
values, rather than on Python values.
This is documented in :ref:`using F() expressions in queries
<using-f-expressions-in-filters>`
Supported operations with ``F()``
---------------------------------
As well as addition, Django supports subtraction, multiplication, division,
and modulo arithmetic with ``F()`` objects, using Python constants,
variables, and even other ``F()`` objects.
``Q()`` objects
===============
.. class:: Q
A ``Q()`` object, like an :class:`~django.db.models.F` object, encapsulates a
SQL expression in a Python object that can be used in database-related
operations.
In general, ``Q() objects`` make it possible to define and reuse conditions.
This permits the :ref:`construction of complex database queries
<complex-lookups-with-q>` using ``|`` (``OR``) and ``&`` (``AND``) operators;
in particular, it is not otherwise possible to use ``OR`` in ``QuerySets``.

View File

@ -46,15 +46,15 @@ Query expressions
Queries can now refer to a another field on the query and can traverse Queries can now refer to a another field on the query and can traverse
relationships to refer to fields on related models. This is implemented in the relationships to refer to fields on related models. This is implemented in the
new :class:`F` object; for full details, including examples, consult the new :class:`F` object; for full details, including examples, consult the
:ref:`documentation for F expressions <query-expressions>`. :class:`F expressions documentation <django.db.models.F>`.
Performance improvements Performance improvements
------------------------ ------------------------
.. currentmodule:: django.test .. currentmodule:: django.test
Tests written using Django's :doc:`testing framework </topics/testing/index>` now run Tests written using Django's :doc:`testing framework </topics/testing/index>`
dramatically faster (as much as 10 times faster in many cases). now run dramatically faster (as much as 10 times faster in many cases).
This was accomplished through the introduction of transaction-based tests: when This was accomplished through the introduction of transaction-based tests: when
using :class:`django.test.TestCase`, your tests will now be run in a transaction using :class:`django.test.TestCase`, your tests will now be run in a transaction

View File

@ -211,7 +211,7 @@ Query expressions
Queries can now refer to a another field on the query and can traverse Queries can now refer to a another field on the query and can traverse
relationships to refer to fields on related models. This is implemented in the relationships to refer to fields on related models. This is implemented in the
new :class:`~django.db.models.F` object; for full details, including examples, new :class:`~django.db.models.F` object; for full details, including examples,
consult the :ref:`documentation for F expressions <query-expressions>`. consult the :class:`F expressions documentation <django.db.models.F>`.
Model improvements Model improvements
------------------ ------------------

View File

@ -324,7 +324,7 @@ requests. These include:
to :meth:`~django.shortcuts.render_to_response()` providing a to :meth:`~django.shortcuts.render_to_response()` providing a
:class:`~django.template.RequestContext` by default. :class:`~django.template.RequestContext` by default.
* Support for combining :ref:`F() expressions <query-expressions>` * Support for combining :class:`F expressions <django.db.models.F>`
with timedelta values when retrieving or updating database values. with timedelta values when retrieving or updating database values.
.. _HTTPOnly: https://www.owasp.org/index.php/HTTPOnly .. _HTTPOnly: https://www.owasp.org/index.php/HTTPOnly

View File

@ -562,11 +562,12 @@ Miscellaneous
needs. The new default value is ``0666`` (octal) and the current umask value needs. The new default value is ``0666`` (octal) and the current umask value
is first masked out. is first masked out.
* The :ref:`F() expressions <query-expressions>` supported bitwise operators by * The :class:`F expressions <django.db.models.F>` supported bitwise operators
``&`` and ``|``. These operators are now available using ``.bitand()`` and by ``&`` and ``|``. These operators are now available using ``.bitand()`` and
``.bitor()`` instead. The removal of ``&`` and ``|`` was done to be consistent with ``.bitor()`` instead. The removal of ``&`` and ``|`` was done to be
:ref:`Q() expressions <complex-lookups-with-q>` and ``QuerySet`` combining where consistent with :ref:`Q() expressions <complex-lookups-with-q>` and
the operators are used as boolean AND and OR operators. ``QuerySet`` combining where the operators are used as boolean AND and OR
operators.
* The :ttag:`csrf_token` template tag is no longer enclosed in a div. If you need * The :ttag:`csrf_token` template tag is no longer enclosed in a div. If you need
HTML validation against pre-HTML5 Strict DTDs, you should add a div around it HTML validation against pre-HTML5 Strict DTDs, you should add a div around it

View File

@ -601,13 +601,14 @@ Miscellaneous
needs. The new default value is ``0666`` (octal) and the current umask value needs. The new default value is ``0666`` (octal) and the current umask value
is first masked out. is first masked out.
* The :ref:`F() expressions <query-expressions>` supported bitwise operators by * The :class:`F expressions <django.db.models.F>` supported bitwise operators by
``&`` and ``|``. These operators are now available using ``.bitand()`` and ``&`` and ``|``. These operators are now available using ``.bitand()`` and
``.bitor()`` instead. The removal of ``&`` and ``|`` was done to be consistent with ``.bitor()`` instead. The removal of ``&`` and ``|`` was done to be
:ref:`Q() expressions <complex-lookups-with-q>` and ``QuerySet`` combining where consistent with :ref:`Q() expressions <complex-lookups-with-q>` and
the operators are used as boolean AND and OR operators. ``QuerySet`` combining where the operators are used as boolean AND and OR
operators.
* In a ``filter()`` call, when :ref:`F() expressions <query-expressions>` * In a ``filter()`` call, when :class:`F expressions <django.db.models.F>`
contained lookups spanning multi-valued relations, they didn't always reuse contained lookups spanning multi-valued relations, they didn't always reuse
the same relations as other lookups along the same chain. This was changed, the same relations as other lookups along the same chain. This was changed,
and now F() expressions will always use the same relations as other lookups and now F() expressions will always use the same relations as other lookups

View File

@ -679,13 +679,14 @@ Miscellaneous
needs. The new default value is ``0666`` (octal) and the current umask value needs. The new default value is ``0666`` (octal) and the current umask value
is first masked out. is first masked out.
* The :ref:`F() expressions <query-expressions>` supported bitwise operators by * The :class:`F expressions <django.db.models.F>` supported bitwise operators by
``&`` and ``|``. These operators are now available using ``.bitand()`` and ``&`` and ``|``. These operators are now available using ``.bitand()`` and
``.bitor()`` instead. The removal of ``&`` and ``|`` was done to be consistent with ``.bitor()`` instead. The removal of ``&`` and ``|`` was done to be
:ref:`Q() expressions <complex-lookups-with-q>` and ``QuerySet`` combining where consistent with :ref:`Q() expressions <complex-lookups-with-q>` and
the operators are used as boolean AND and OR operators. ``QuerySet`` combining where the operators are used as boolean AND and OR
operators.
* In a ``filter()`` call, when :ref:`F() expressions <query-expressions>` * In a ``filter()`` call, when :class:`F expressions <django.db.models.F>`
contained lookups spanning multi-valued relations, they didn't always reuse contained lookups spanning multi-valued relations, they didn't always reuse
the same relations as other lookups along the same chain. This was changed, the same relations as other lookups along the same chain. This was changed,
and now F() expressions will always use the same relations as other lookups and now F() expressions will always use the same relations as other lookups

View File

@ -111,7 +111,7 @@ For instance:
* At the most basic level, use :ref:`filter and exclude <queryset-api>` to do * At the most basic level, use :ref:`filter and exclude <queryset-api>` to do
filtering in the database. filtering in the database.
* Use :ref:`F() object query expressions <query-expressions>` to do filtering * Use :class:`F expressions <django.db.models.F>` to do filtering
against other fields within the same model. against other fields within the same model.
* Use :doc:`annotate to do aggregation in the database </topics/db/aggregation>`. * Use :doc:`annotate to do aggregation in the database </topics/db/aggregation>`.

View File

@ -588,18 +588,16 @@ relation). Conditions in subsequent
:meth:`~django.db.models.query.QuerySet.exclude` calls that refer to the same :meth:`~django.db.models.query.QuerySet.exclude` calls that refer to the same
relation may end up filtering on different linked objects. relation may end up filtering on different linked objects.
.. _query-expressions: .. _using-f-expressions-in-filters:
Filters can reference fields on the model Filters can reference fields on the model
----------------------------------------- -----------------------------------------
.. class:: F
In the examples given so far, we have constructed filters that compare In the examples given so far, we have constructed filters that compare
the value of a model field with a constant. But what if you want to compare the value of a model field with a constant. But what if you want to compare
the value of a model field with another field on the same model? the value of a model field with another field on the same model?
Django provides the :ref:`F() expressions <query-expressions>` to allow such Django provides :class:`F expressions <django.db.models.F>` to allow such
comparisons. Instances of ``F()`` act as a reference to a model field within a comparisons. Instances of ``F()`` act as a reference to a model field within a
query. These references can then be used in query filters to compare the values query. These references can then be used in query filters to compare the values
of two different fields on the same model instance. of two different fields on the same model instance.
@ -779,15 +777,11 @@ being evaluated and therefore populate the cache::
Complex lookups with Q objects Complex lookups with Q objects
============================== ==============================
.. class:: Q
Keyword argument queries -- in :meth:`~django.db.models.query.QuerySet.filter`, Keyword argument queries -- in :meth:`~django.db.models.query.QuerySet.filter`,
etc. -- are "AND"ed together. If you need to execute more complex queries (for etc. -- are "AND"ed together. If you need to execute more complex queries (for
example, queries with ``OR`` statements), you can use ``Q`` objects. example, queries with ``OR`` statements), you can use :class:`Q objects <django.db.models.Q>`.
.. comment: Link to Q does not work, since this documentation does not exist yet. A :class:`Q object <django.db.models.Q>` (``django.db.models.Q``) is an object
A :class:`~django.db.models.Q` object (``django.db.models.Q``) is an object
used to encapsulate a collection of keyword arguments. These keyword arguments used to encapsulate a collection of keyword arguments. These keyword arguments
are specified as in "Field lookups" above. are specified as in "Field lookups" above.
@ -1019,10 +1013,10 @@ over them and call :meth:`~django.db.models.Model.save`::
for item in my_queryset: for item in my_queryset:
item.save() item.save()
Calls to update can also use :ref:`F() objects <query-expressions>` to update Calls to update can also use :class:`F expressions <django.db.models.F>` to
one field based on the value of another field in the model. This is especially update one field based on the value of another field in the model. This is
useful for incrementing counters based upon their current value. For example, to especially useful for incrementing counters based upon their current value. For
increment the pingback count for every entry in the blog:: example, to increment the pingback count for every entry in the blog::
>>> Entry.objects.all().update(n_pingbacks=F('n_pingbacks') + 1) >>> Entry.objects.all().update(n_pingbacks=F('n_pingbacks') + 1)