Advertisement
Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- <html>
- <head>
- <title>Class deletions</title>
- </head>
- <body>
- <h1>Class deletions</h1>
- <table>
- <tr>
- <td>Document number:</td>
- <td><i>Nnnnn=yy–nnnn</i></td>
- </tr>
- <tr>
- <td>Date:</td>
- <td><i>2014–05–28</i></td>
- </tr>
- <tr>
- <td>Project:</td>
- <td>Programming Language C++, Evolutionary Working Group</td>
- </tr>
- <tr>
- <td>Reply-to:</td>
- <td><a href="mailto:[email protected]">Douglas Boffey
- </tr>
- </table>
- <h2>Table of Contents</h2>
- <a href="#intro">Introduction</a><p/>
- <a href="#motivation">Motivation and Scope</a><p/>
- <a href="#impact">Impact on the Standard</a><p/>
- <a href="#design">Design decisions</a><p/>
- <a href="#technical">Technical Specifications</a><p/>
- <a href="#ack">Acknowledgements</a><p/>
- <a href="#refs">References</a><p/>
- <a href="#change">Change Log</a><p/>
- <h2><a id="intro"/>Introduction</h2>
- This proposal is for the introduction of deleted classes, allowing the following class definition:<p/>
- <code>class Foo = delete;</code><p/>
- This would make any further reference to class Foo an error.<p/>
- <h2><a id="motivation"/>Motivation and Scope</h2>
- There are two problems this proposal addresses:<p/>
- <ol>
- <li><i>templates that are not substantiable</i>—sometimes, you may need to create a
- template class, but only specialisations have any meaning [EXAMPLE: consider the
- template class std::numeric_limits<T>. This only makes sense if T is an arithmetic
- type. A design decision was made to use default values if T is not an arithmetic
- type. An alternative design decision would be to leave the general template
- unsubstantiable. The best way to achieve this would be to delete the class.
- – –END EXAMPLE].<p/>
- Alternatively, a specialisation of a group of specialisations needs to be
- deleted [EXAMPLE: in Chapter 29 of <a href="#stro2013">[STRO2013]</a>, a
- Matrix class is developed that is templated on the cell type T, and the number
- of dimensions, N, of the matrix.<p/>
- Matrix<T,0> could have been defined equivalent to a scalar, but, on p. 844,
- we read:<p/>
- To avoid surprises, we define N=0 to be an error:<p/>
- <code><b>template</b><<b>typename</b> T><p/>
- <b>struct</b> Matrix_init<T,0>; <i>// undefined on purpose</i></code><p/>
- It would surely be preferable to explicitly prevent it being defined by deleting
- it– –END EXAMPLE]; and</li>
- <li><i>deprecated classes</i>—after a class has been deprecated, it may
- no longer be supported. This is best done by deleting it.</li>
- </ol>
- The notation was chosen to mirror the way functions can be deleted at the
- moment.<p/>
- One shortcoming is that there would be no method of indicating the reason for the
- deletion, except within documentation.<p/>
- <h2><a id="impact"/>Impact on the Standard</h2>
- This would allow a class definition to indicate that the class was deleted.<p/>
- <h2><a id="design"/>Design Decisions</h2>
- There are currently two methods of achieving this:<p/>
- <ol>
- <li>not creating a class definition—this would leave the programmer to
- write his/her own, this negates the purpose of not defining the class; and</li>
- <li>adding a <code>std::static_assert</code> to the body of the class—this
- has the following problems:</li>
- <ul>
- <li>having a static_assert seems to be abusing the rôle of a static_assert, and</li>
- <li>code to indicate a class is not to be substantiated should be part of the
- header, not the body of the definition.</li>
- </ul>
- </ol>
- This could alternatively be solved by adding an attribute, for example, [[poison]],
- but this would be changing the standard in a way that is contrary to the current language.<p/>
- To mirror what is done with deleted functions, and to lessen the burden on compiler writers,
- it was decided to allow deleted classes to be pointed to and referenced. Since it is
- meaningless to speak of the size of a nonexistant object, it was decided to disallow taking
- the sizeof, and of using pointer arithmetic, on deleted classes<p/>
- As there would be complications in having a deleted class participate in a class hierarchy,
- it was decided not to allow deleted classes to have base classes, nor be derived from.
- <h2><a id="technical"/>Technical specifications</h2>
- A class deletion would define a class that could not be used. It would have the following
- properties:<p/>
- <ul>
- <li>Deleted classes can be declared any number of times, but only defined once (as per the ODR).
- [EXAMPLE:
- <code>
- <b>class</b> A;<br/>
- <br/>
- <b>class</b> A = <b>delete</b>; <i>// ok</i><br/>
- <br/>
- <b>class</b> A; <i>// ok</i><br/>
- <br/>
- <b>class</b> A {}; <i>// error: class deleted</i><br/>
- <br/>
- <b>class</b> A = <b>delete</b>; <i>// error: violation of ODR</i><br/>
- </code> – –END EXAMPLE]</li>
- <li>Deleted classes can not have base classes, nor can they be derived from. [EXAMPLE:
- <code>
- <b>class</b> A {};<br/>
- <br/>
- <b>class</b> B : A = <b>delete</b>; <i>// error: deleted classes cannot have a base class</i><br/>
- <br/>
- <b>class</b> C : B; <i>// error: class B deleted</i><br/>
- </code> – –END EXAMPLE]</li>
- <li>Deleted classes cannot be substantiated but can be pointed to and referenced. [EXAMPLE:
- <code>
- <b>class</b> A;<br/>
- <br/>
- A *a;<br/>
- <br/>
- <b>class</b> A = <b>delete</b>; <i>// ok</i><br/>
- <br/>
- <b>class</b> B;<br/>
- <br/>
- <b>class</b> C {<br/>
- B *b;<br/>
- };<br/>
- <br/>
- <b>class</b> B = <b>delete</b>; <i>// ok</i><br/>
- <br/>
- <b>class</b> D = <b>delete</b>;<br/>
- <br/>
- D *d; <i>// ok</i><br/>
- D d; <i>// error: D deleted</i><br/>
- <b>void</b> fn1(D d); <i>// error: D deleted</i><br/>
- <b>void</b> fn2(D *d); <i>// ok</i><br/>
- <b>void</b> fn3(D &d); <i>// ok</i><br/>
- <b>void</b> fn4(D &&d); <i>// ok</i><br/>
- D fn5(); <i>// error: D deleted</i><br/>
- D *fn6(); <i>// ok</i><br/>
- D &fn7(); <i>// ok</i><br/>
- D &&fn8(); <i>// ok</i><br/>
- </code> – –END EXAMPLE]</li>
- <li>Deleted classes can be befriended. [EXAMPLE:
- <code>
- <b>class</b> A = <b>delete</b>;<br/>
- <br/>
- <b>class</b> B {<br/>
- <b>friend</b> <b>class</b> A; <i>// ok</i><br/>
- <b>friend</b> <b>class</b> C;<br/>
- };<br/>
- <br/>
- <b>class</b> C = <b>delete</b>; <i>// ok</i><br/>
- </code> – –END EXAMPLE]</li>
- <li>Deleted classes can be aliased. [EXAMPLE:
- <code>
- <b>class</b> A = <b>delete</b>;<br/>
- <br/>
- <b>typedef</b> A B; <i>// ok: B treated as deleted</i><br/>
- <b>using</b> C = A; <i>// ok: C treated as deleted</i><br/>
- <br/>
- B *b; <i>// error: A deleted</i><br/>
- C *c; <i>// error: A deleted</i><br/>
- </code> – –END EXAMPLE]</li>
- <li>sizeof cannot be called on a deleted class, but typeid can be. operator new is
- not allowed to return a deleted class. No pointer arithmetic is allowed on deleted
- classes. [EXAMPLE:
- <code>
- <b>class</b> A = deleted;<br/>
- <b>auto</b> s{<b>sizeof</b>(A)}; <i>// error: A deleted</i><br/>
- <b>auto</b> t{<b>typeid</b>(A)}; <i>// ok</i><br/>
- <br/>
- <b>auto</b> b = <b>new</b> A; <i>// error: A deleted</i><br/>
- <b>auto</b> c = <b>new</b> A[10]; <i>// error: A deleted</i><br/>
- <b>auto</b> d = <b>new</b> (/* whatever */) A; <i>// error: A deleted</i><br/>
- <br/>
- A *p;<br/>
- A *q;<br/>
- <br/>
- ++p; <i>// error: A deleted</i><br/>
- q = p++; <i>// error: A deleted</i><br/>
- --p; <i>// error: A deleted</i><br/>
- q = p--; <i>// error: A deleted</i><br/>
- q = p + 2; <i>// error: A deleted</i><br/>
- p += 2; <i>// error: A deleted</i><br/>
- q = p - 2; <i>// error: A deleted</i><br/>
- p -= 2; <i>// error: A deleted</i><br/>
- std::ptrdiff r = p - q; <i>// error: A deleted</i><br/>
- </code> – –END EXAMPLE]</li>
- <li>Similarly, casting to a pointer to a deleted class is an error. [EXAMPLE:
- <code>
- </code> – –END EXAMPLE]</li>
- </ul>
- <h2><a id="ack"/>Acknowledgements</h2>
- I would like to give thanks to everyone who has provided advice on the proposal website.<p/>
- <h2><a id="refs"/>References</h2>
- <a id="#stro2013"/><b>[STRO2013]</b> Stroustrup, B. <i>The C++ Programming
- Language, 4<sup>th</sup> edition</i> Addison-Wesley, 2013, p. 844<p/>
- <h2><a id="change"/>Change Log</h2>
- <ul>
- <li>2014–06–05
- <ul>
- <li>added this change log</li>
- <li>added title</li>
- <li>removed all inappropriate <code> tags</li>
- <li>removed the ‘unconditional’ from the static_assert</li>
- <li>made a first draft at the technical specifications</li>
- <li>changed the format of the references</li>
- </ul>
- </li>
- <li>2014–06–06
- <ul>corrected <p> tags</ul>
- <ul>changed rule for classes derived from a deleted class</ul>
- <ul>added rationale for not allowing pointers to deleted classes, using sizeof on them</ul>
- <ul>added rules for sizeof, typeid, operator new and *_cast</ul>
- </li>
- <li>2014–06–10
- <ul>improved example of general template with deleted specialisation</ul>
- <ul>added example of deleted general template with substantiated specialisations</ul>
- <ul>the <a href="motivation">motivation</a> was updated to reflect revised rules</ul>
- <ul>changed the examples in the <a href="technical">Technical Specification</a>
- to reflect the change in rules</ul>
- </li>
- </ul>
- </body>
- </html>
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement