style fixes

This commit is contained in:
Thibault Kruse 2016-08-11 10:54:08 +02:00
parent db98d5f8b3
commit 6e1599f6f9

View File

@ -592,7 +592,7 @@ This example is easily simplified
##### Example ##### Example
void read(int* p, int n); // read max n integers into *p void read(int* p, int n); // read max n integers into *p
int a[100]; int a[100];
read(a, 1000); // bad read(a, 1000); // bad
@ -802,7 +802,6 @@ Excess checking can be costly.
There are cases where checking early is dumb because you may not ever need the value, or may only need part of the value that is more easily checked than the whole. Similarly, don't add validity checks that change the asymptotic behavior of your interface (e.g., don't add a `O(n)` check to an interface with an average complexity of `O(1)`). There are cases where checking early is dumb because you may not ever need the value, or may only need part of the value that is more easily checked than the whole. Similarly, don't add validity checks that change the asymptotic behavior of your interface (e.g., don't add a `O(n)` check to an interface with an average complexity of `O(1)`).
class Jet { // Physics says: e * e < x * x + y * y + z * z class Jet { // Physics says: e * e < x * x + y * y + z * z
float x; float x;
float y; float y;
float z; float z;
@ -823,7 +822,7 @@ There are cases where checking early is dumb because you may not ever need the v
??? ???
}; };
The physical law for a jet (`e*e < x*x + y*y + z*z`) is not an invariant because of the possibility for measurement errors. The physical law for a jet (`e * e < x * x + y * y + z * z`) is not an invariant because of the possibility for measurement errors.
??? ???
@ -983,17 +982,17 @@ Messy, low-level code breads more such code.
##### Example ##### Example
int sz = 100; int sz = 100;
int* p = (int*) malloc(sizeof(int)*sz); int* p = (int*) malloc(sizeof(int) * sz);
// ... // ...
if (count==sz) if (count == sz)
p = (int*) realloc(p,sizeof(int)*sz*2); p = (int*) realloc(p, sizeof(int) * sz * 2);
// ... // ...
This is low-level, verbose, and error-prone. This is low-level, verbose, and error-prone.
Instead, we could use `vector`: Instead, we could use `vector`:
vector<int> v(100); vector<int> v(100);
v.push_back(yet_another)int); v.push_back(yet_another)int);
##### Note ##### Note
@ -1554,7 +1553,7 @@ A facility [structured bindings](http://www.open-std.org/jtc1/sc22/wg21/docs/pap
// ... handle the error or exit ... // ... handle the error or exit ...
} }
// ... use val ... // ... use val ...
##### Note ##### Note
@ -1791,7 +1790,7 @@ To really reduce the number of arguments, we need to bundle the arguments into h
Grouping arguments into "bundles" is a general technique to reduce the number of arguments and to increase the opportunities for checking. Grouping arguments into "bundles" is a general technique to reduce the number of arguments and to increase the opportunities for checking.
Alternatively, we could use concepts (as defined by the ISO TS) to define the notion of three types that must be usable for merging: Alternatively, we could use concepts (as defined by the ISO TS) to define the notion of three types that must be usable for merging:
Mergeable{In1 In2, Out} Mergeable{In1 In2, Out}
OutputIterator merge(In1 r1, In2 r2, Out result); OutputIterator merge(In1 r1, In2 r2, Out result);
@ -2101,7 +2100,6 @@ Consider:
// simpleFunc: takes a value and calculates the expected ASIC output, // simpleFunc: takes a value and calculates the expected ASIC output,
// given the two mode flags. // given the two mode flags.
{ {
double intermediate; double intermediate;
if (flag1 > 0) { if (flag1 > 0) {
intermediate = func1(val); intermediate = func1(val);
@ -2118,9 +2116,10 @@ Consider:
intermediate = func2(intermediate); intermediate = func2(intermediate);
} }
switch (flag2 / 10) { switch (flag2 / 10) {
case 1: if (flag1 == -1) return finalize(intermediate, 1.171); break; case 1: if (flag1 == -1) return finalize(intermediate, 1.171);
break;
case 2: return finalize(intermediate, 13.1); case 2: return finalize(intermediate, 13.1);
default: ; default: break;
} }
return finalize(intermediate, 0.); return finalize(intermediate, 0.);
} }
@ -2247,7 +2246,7 @@ Specifying `inline` encourages the compiler to do a better job.
##### Example ##### Example
inline string cat(const string& s, const string& s2) { return s+s2; } inline string cat(const string& s, const string& s2) { return s + s2; }
**Exception**: Do not put an `inline` function in what is meant to be a stable interface unless you are really sure that it will not change. **Exception**: Do not put an `inline` function in what is meant to be a stable interface unless you are really sure that it will not change.
An inline function is part of the ABI. An inline function is part of the ABI.
@ -2344,13 +2343,13 @@ Passing a shared smart pointer (e.g., `std::shared_ptr`) implies a run-time cost
void g(unique_ptr<int>); void g(unique_ptr<int>);
// can only accept ints for which you are willing to share ownership // can only accept ints for which you are willing to share ownership
void g(shared_ptr<int>); void g(shared_ptr<int>);
// doesn't change ownership, but requires a particular ownership of the caller // doesn't change ownership, but requires a particular ownership of the caller
void h(const unique_ptr<int>&); void h(const unique_ptr<int>&);
// accepts any int // accepts any int
void h(int&); void h(int&);
##### Example, bad ##### Example, bad
@ -2697,7 +2696,7 @@ Explicitly passing an in-out parameter back out again as a return value is often
For example: For example:
istream& operator>>(istream& is, string& s); // much like std::operator>>() istream& operator>>(istream& is, string& s); // much like std::operator>>()
for (string s; cin >> s; ) { for (string s; cin >> s; ) {
// do something with line // do something with line
} }
@ -2733,7 +2732,7 @@ However, we prefer to be explicit, rather than subtle.
In many cases it may be useful to return a specific, user-defined "Value or error" type. In many cases it may be useful to return a specific, user-defined "Value or error" type.
For example: For example:
struct struct
The overly-generic `pair` and `tuple` should be used only when the value returned represents to independent entities rather than an abstraction. The overly-generic `pair` and `tuple` should be used only when the value returned represents to independent entities rather than an abstraction.
@ -2769,21 +2768,21 @@ It complicates checking and tool support.
void use(int* p, int n, char* s, int* q) void use(int* p, int n, char* s, int* q)
{ {
p[n-1] = 666; // Bad: we don't know if p points to n elements; p[n - 1] = 666; // Bad: we don't know if p points to n elements;
// assume it does not or use span<int> // assume it does not or use span<int>
cout << s; // Bad: we don't know if that s points to a zero-terminated array of char; cout << s; // Bad: we don't know if that s points to a zero-terminated array of char;
// assume it does not or use zstring // assume it does not or use zstring
delete q; // Bad: we don't know if *q is allocated on the free store; delete q; // Bad: we don't know if *q is allocated on the free store;
// assume it does not or use owner // assume it does not or use owner
} }
better better
void use2(span<int> p, zstring s, owner<int*> q) void use2(span<int> p, zstring s, owner<int*> q)
{ {
p[p.size()-1] = 666; // OK, a range error can be caught p[p.size() - 1] = 666; // OK, a range error can be caught
cout << s; // OK cout << s; // OK
delete q; // OK delete q; // OK
} }
##### Note ##### Note
@ -3257,7 +3256,7 @@ principle of "do as the ints do."
##### Note ##### Note
Historically there was some guidance to make the assignment operator return `const T&`. Historically there was some guidance to make the assignment operator return `const T&`.
This was primarily to avoid code of the form `(a=b)=c` -- such code is not common enough to warrant violating consistency with standard types. This was primarily to avoid code of the form `(a = b) = c` -- such code is not common enough to warrant violating consistency with standard types.
##### Example ##### Example
@ -3338,7 +3337,7 @@ There is not a choice when a set of functions are used to do a semantically equi
##### See also ##### See also
[Default arguments for virtual functions](#Rf-virtual-default-arg} [Default arguments for virtual functions](#Rf-virtual-default-arg}
##### Enforcement ##### Enforcement
@ -3709,7 +3708,7 @@ Flag protected data.
One ideal for a class is to be a regular type. One ideal for a class is to be a regular type.
That means roughly "behaves like an `int`." A concrete type is the simplest kind of class. That means roughly "behaves like an `int`." A concrete type is the simplest kind of class.
A value of regular type can be copied and the result of a copy is an independent object with the same value as the original. A value of regular type can be copied and the result of a copy is an independent object with the same value as the original.
If a concrete type has both `=` and `==`, `a=b` should result in `a == b` being `true`. If a concrete type has both `=` and `==`, `a = b` should result in `a == b` being `true`.
Concrete classes without assignment and equality can be defined, but they are (and should be) rare. Concrete classes without assignment and equality can be defined, but they are (and should be) rare.
The C++ built-in types are regular, and so are standard-library classes, such as `string`, `vector`, and `map`. The C++ built-in types are regular, and so are standard-library classes, such as `string`, `vector`, and `map`.
Concrete types are also often referred to as value types to distinguish them from types uses as part of a hierarchy. Concrete types are also often referred to as value types to distinguish them from types uses as part of a hierarchy.
@ -3793,7 +3792,7 @@ Regular types are easier to understand and reason about than types that are not
b2.name = "the other bundle"; b2.name = "the other bundle";
if (b1 == b2) error("No!"); if (b1 == b2) error("No!");
In particular, if a concrete type has an assignment also give it an equals operator so that `a=b` implies `a == b`. In particular, if a concrete type has an assignment also give it an equals operator so that `a = b` implies `a == b`.
##### Enforcement ##### Enforcement
@ -4247,7 +4246,7 @@ Independently of whether `Handle` owns its `Shape`, we must consider the default
Handle y {*new Triangle{p1, p2, p3}}; Handle y {*new Triangle{p1, p2, p3}};
x = y; // the default assignment will try *x.s = *y.s x = y; // the default assignment will try *x.s = *y.s
That `x=y` is highly suspect. That `x = y` is highly suspect.
Assigning a `Triangle` to a `Circle`? Assigning a `Triangle` to a `Circle`?
Unless `Shape` has its [copy assignment `=deleted`](#Rc-copy-virtual), only the `Shape` part of `Triangle` is copied into the `Circle`. Unless `Shape` has its [copy assignment `=deleted`](#Rc-copy-virtual), only the `Shape` part of `Triangle` is copied into the `Circle`.
@ -5092,7 +5091,7 @@ See [copy constructor vs. `clone()`](#Rc-copy-virtual).
##### Reason ##### Reason
That is the generally assumed semantics. After `x=y`, we should have `x == y`. That is the generally assumed semantics. After `x = y`, we should have `x == y`.
After a copy `x` and `y` can be independent objects (value semantics, the way non-pointer built-in types and the standard-library types work) or refer to a shared object (pointer semantics, the way pointers work). After a copy `x` and `y` can be independent objects (value semantics, the way non-pointer built-in types and the standard-library types work) or refer to a shared object (pointer semantics, the way pointers work).
##### Example ##### Example
@ -5163,7 +5162,7 @@ Prefer copy semantics unless you are building a "smart pointer". Value semantics
##### Reason ##### Reason
If `x=x` changes the value of `x`, people will be surprised and bad errors will occur (often including leaks). If `x = x` changes the value of `x`, people will be surprised and bad errors will occur (often including leaks).
##### Example ##### Example
@ -5246,7 +5245,7 @@ Equivalent to what is done for [copy-assignment](#Rc-copy-assignment).
##### Reason ##### Reason
That is the generally assumed semantics. That is the generally assumed semantics.
After `y=std::move(x)` the value of `y` should be the value `x` had and `x` should be in a valid state. After `y = std::move(x)` the value of `y` should be the value `x` had and `x` should be in a valid state.
##### Example ##### Example
@ -5324,7 +5323,7 @@ The one-in-a-million argument against `if (this == &a) return *this;` tests from
##### Note ##### Note
There is no know general way of avoiding a `if (this == &a) return *this;` test for a move assignment and still get a correct answer (i.e., after `x=x` the value of `x` is unchanged). There is no know general way of avoiding a `if (this == &a) return *this;` test for a move assignment and still get a correct answer (i.e., after `x = x` the value of `x` is unchanged).
##### Note ##### Note
@ -5668,14 +5667,18 @@ Asymmetric treatment of operands is surprising and a source of errors where conv
int number; int number;
}; };
bool operator==(const X& a, const X& b) noexcept { return a.name == b.name && a.number == b.number; } bool operator==(const X& a, const X& b) noexcept {
return a.name == b.name && a.number == b.number;
}
##### Example, bad ##### Example, bad
class B { class B {
string name; string name;
int number; int number;
bool operator==(const B& a) const { return name == a.name && number == a.number; } bool operator==(const B& a) const {
return name == a.name && number == a.number;
}
// ... // ...
}; };
@ -6082,7 +6085,7 @@ by making useful operations available for implementers of related new operations
A pure interface class is simply a set of pure virtual functions; see [I.25](#Ri-abstract). A pure interface class is simply a set of pure virtual functions; see [I.25](#Ri-abstract).
In early OOP (e.g., in the 1980s and 1990s), implementation inheritance and interface inheritance were often mixed In early OOP (e.g., in the 1980s and 1990s), implementation inheritance and interface inheritance were often mixed
and bad habits die hard. and bad habits die hard.
Even now, mixtures are not uncommon in old code bases and in old-style teaching material. Even now, mixtures are not uncommon in old code bases and in old-style teaching material.
@ -6099,7 +6102,7 @@ The importance of keeping the two kinds of inheritance increases
class Shape { // BAD, mixed interface and implementation class Shape { // BAD, mixed interface and implementation
public: public:
Shape(); Shape();
Shape(Point ce = {0,0}, Color co = none): cent{ce}, col {co} { /* ... */} Shape(Point ce = {0, 0}, Color co = none): cent{ce}, col {co} { /* ... */}
Point center() const { return cent; } Point center() const { return cent; }
Color color() const { return col; } Color color() const { return col; }
@ -6164,7 +6167,7 @@ Note that a pure interface rarely have constructors: there is nothing to constru
class Circle : public Shape { class Circle : public Shape {
public: public:
Circle(Point c, int r, Color c) :cent{c}, rad{r}, col{c} { /* ... */ } Circle(Point c, int r, Color c) :cent{c}, rad{r}, col{c} { /* ... */ }
Point center() const override { return cent; } Point center() const override { return cent; }
Color color() const override { return col; } Color color() const override { return col; }
@ -6503,7 +6506,8 @@ Capping an individual virtual function with `final` is error-prone as that `fina
class Widget { /* ... */ }; class Widget { /* ... */ };
class My_widget final : public Widget { /* ... */ }; // nobody will ever want to improve My_widget (or so you thought) // nobody will ever want to improve My_widget (or so you thought)
class My_widget final : public Widget { /* ... */ };
class My_improved_widget : public My_widget { /* ... */ }; // error: can't do that class My_improved_widget : public My_widget { /* ... */ }; // error: can't do that
@ -6876,7 +6880,7 @@ Minimize surprises.
// ... // ...
X& operator=(const X&); // member function defining assignment X& operator=(const X&); // member function defining assignment
friend bool operator==(const X&, const X&); // == needs access to representation friend bool operator==(const X&, const X&); // == needs access to representation
// after a=b we have a==b // after a = b we have a == b
// ... // ...
}; };
@ -7409,7 +7413,7 @@ Convenience of use and avoidance of errors.
Day operator++(Day& d) Day operator++(Day& d)
{ {
return d==sun?mon:Day{++d}; return d == sun ? mon : Day{++d};
} }
Day today = sat; Day today = sat;
@ -7435,7 +7439,7 @@ Avoid clashes with macros.
// productinfo.h // productinfo.h
// The following define product subtypes based on color // The following define product subtypes based on color
enum class Productinfo { RED, PURPLE, BLUE }; // syntax error enum class Productinfo { RED, PURPLE, BLUE }; // syntax error
##### Enforcement ##### Enforcement
@ -7458,7 +7462,7 @@ Such code is not uncommon in code written before there were convenient alternati
Use `constexpr` values instead. For example: Use `constexpr` values instead. For example:
constexpr int red = 0x,FF0000; constexpr int red = 0xFF0000;
constexpr short scale = 4; constexpr short scale = 4;
constexpr bool signed = true; constexpr bool signed = true;
@ -7479,7 +7483,7 @@ The default is the easiest to read and write.
enum class Direction : char { n, s, e, w, enum class Direction : char { n, s, e, w,
ne, nw, se, sw }; // underlying type saves space ne, nw, se, sw }; // underlying type saves space
enum class Webcolor : int { red = 0xFF0000, enum class Webcolor : int { red = 0xFF0000,
green = 0x00FF00, green = 0x00FF00,
blue = 0x0000FF }; // underlying type is redundant blue = 0x0000FF }; // underlying type is redundant
@ -7491,7 +7495,7 @@ Specifying the underlying type is necessary in forward declarations of enumerati
enum Flags : char; enum Flags : char;
void f(Flags); void f(Flags);
// .... // ....
enum flags : char { /* ... */ }; enum flags : char { /* ... */ };
@ -7626,8 +7630,8 @@ What is `Port`? A handy wrapper that encapsulates the resource:
operator PortHandle() { return port; } operator PortHandle() { return port; }
// port handles can't usually be cloned, so disable copying and assignment if necessary // port handles can't usually be cloned, so disable copying and assignment if necessary
Port(const Port&) =delete; Port(const Port&) = delete;
Port& operator=(const Port&) =delete; Port& operator=(const Port&) = delete;
}; };
##### Note ##### Note
@ -8689,7 +8693,8 @@ Here, there is a chance that the reader knows what `trim_tail` means and that th
Argument names of large functions are de facto non-local and should be meaningful: Argument names of large functions are de facto non-local and should be meaningful:
void complicated_algorithm(vector<Record>& vr, const vector<int>& vi, map<string, int>& out) void complicated_algorithm(vector<Record>& vr, const vector<int>& vi, map<string, int>& out)
// read from events in vr (marking used Records) for the indices in vi placing (name, index) pairs into out // read from events in vr (marking used Records) for the indices in
// vi placing (name, index) pairs into out
{ {
// ... 500 lines of code using vr, vi, and out ... // ... 500 lines of code using vr, vi, and out ...
} }
@ -9416,13 +9421,15 @@ Requires messy cast-and-macro-laden code to get working right.
#include <cstdarg> #include <cstdarg>
void error(int severity ...) // "severity" followed by a zero-terminated list of char*s; write the C-style strings to cerr // "severity" followed by a zero-terminated list of char*s; write the C-style strings to cerr
void error(int severity ...)
{ {
va_list ap; // a magic type for holding arguments va_list ap; // a magic type for holding arguments
va_start(ap, severity); // arg startup: "severity" is the first argument of error() va_start(ap, severity); // arg startup: "severity" is the first argument of error()
for (;;) { for (;;) {
char* p = va_arg(ap, char*); // treat the next var as a char*; no checking: a cast in disguise // treat the next var as a char*; no checking: a cast in disguise
char* p = va_arg(ap, char*);
if (p == nullptr) break; if (p == nullptr) break;
cerr << p << ' '; cerr << p << ' ';
} }
@ -11139,14 +11146,14 @@ This is asking for deadlock:
Instead, use `lock()`: Instead, use `lock()`:
// thread 1 // thread 1
lock_guard<mutex> lck1(m1,defer_lock); lock_guard<mutex> lck1(m1, defer_lock);
lock_guard<mutex> lck2(m2,defer_lock); lock_guard<mutex> lck2(m2, defer_lock);
lock(lck1,lck2); lock(lck1, lck2);
// thread 2 // thread 2
lock_guard<mutex> lck2(m2,defer_lock); lock_guard<mutex> lck2(m2, defer_lock);
lock_guard<mutex> lck1(m1,defer_lock); lock_guard<mutex> lck1(m1, defer_lock);
lock(lck2,lck1); lock(lck2, lck1);
Here, the writers of `thread1` and `thread2` are still not agreeing on the order of the `mutex`es, but order no longer matters. Here, the writers of `thread1` and `thread2` are still not agreeing on the order of the `mutex`es, but order no longer matters.
@ -11157,7 +11164,7 @@ In real code, `mutex`es are not always conveniently acquired on consecutive line
I'm really looking forward to be able to write plain I'm really looking forward to be able to write plain
lock_guard lck1(m1,defer_lock); lock_guard lck1(m1, defer_lock);
and have the `mutex` type deduced. and have the `mutex` type deduced.
@ -11203,7 +11210,7 @@ Such problem cal often be solved by using a `recursive_mutex`. For example:
// ... // ...
} }
If, as it is likely, `f()` invokes operations on `*this`, we must make sure that the object's invariant holds before the call. If, as it is likely, `f()` invokes operations on `*this`, we must make sure that the object's invariant holds before the call.
##### Enforcement ##### Enforcement
@ -11231,11 +11238,11 @@ If a `thread` joins, we can safely pass pointers to objects in the scope of the
void some_fct(int* p) void some_fct(int* p)
{ {
int x = 77; int x = 77;
raii_thread t0(f,&x); // OK raii_thread t0(f, &x); // OK
raii_thread t1(f,p); // OK raii_thread t1(f, p); // OK
raii_thread t2(f,&glob); // OK raii_thread t2(f, &glob); // OK
auto q = make_unique<int>(99); auto q = make_unique<int>(99);
raii_thread t3(f,q.get()); // OK raii_thread t3(f, q.get()); // OK
// ... // ...
} }
@ -11452,12 +11459,12 @@ Defining "small amount" precisely is impossible.
##### Example ##### Example
string modify1(string); string modify1(string);
void modify2(shared_ptr<string); void modify2(shared_ptr<string>);
void fct(string& s) void fct(string& s)
{ {
auto res = async(modify1,string); auto res = async(modify1, s);
async(modify2,&s); async(modify2, &s);
} }
The call of `modify1` involves copying two `string` values; the call of `modify2` does not. The call of `modify1` involves copying two `string` values; the call of `modify2` does not.
@ -11528,8 +11535,8 @@ Thread creation is expensive.
void master(istream& is) void master(istream& is)
{ {
for (Message m; is>>m; ) for (Message m; is >> m; )
run_list.push_back(new thread(worker,m);} run_list.push_back(new thread(worker, m));
} }
This spawns a `thread` per message, and the `run_list` is presumably managed to destroy those tasks once they are finished. This spawns a `thread` per message, and the `run_list` is presumably managed to destroy those tasks once they are finished.
@ -11546,7 +11553,7 @@ Instead, we could have a set of pre-created worker threads processing the messag
void worker() void worker()
{ {
for (Message m; m=work.get(); ) { for (Message m; m = work.get(); ) {
// process // process
} }
} }
@ -11698,7 +11705,7 @@ An unnamed local objects is a temporary that immediately goes out of scope.
unique_lock<mutex>(m1); unique_lock<mutex>(m1);
lock_guard<mutex> {m2}; lock_guard<mutex> {m2};
lock(m1,m2); lock(m1, m2);
This looks innocent enough, but it isn't. This looks innocent enough, but it isn't.
@ -12086,7 +12093,7 @@ The `File_handle` constructor might defined like this:
:f{fopen(name.c_str(), mode.c_str())} :f{fopen(name.c_str(), mode.c_str())}
{ {
if (!f) if (!f)
throw runtime_error{"File_handle: could not open "S-+ name + " as " + mode"} throw runtime_error{"File_handle: could not open " + name + " as " + mode};
} }
##### Note ##### Note
@ -12097,7 +12104,7 @@ Examples:
* A precondition that cannot be met * A precondition that cannot be met
* A constructor that cannot construct an object (failure to establish its class's [invariant](#Rc-struct)) * A constructor that cannot construct an object (failure to establish its class's [invariant](#Rc-struct))
* An out-of-range error (e.g., `v[v.size()] =7`) * An out-of-range error (e.g., `v[v.size()] = 7`)
* Inability to acquire a resource (e.g., the network is down) * Inability to acquire a resource (e.g., the network is down)
In contrast, termination of an ordinary loop is not exceptional. In contrast, termination of an ordinary loop is not exceptional.
@ -12132,7 +12139,7 @@ C++ implementations tend to be optimized based on the assumption that exceptions
int find_index(vector<string>& vec, const string& x) int find_index(vector<string>& vec, const string& x)
{ {
try { try {
for (int i =0; i < vec.size(); ++i) for (int i = 0; i < vec.size(); ++i)
if (vec[i] == x) throw i; // found x if (vec[i] == x) throw i; // found x
} catch (int i) { } catch (int i) {
return i; return i;
@ -13772,10 +13779,10 @@ Otherwise they cannot be distinguished automatically by the compiler.
##### Example (using TS concepts) ##### Example (using TS concepts)
template<typename I> template<typename I>
concept bool Input_iter = requires (I iter) { ++iter; }; concept bool Input_iter = requires(I iter) { ++iter; };
template<typename I> template<typename I>
concept bool Fwd_iter = Input_iter<I> && requires (I iter) { iter++; } concept bool Fwd_iter = Input_iter<I> && requires(I iter) { iter++; }
The compiler can determine refinement based on the sets of required operations (here, suffix `++`). The compiler can determine refinement based on the sets of required operations (here, suffix `++`).
This decreases the burden on implementers of these types since This decreases the burden on implementers of these types since
@ -13807,7 +13814,7 @@ The programmer (in a library) must define `is_contiguous` (a trait) appropriatel
Wrapping a tag class into a concept leads to a simpler expression of this idea: Wrapping a tag class into a concept leads to a simpler expression of this idea:
template<typename I> concept Contiguous = is_contiguous<I>::value; template<typename I> concept Contiguous = is_contiguous<I>::value;
template<typename I> template<typename I>
concept bool Contiguous_iter = RA_iter<I> && Contiguous<I>; concept bool Contiguous_iter = RA_iter<I> && Contiguous<I>;
@ -13907,10 +13914,10 @@ Obviously, it would be better and easier just to use the standard `EqualityCompa
but - just as an example - if you had to define such a concept, prefer: but - just as an example - if you had to define such a concept, prefer:
template<typename T> concept Equality = requires(T a, T b) { template<typename T> concept Equality = requires(T a, T b) {
bool == { a==b } bool == { a == b }
bool == { a!=b } bool == { a != b }
// axiom { !(a==b)==(a!=b) } // axiom { !(a == b) == (a != b) }
// axiom { a=b; => a==b } // => means "implies" // axiom { a = b; => a == b } // => means "implies"
} }
as opposed to defining two meaningless concepts `has_equal` and `has_not_equal` just as helpers in the definition of `Equality`. as opposed to defining two meaningless concepts `has_equal` and `has_not_equal` just as helpers in the definition of `Equality`.
@ -14210,7 +14217,7 @@ That is, it is highly visible.
##### Note ##### Note
This rule should not be necessary, but the committee cannot agree to exclude unconstrained templated from ADL. This rule should not be necessary, but the committee cannot agree to exclude unconstrained templated from ADL.
Unfortunately this will get many false positives; the standard library violates this widely, by putting many unconstrained templates and types into the single namespace `std`. Unfortunately this will get many false positives; the standard library violates this widely, by putting many unconstrained templates and types into the single namespace `std`.
@ -14353,7 +14360,6 @@ This looks innocent enough, but ???
// requires Regular<T> && Allocator<A> // requires Regular<T> && Allocator<A>
class List2 { class List2 {
public: public:
using iterator = Link<T>*; using iterator = Link<T>*;
iterator first() const { return head; } iterator first() const { return head; }
@ -14448,7 +14454,7 @@ This is a simplified version of `std::copy` (ignoring the possibility of non-con
struct pod_tag {}; struct pod_tag {};
struct non_pod_tag; struct non_pod_tag;
template<class T> struct copy_trait { using tag = non_pod_tag; }; // T is not "plain old data" template<class T> struct copy_trait { using tag = non_pod_tag; }; // T is not "plain old data"
template<> struct copy_trait<int> { using tab = pod_tag; }; // int is "plain old data" template<> struct copy_trait<int> { using tab = pod_tag; }; // int is "plain old data"
@ -15004,7 +15010,7 @@ Documentation, readability, opportunity for reuse.
auto x = find_if(vr.begin(), vr.end(), auto x = find_if(vr.begin(), vr.end(),
[&](Rec& r) { [&](Rec& r) {
if (r.name.size() != n.size()) return false; // name to compare to is in n if (r.name.size() != n.size()) return false; // name to compare to is in n
for (int i=0; i < r.name.size(); ++i) for (int i = 0; i < r.name.size(); ++i)
if (tolower(r.name[i]) != tolower(n[i])) return false; if (tolower(r.name[i]) != tolower(n[i])) return false;
return true; return true;
} }
@ -15014,20 +15020,20 @@ There is a useful function lurking here (case insensitive string comparison), as
bool compare_insensitive(const string& a, const string& b) bool compare_insensitive(const string& a, const string& b)
{ {
if (a.size()!=b.size()) return false; if (a.size() != b.size()) return false;
for (int i=0; i<a.size(); ++i) if (tolower(a[i])!=tolower(b[i])) return false; for (int i = 0; i < a.size(); ++i) if (tolower(a[i]) != tolower(b[i])) return false;
return true; return true;
} }
auto x = find_if(vr.begin(),vr.end(), auto x = find_if(vr.begin(), vr.end(),
[&](Rec& r) { compare_insensitive(r.name, n); } [&](Rec& r) { compare_insensitive(r.name, n); }
); );
Or maybe (if you prefer to avoid the implicit name binding to n): Or maybe (if you prefer to avoid the implicit name binding to n):
auto cmp_to_n = [&n](const string& a) { return compare_insensitive(a, n); }; auto cmp_to_n = [&n](const string& a) { return compare_insensitive(a, n); };
auto x = find_if(vr.begin(),vr.end(), auto x = find_if(vr.begin(), vr.end(),
[](const Rec& r) { return cmp_to_n(r.name); } [](const Rec& r) { return cmp_to_n(r.name); }
); );
@ -15055,7 +15061,7 @@ That makes the code concise and gives better locality than alternatives.
auto earlyUsersEnd = std::remove_if(users.begin(), users.end(), auto earlyUsersEnd = std::remove_if(users.begin(), users.end(),
[](const User &a) { return a.id > 100; }); [](const User &a) { return a.id > 100; });
##### Exception ##### Exception
@ -16756,7 +16762,8 @@ Code says what is done, not what is supposed to be done. Often intent can be sta
##### Example ##### Example
void stable_sort(Sortable& c) void stable_sort(Sortable& c)
// sort c in the order determined by <, keep equal elements (as defined by ==) in their original relative order // sort c in the order determined by <, keep equal elements (as defined by ==) in
// their original relative order
{ {
// ... quite a few lines of non-trivial code ... // ... quite a few lines of non-trivial code ...
} }
@ -16799,9 +16806,9 @@ Readability. Avoidance of "silly mistakes."
Always indenting the statement after `if (...)`, `for (...)`, and `while (...)` is usually a good idea: Always indenting the statement after `if (...)`, `for (...)`, and `while (...)` is usually a good idea:
if (i<0) error("negative argument"); if (i < 0) error("negative argument");
if (i<0) if (i < 0)
error("negative argument"); error("negative argument");
##### Enforcement ##### Enforcement
@ -16828,10 +16835,10 @@ Minimize unintentional conversions.
Names with types encoded are either verbose or cryptic. Names with types encoded are either verbose or cryptic.
printS // print a std::string printS // print a std::string
prints // print a C-style string prints // print a C-style string
printi // print an int printi // print an int
PS. Hungarian notation is evil (at least in a strongly statically-typed language). PS. Hungarian notation is evil (at least in a strongly statically-typed language).
##### Note ##### Note
@ -16992,7 +16999,7 @@ Too much space makes the text larger and distracts.
#include < map > #include < map >
int main (int argc, char * argv [ ]) int main(int argc, char * argv [ ])
{ {
// ... // ...
} }
@ -17167,8 +17174,8 @@ It is really easy to overlook a statement when there is more on a line.
##### Example ##### Example
int x = 7; char* p = 29; // dont int x = 7; char* p = 29; // don't
int x = 7; f(x); ++x; // dont int x = 7; f(x); ++x; // don't
##### Enforcement ##### Enforcement