UsingAnnotations.rst 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. Understanding annotations
  2. =========================
  3. What are annotations?
  4. ^^^^^^^^^^^^^^^^^^^^^
  5. Annotations are meta-data that can be embedded in source code.
  6. You may already be familiar with the `PHP-DOC`_ flavor of source code annotations, which are seen in most modern
  7. PHP codebases - they look like this:
  8. .. code-block:: php
  9. class Foo
  10. {
  11. /**
  12. * @var integer
  13. */
  14. public $bar;
  15. }
  16. When you see ``@var integer`` in the source code, immediately preceding the ``public $bar`` declaration, this little
  17. piece of meta-data tells us that the ``$bar`` property is expected to contain a value of type ``integer``.
  18. This information is useful to programmers - an IDE can use this information to display popup hints when you're
  19. writing code that works with an instance of ``Foo``, documentation generators can display this information in
  20. reference material, etc.
  21. Now imagine you had to build an HTML form that allows someone to edit a ``Foo`` object. When this information
  22. comes back from a form-post, you will need to validate that the input is in fact an integer - using the information
  23. from the ``@var`` annotation, you could abstract and automate this process, rather than having to code in by hand
  24. every time.
  25. The same information about the type of value required for this property could have many other uses besides
  26. validation - for example, you could use it to decide what type of input to render on a form, or how to
  27. persist the value to a database column.
  28. Using other types of annotations besides ``@var``, you could provide more information about the ``$bar``
  29. property - for example, you might use an annotation to specify the minimum and maximum allowed integer values of a
  30. property for validation, or define a label to be displayed next to the input on forms or in the column-header of a
  31. list of ``Foo`` objects:
  32. .. code-block:: php
  33. class Foo
  34. {
  35. /**
  36. * @var integer
  37. * @range(0, 100)
  38. * @label('Number of Bars')
  39. */
  40. public $bar;
  41. }
  42. We now have the information about the type of value, the allowed range, and the label, all associated with
  43. the ``Foo::$bar`` property member. You may have noticed a subtle difference between the ``@var`` annotation
  44. and the other two annotations in this example: the extra parentheses - we'll get to that below.
  45. It's important to understand that this meta-data does not have a single predefined purpose - it is general information,
  46. which when put to use in creative ways, can be used to simplify or eliminate repetitive work, and enables you to write
  47. more elegant and reusable code.
  48. What does this library do?
  49. ^^^^^^^^^^^^^^^^^^^^^^^^^^
  50. This library allows you to implement annotation-types as classes, and apply them as objects.
  51. Annotations are translated into objects using a simple rule: ``@name`` is essentially equivalent
  52. to ``new NameAnnotation()`` - in other words, the annotation name is capitalized and an "Annotation" suffix is added
  53. to the class-name; this prevents the class-names of annotation-types from colliding with the names of other classes.
  54. Using this library, annotations applied to classes, methods and properties may be inspected at run-time. Combined with
  55. the `Reflection`_ API, this enables you to apply `reflective meta-programming`_ in your PHP projects.
  56. Annotation Syntax
  57. ^^^^^^^^^^^^^^^^^
  58. This library provides support for two types of annotation syntax. While the difference between the two syntaxes is
  59. subtle, the difference in terms of how they function is very different.
  60. The first type of syntax is based on `PHP-DOC`_ syntax:
  61. .. code-block:: php
  62. /**
  63. * @var integer
  64. */
  65. public $bar;
  66. PHP-DOC annotations do not have a fixed syntax - that is, everything after ``@name`` is interpreted differently for
  67. each type of annotation. For example, the syntax for ``@var`` is ``@var {type} {description}``, while the syntax for
  68. ``@param`` is ``@param {type} {$name} {description}``.
  69. In other words, PHP-DOC style annotations have to go through an extra step at compile-time. (note that there is no
  70. performance penalty for this extra step, since the compiled annotation properties are cached.)
  71. For simple annotations (like those defined by the `PHP-DOC specification`_), this syntax is usually preferable.
  72. For custom annotations (perhaps requiring more complex properties), a second syntax using parentheses is supported:
  73. .. code-block:: php
  74. /**
  75. * @range(0, 100)
  76. */
  77. public $bar;
  78. When this syntax is used, the run-time equivalent for this example is something along the lines of:
  79. .. code-block:: php
  80. $annotation = new RangeAnnotation();
  81. $annotation->initAnnotation(array(0, 100));
  82. In other words, everything between the parentheses is standard PHP `array`_ syntax - as you're probably already
  83. comfortable with this syntax, there is no additional syntax to learn.
  84. While an annotation-type can optionally implement a custom (PHP-DOC style) syntax annotation, the array-style syntax
  85. is supported by every annotation. To achieve compatibility with IDEs and documentation generators, you should use the
  86. PHP-DOC style syntax for annotations defined by the PHP-DOC standard.
  87. Both syntaxes have unique advantages:
  88. * PHP-DOC style offers shorter syntax for commonly used annotation-types, and compatibility with IDEs and documentation
  89. generators.
  90. * Array-style syntax offers direct access to PHP language features, such as access to class-constants, static
  91. method-calls, nested arrays, etc.
  92. Note that both syntaxes cause annotations to initialize in the same way, at run-time - via a call to
  93. the ``IAnnotation::initAnnotation()`` interface, passing an array of property values. The PHP-DOC style syntax simply
  94. adds an extra step where the annotation values are parsed, and the initialization code for those properties is
  95. generated (and cached).
  96. So what can I do with this?
  97. ^^^^^^^^^^^^^^^^^^^^^^^^^^^
  98. See a real, working example of declarative meta-programming with this library in
  99. :doc:`this commented, step-by-step example <DemoScript>` - the same script is available in the `demo folder`_ in the
  100. project, and can be run from your local web-server.
  101. .. _PHP-DOC: http://www.phpdoc.org/
  102. .. _Reflection: http://php.net/manual/en/book.reflection.php
  103. .. _reflective meta-programming: http://en.wikipedia.org/wiki/Reflection_(computer_programming)
  104. .. _PHP-DOC specification: http://www.phpdoc.org/docs/latest/for-users/list-of-tags.html
  105. .. _array: http://php.net/manual/en/language.types.array.php
  106. .. _demo folder: https://github.com/php-annotations/php-annotations/tree/master/demo