reSTify PEP 285 (#366)

This commit is contained in:
Huang Huang 2017-09-13 07:27:12 +08:00 committed by Guido van Rossum
parent 27d218f1bb
commit b596c6abe3
1 changed files with 328 additions and 312 deletions

View File

@ -5,437 +5,453 @@ Last-Modified: $Date$
Author: guido@python.org (Guido van Rossum) Author: guido@python.org (Guido van Rossum)
Status: Final Status: Final
Type: Standards Track Type: Standards Track
Content-Type: text/x-rst
Created: 8-Mar-2002 Created: 8-Mar-2002
Python-Version: 2.3 Python-Version: 2.3
Post-History: 8-Mar-2002, 30-Mar-2002, 3-Apr-2002 Post-History: 8-Mar-2002, 30-Mar-2002, 3-Apr-2002
Abstract Abstract
========
This PEP proposes the introduction of a new built-in type, bool, This PEP proposes the introduction of a new built-in type, bool,
with two constants, False and True. The bool type would be a with two constants, ``False`` and ``True``. The bool type would be a
straightforward subtype (in C) of the int type, and the values straightforward subtype (in C) of the int type, and the values
False and True would behave like 0 and 1 in most respects (for ``False`` and ``True`` would behave like 0 and 1 in most respects (for
example, False==0 and True==1 would be true) except repr() and example, ``False==0`` and ``True==1`` would be true) except ``repr()`` and
str(). All built-in operations that conceptually return a Boolean ``str()``. All built-in operations that conceptually return a Boolean
result will be changed to return False or True instead of 0 or 1; result will be changed to return ``False`` or ``True`` instead of 0 or 1;
for example, comparisons, the "not" operator, and predicates like for example, comparisons, the "not" operator, and predicates like
isinstance(). ``isinstance()``.
Review Review
======
I've collected enough feedback to last me a lifetime, so I declare I've collected enough feedback to last me a lifetime, so I declare
the review period officially OVER. I had Chinese food today; my the review period officially OVER. I had Chinese food today; my
fortune cookie said "Strong and bitter words indicate a weak fortune cookie said "Strong and bitter words indicate a weak
cause." It reminded me of some of the posts against this cause." It reminded me of some of the posts against this
PEP... :-) PEP... :-)
Anyway, here are my BDFL pronouncements. (Executive summary: I'm Anyway, here are my BDFL pronouncements. (Executive summary: I'm
not changing a thing; all variants are rejected.) not changing a thing; all variants are rejected.)
1) Should this PEP be accepted? 1) Should this PEP be accepted?
=> Yes. => Yes.
There have been many arguments against the PEP. Many of them There have been many arguments against the PEP. Many of them
were based on misunderstandings. I've tried to clarify some of were based on misunderstandings. I've tried to clarify some of
the most common misunderstandings below in the main text of the the most common misunderstandings below in the main text of the
PEP. The only issue that weighs at all for me is the tendency PEP. The only issue that weighs at all for me is the tendency
of newbies to write "if x == True" where "if x" would suffice. of newbies to write "if x == True" where "if x" would suffice.
More about that below too. I think this is not a sufficient More about that below too. I think this is not a sufficient
reason to reject the PEP. reason to reject the PEP.
2) Should str(True) return "True" or "1"? "1" might reduce 2) Should ``str(True)`` return "True" or "1"? "1" might reduce
backwards compatibility problems, but looks strange. backwards compatibility problems, but looks strange.
(repr(True) would always return "True".) (``repr(True)`` would always return "True".)
=> "True". => "True".
Almost all reviewers agree with this. Almost all reviewers agree with this.
3) Should the constants be called 'True' and 'False' (similar to 3) Should the constants be called 'True' and 'False' (similar to
None) or 'true' and 'false' (as in C++, Java and C99)? None) or 'true' and 'false' (as in C++, Java and C99)?
=> True and False. => True and False.
Most reviewers agree that consistency within Python is more Most reviewers agree that consistency within Python is more
important than consistency with other languages. important than consistency with other languages.
4) Should we strive to eliminate non-Boolean operations on bools 4) Should we strive to eliminate non-Boolean operations on bools
in the future, through suitable warnings, so that for example in the future, through suitable warnings, so that for example
True+1 would eventually (in Python 3000) be illegal? True+1 would eventually (in Python 3000) be illegal?
=> No. => No.
There's a small but vocal minority that would prefer to see There's a small but vocal minority that would prefer to see
"textbook" bools that don't support arithmetic operations at "textbook" bools that don't support arithmetic operations at
all, but most reviewers agree with me that bools should always all, but most reviewers agree with me that bools should always
allow arithmetic operations. allow arithmetic operations.
5) Should operator.truth(x) return an int or a bool? 5) Should ``operator.truth(x)`` return an int or a bool?
=> bool. => bool.
Tim Peters believes it should return an int, but almost all Tim Peters believes it should return an int, but almost all
other reviewers agree that it should return a bool. My other reviewers agree that it should return a bool. My
rationale: operator.truth() exists to force a Boolean context rationale: ``operator.truth()`` exists to force a Boolean context
on its argument (it calls the C API PyObject_IsTrue()). on its argument (it calls the C API ``PyObject_IsTrue())``.
Whether the outcome is reported as int or bool is secondary; if Whether the outcome is reported as int or bool is secondary; if
bool exists there's no reason not to use it. (Under the PEP, bool exists there's no reason not to use it. (Under the PEP,
operator.truth() now becomes an alias for bool(); that's fine.) ``operator.truth()`` now becomes an alias for ``bool()``; that's fine.)
6) Should bool inherit from int? 6) Should bool inherit from int?
=> Yes. => Yes.
In an ideal world, bool might be better implemented as a In an ideal world, bool might be better implemented as a
separate integer type that knows how to perform mixed-mode separate integer type that knows how to perform mixed-mode
arithmetic. However, inheriting bool from int eases the arithmetic. However, inheriting bool from int eases the
implementation enormously (in part since all C code that calls implementation enormously (in part since all C code that calls
PyInt_Check() will continue to work -- this returns true for ``PyInt_Check()`` will continue to work -- this returns true for
subclasses of int). Also, I believe this is right in terms of subclasses of int). Also, I believe this is right in terms of
substitutability: code that requires an int can be fed a bool substitutability: code that requires an int can be fed a bool
and it will behave the same as 0 or 1. Code that requires a and it will behave the same as 0 or 1. Code that requires a
bool may not work when it is given an int; for example, 3 & 4 bool may not work when it is given an int; for example, 3 & 4
is 0, but both 3 and 4 are true when considered as truth is 0, but both 3 and 4 are true when considered as truth
values. values.
7) Should the name 'bool' be changed? 7) Should the name 'bool' be changed?
=> No. => No.
Some reviewers have argued for boolean instead of bool, because Some reviewers have argued for boolean instead of bool, because
this would be easier to understand (novices may have heard of this would be easier to understand (novices may have heard of
Boolean algebra but may not make the connection with bool) or Boolean algebra but may not make the connection with bool) or
because they hate abbreviations. My take: Python uses because they hate abbreviations. My take: Python uses
abbreviations judiciously (like 'def', 'int', 'dict') and I abbreviations judiciously (like 'def', 'int', 'dict') and I
don't think these are a burden to understanding. To a newbie, don't think these are a burden to understanding. To a newbie,
it doesn't matter whether it's called a waffle or a bool; it's it doesn't matter whether it's called a waffle or a bool; it's
a new word, and they learn quickly what it means. a new word, and they learn quickly what it means.
One reviewer has argued to make the name 'truth'. I find this One reviewer has argued to make the name 'truth'. I find this
an unattractive name, and would actually prefer to reserve this an unattractive name, and would actually prefer to reserve this
term (in documentation) for the more abstract concept of truth term (in documentation) for the more abstract concept of truth
values that already exists in Python. For example: "when a values that already exists in Python. For example: "when a
container is interpreted as a truth value, an empty container container is interpreted as a truth value, an empty container
is considered false and a non-empty one is considered true." is considered false and a non-empty one is considered true."
8) Should we strive to require that Boolean operations (like "if", 8) Should we strive to require that Boolean operations (like "if",
"and", "not") have a bool as an argument in the future, so that "and", "not") have a bool as an argument in the future, so that
for example "if []:" would become illegal and would have to be for example "if []:" would become illegal and would have to be
written as "if bool([]):" ??? written as "if bool([]):" ???
=> No!!! => No!!!
Some people believe that this is how a language with a textbook Some people believe that this is how a language with a textbook
Boolean type should behave. Because it was brought up, others Boolean type should behave. Because it was brought up, others
have worried that I might agree with this position. Let me have worried that I might agree with this position. Let me
make my position on this quite clear. This is not part of the make my position on this quite clear. This is not part of the
PEP's motivation and I don't intend to make this change. (See PEP's motivation and I don't intend to make this change. (See
also the section "Clarification" below.) also the section "Clarification" below.)
Rationale Rationale
=========
Most languages eventually grow a Boolean type; even C99 (the new Most languages eventually grow a Boolean type; even C99 (the new
and improved C standard, not yet widely adopted) has one. and improved C standard, not yet widely adopted) has one.
Many programmers apparently feel the need for a Boolean type; most Many programmers apparently feel the need for a Boolean type; most
Python documentation contains a bit of an apology for the absence Python documentation contains a bit of an apology for the absence
of a Boolean type. I've seen lots of modules that defined of a Boolean type. I've seen lots of modules that defined
constants "False=0" and "True=1" (or similar) at the top and used constants "False=0" and "True=1" (or similar) at the top and used
those. The problem with this is that everybody does it those. The problem with this is that everybody does it
differently. For example, should you use "FALSE", "false", differently. For example, should you use "FALSE", "false",
"False", "F" or even "f"? And should false be the value zero or "False", "F" or even "f"? And should false be the value zero or
None, or perhaps a truth value of a different type that will print None, or perhaps a truth value of a different type that will print
as "true" or "false"? Adding a standard bool type to the language as "true" or "false"? Adding a standard bool type to the language
resolves those issues. resolves those issues.
Some external libraries (like databases and RPC packages) need to Some external libraries (like databases and RPC packages) need to
be able to distinguish between Boolean and integral values, and be able to distinguish between Boolean and integral values, and
while it's usually possible to craft a solution, it would be while it's usually possible to craft a solution, it would be
easier if the language offered a standard Boolean type. This also easier if the language offered a standard Boolean type. This also
applies to Jython: some Java classes have separately overloaded applies to Jython: some Java classes have separately overloaded
methods or constructors for int and boolean arguments. The bool methods or constructors for int and boolean arguments. The bool
type can be used to select the boolean variant. (The same is type can be used to select the boolean variant. (The same is
apparently the case for some COM interfaces.) apparently the case for some COM interfaces.)
The standard bool type can also serve as a way to force a value to The standard bool type can also serve as a way to force a value to
be interpreted as a Boolean, which can be used to normalize be interpreted as a Boolean, which can be used to normalize
Boolean values. When a Boolean value needs to be normalized to Boolean values. When a Boolean value needs to be normalized to
one of two values, bool(x) is much clearer than "not not x" and one of two values, ``bool(x)`` is much clearer than "not not x" and
much more concise than much more concise than
if x: ::
return 1
else:
return 0
Here are some arguments derived from teaching Python. When if x:
showing people comparison operators etc. in the interactive shell, return 1
I think this is a bit ugly: else:
return 0
>>> a = 13 Here are some arguments derived from teaching Python. When
>>> b = 12 showing people comparison operators etc. in the interactive shell,
>>> a > b I think this is a bit ugly::
1
>>>
If this was: >>> a = 13
>>> b = 12
>>> a > b
1
>>>
>>> a > b If this was::
True
>>>
it would require a millisecond less thinking each time a 0 or 1 >>> a > b
was printed. True
>>>
There's also the issue (which I've seen baffling even experienced it would require a millisecond less thinking each time a 0 or 1
Pythonistas who had been away from the language for a while) that was printed.
if you see:
>>> cmp(a, b) There's also the issue (which I've seen baffling even experienced
1 Pythonistas who had been away from the language for a while) that
>>> cmp(a, a) if you see::
0
>>>
you might be tempted to believe that cmp() also returned a truth >>> cmp(a, b)
value, whereas in reality it can return three different values 1
(-1, 0, 1). If ints were not (normally) used to represent >>> cmp(a, a)
Booleans results, this would stand out much more clearly as 0
something completely different. >>>
you might be tempted to believe that ``cmp()`` also returned a truth
value, whereas in reality it can return three different values
``(-1, 0, 1)``. If ints were not (normally) used to represent
Booleans results, this would stand out much more clearly as
something completely different.
Specification Specification
=============
The following Python code specifies most of the properties of the The following Python code specifies most of the properties of the
new type: new type::
class bool(int): class bool(int):
def __new__(cls, val=0): def __new__(cls, val=0):
# This constructor always returns an existing instance # This constructor always returns an existing instance
if val: if val:
return True return True
else: else:
return False return False
def __repr__(self): def __repr__(self):
if self: if self:
return "True" return "True"
else: else:
return "False" return "False"
__str__ = __repr__ __str__ = __repr__
def __and__(self, other): def __and__(self, other):
if isinstance(other, bool): if isinstance(other, bool):
return bool(int(self) & int(other)) return bool(int(self) & int(other))
else: else:
return int.__and__(self, other) return int.__and__(self, other)
__rand__ = __and__ __rand__ = __and__
def __or__(self, other): def __or__(self, other):
if isinstance(other, bool): if isinstance(other, bool):
return bool(int(self) | int(other)) return bool(int(self) | int(other))
else: else:
return int.__or__(self, other) return int.__or__(self, other)
__ror__ = __or__ __ror__ = __or__
def __xor__(self, other): def __xor__(self, other):
if isinstance(other, bool): if isinstance(other, bool):
return bool(int(self) ^ int(other)) return bool(int(self) ^ int(other))
else: else:
return int.__xor__(self, other) return int.__xor__(self, other)
__rxor__ = __xor__ __rxor__ = __xor__
# Bootstrap truth values through sheer willpower # Bootstrap truth values through sheer willpower
False = int.__new__(bool, 0) False = int.__new__(bool, 0)
True = int.__new__(bool, 1) True = int.__new__(bool, 1)
The values False and True will be singletons, like None. Because The values ``False`` and ``True`` will be singletons, like None. Because
the type has two values, perhaps these should be called the type has two values, perhaps these should be called
"doubletons"? The real implementation will not allow other "doubletons"? The real implementation will not allow other
instances of bool to be created. instances of bool to be created.
True and False will properly round-trip through pickling and ``True`` and ``False`` will properly round-trip through pickling and
marshalling; for example pickle.loads(pickle.dumps(True)) will marshalling; for example ``pickle.loads(pickle.dumps(True))`` will
return True, and so will marshal.loads(marshal.dumps(True)). return ``True``, and so will ``marshal.loads(marshal.dumps(True))``.
All built-in operations that are defined to return a Boolean All built-in operations that are defined to return a Boolean
result will be changed to return False or True instead of 0 or 1. result will be changed to return ``False`` or ``True`` instead of 0 or 1.
In particular, this affects comparisons (<, <=, ==, !=, >, >=, is, In particular, this affects comparisons (``<``, ``<=``, ``==``, ``!=``,
is not, in, not in), the unary operator 'not', the built-in ``>``, ``>=``, is, is not, in, not in), the unary operator 'not', the built-in
functions callable(), hasattr(), isinstance() and issubclass(), functions ``callable()``, ``hasattr()``, ``isinstance()`` and ``issubclass()``,
the dict method has_key(), the string and unicode methods the dict method ``has_key()``, the string and unicode methods
endswith(), isalnum(), isalpha(), isdigit(), islower(), isspace(), ``endswith()``, ``isalnum()``, ``isalpha()``, ``isdigit()``, ``islower()``, ``isspace()``,
istitle(), isupper(), and startswith(), the unicode methods ``istitle()``, ``isupper()``, and ``startswith()``, the unicode methods
isdecimal() and isnumeric(), and the 'closed' attribute of file ``isdecimal()`` and ``isnumeric()``, and the 'closed' attribute of file
objects. The predicates in the operator module are also changed objects. The predicates in the operator module are also changed
to return a bool, including operator.truth(). to return a bool, including ``operator.truth()``.
Because bool inherits from int, True+1 is valid and equals 2, and Because bool inherits from int, True+1 is valid and equals 2, and
so on. This is important for backwards compatibility: because so on. This is important for backwards compatibility: because
comparisons and so on currently return integer values, there's no comparisons and so on currently return integer values, there's no
way of telling what uses existing applications make of these way of telling what uses existing applications make of these
values. values.
It is expected that over time, the standard library will be It is expected that over time, the standard library will be
updated to use False and True when appropriate (but not to require updated to use ``False`` and ``True`` when appropriate (but not to require
a bool argument type where previous an int was allowed). This a bool argument type where previous an int was allowed). This
change should not pose additional problems and is not specified in change should not pose additional problems and is not specified in
detail by this PEP. detail by this PEP.
C API C API
=====
The header file "boolobject.h" defines the C API for the bool The header file "boolobject.h" defines the C API for the bool
type. It is included by "Python.h" so there is no need to include type. It is included by "Python.h" so there is no need to include
it directly. it directly.
The existing names Py_False and Py_True reference the unique bool The existing names ``Py_False`` and ``Py_True`` reference the unique bool
objects False and True (previously these referenced static int objects ``False`` and ``True`` (previously these referenced static int
objects with values 0 and 1, which were not unique amongst int objects with values 0 and 1, which were not unique amongst int
values). values).
A new API, PyObject *PyBool_FromLong(long), takes a C long int A new API, ``PyObject *PyBool_FromLong(long)``, takes a C long int
argument and returns a new reference to either Py_False (when the argument and returns a new reference to either ``Py_False`` (when the
argument is zero) or Py_True (when it is nonzero). argument is zero) or ``Py_True`` (when it is nonzero).
To check whether an object is a bool, the macro PyBool_Check() can To check whether an object is a bool, the macro ``PyBool_Check()`` can
be used. be used.
The type of bool instances is PyBoolObject *. The type of bool instances is ``PyBoolObject *``.
The bool type object is available as PyBool_Type. The bool type object is available as PyBool_Type.
Clarification Clarification
=============
This PEP does *not* change the fact that almost all object types This PEP does **not** change the fact that almost all object types
can be used as truth values. For example, when used in an if can be used as truth values. For example, when used in an if
statement, an empty list is false and a non-empty one is true; statement, an empty list is false and a non-empty one is true;
this does not change and there is no plan to ever change this. this does not change and there is no plan to ever change this.
The only thing that changes is the preferred values to represent The only thing that changes is the preferred values to represent
truth values when returned or assigned explicitly. Previously, truth values when returned or assigned explicitly. Previously,
these preferred truth values were 0 and 1; the PEP changes the these preferred truth values were 0 and 1; the PEP changes the
preferred values to False and True, and changes built-in preferred values to ``False`` and ``True``, and changes built-in
operations to return these preferred values. operations to return these preferred values.
Compatibility Compatibility
=============
Because of backwards compatibility, the bool type lacks many Because of backwards compatibility, the bool type lacks many
properties that some would like to see. For example, arithmetic properties that some would like to see. For example, arithmetic
operations with one or two bool arguments is allowed, treating operations with one or two bool arguments is allowed, treating
False as 0 and True as 1. Also, a bool may be used as a sequence ``False`` as 0 and ``True`` as 1. Also, a bool may be used as a sequence
index. index.
I don't see this as a problem, and I don't want evolve the I don't see this as a problem, and I don't want evolve the
language in this direction either. I don't believe that a language in this direction either. I don't believe that a
stricter interpretation of "Booleanness" makes the language any stricter interpretation of "Booleanness" makes the language any
clearer. clearer.
Another consequence of the compatibility requirement is that the Another consequence of the compatibility requirement is that the
expression "True and 6" has the value 6, and similarly the expression "True and 6" has the value 6, and similarly the
expression "False or None" has the value None. The "and" and "or" expression "False or None" has the value None. The "and" and "or"
operators are usefully defined to return the first argument that operators are usefully defined to return the first argument that
determines the outcome, and this won't change; in particular, they determines the outcome, and this won't change; in particular, they
don't force the outcome to be a bool. Of course, if both don't force the outcome to be a bool. Of course, if both
arguments are bools, the outcome is always a bool. It can also arguments are bools, the outcome is always a bool. It can also
easily be coerced into being a bool by writing for example "bool(x easily be coerced into being a bool by writing for example "bool(x
and y)". and y)".
Resolved Issues Resolved Issues
===============
(See also the Review section above.) (See also the Review section above.)
- Because the repr() or str() of a bool value is different from an - Because the ``repr()`` or ``str()`` of a bool value is different from an
int value, some code (for example doctest-based unit tests, and int value, some code (for example doctest-based unit tests, and
possibly database code that relies on things like "%s" % truth) possibly database code that relies on things like "%s" % truth)
may fail. It is easy to work around this (without explicitly may fail. It is easy to work around this (without explicitly
referencing the bool type), and it is expected that this only referencing the bool type), and it is expected that this only
affects a very small amount of code that can easily be fixed. affects a very small amount of code that can easily be fixed.
- Other languages (C99, C++, Java) name the constants "false" and - Other languages (C99, C++, Java) name the constants "false" and
"true", in all lowercase. For Python, I prefer to stick with "true", in all lowercase. For Python, I prefer to stick with
the example set by the existing built-in constants, which all the example set by the existing built-in constants, which all
use CapitalizedWords: None, Ellipsis, NotImplemented (as well as use CapitalizedWords: ``None``, ``Ellipsis``, ``NotImplemented`` (as well as
all built-in exceptions). Python's built-in namespace uses all all built-in exceptions). Python's built-in namespace uses all
lowercase for functions and types only. lowercase for functions and types only.
- It has been suggested that, in order to satisfy user - It has been suggested that, in order to satisfy user
expectations, for every x that is considered true in a Boolean expectations, for every x that is considered true in a Boolean
context, the expression x == True should be true, and likewise context, the expression ``x == True`` should be true, and likewise
if x is considered false, x == False should be true. In if x is considered false, ``x == False`` should be true. In
particular newbies who have only just learned about Boolean particular newbies who have only just learned about Boolean
variables are likely to write variables are likely to write
::
if x == True: ... if x == True: ...
instead of the correct form, instead of the correct form,
::
if x: ... if x: ...
There seem to be strong psychological and linguistic reasons why There seem to be strong psychological and linguistic reasons why
many people are at first uncomfortable with the latter form, but many people are at first uncomfortable with the latter form, but
I believe that the solution should be in education rather than I believe that the solution should be in education rather than
in crippling the language. After all, == is general seen as a in crippling the language. After all, == is general seen as a
transitive operator, meaning that from a==b and b==c we can transitive operator, meaning that from ``a==b`` and ``b==c`` we can
deduce a==c. But if any comparison to True were to report deduce ``a==c``. But if any comparison to ``True`` were to report
equality when the other operand was a true value of any type, equality when the other operand was a true value of any type,
atrocities like 6==True==7 would hold true, from which one could atrocities like ``6==True==7`` would hold true, from which one could
infer the falsehood 6==7. That's unacceptable. (In addition, infer the falsehood ``6==7``. That's unacceptable. (In addition,
it would break backwards compatibility. But even if it didn't, it would break backwards compatibility. But even if it didn't,
I'd still be against this, for the stated reasons.) I'd still be against this, for the stated reasons.)
Newbies should also be reminded that there's never a reason to Newbies should also be reminded that there's never a reason to
write write
::
if bool(x): ... if bool(x): ...
since the bool is implicit in the "if". Explicit is *not* since the bool is implicit in the "if". Explicit is **not**
better than implicit here, since the added verbiage impairs better than implicit here, since the added verbiage impairs
redability and there's no other interpretation possible. There redability and there's no other interpretation possible. There
is, however, sometimes a reason to write is, however, sometimes a reason to write
::
b = bool(x) b = bool(x)
This is useful when it is unattractive to keep a reference to an This is useful when it is unattractive to keep a reference to an
arbitrary object x, or when normalization is required for some arbitrary object x, or when normalization is required for some
other reason. It is also sometimes appropriate to write other reason. It is also sometimes appropriate to write
::
i = int(bool(x)) i = int(bool(x))
which converts the bool to an int with the value 0 or 1. This which converts the bool to an int with the value 0 or 1. This
conveys the intention to henceforth use the value as an int. conveys the intention to henceforth use the value as an int.
Implementation Implementation
==============
A complete implementation in C has been uploaded to the A complete implementation in C has been uploaded to the
SourceForge patch manager: SourceForge patch manager: http://python.org/sf/528022
http://python.org/sf/528022 This will soon be checked into CVS for python 2.3a0.
This will soon be checked into CVS for python 2.3a0.
Copyright Copyright
=========
This document has been placed in the public domain. This document has been placed in the public domain.
..
Local Variables: Local Variables:
mode: indented-text mode: indented-text
indent-tabs-mode: nil indent-tabs-mode: nil
fill-column: 70 fill-column: 70
End: End: