C++17 introduced a construct to create and initialize a variable within the condition of if and switch statements and
C++20 added this construct to range-based for loops. Using this new feature simplifies common code patterns and helps in giving variables
the right scope.
Previously, variables were either declared before the statement, hence leaked into the ambient scope, or an explicit scope was used to keep the
scope tight, especially when using RAII objects. This was inconvenient as it would lead to error-prone patterns.
For example, this verbose error-prone initialization:
bool error_prone_init() {
  { // explicit scope
    std::unique_lock<std::mutex> lock(mtx, std::try_to_lock);
    if (lock.owns_lock()) {
       //...
     }
  } // mutex unlock
  // ... code
  return true;
}
can now be replaced by the following code, which is safer and more readable:
bool better_init() {
  if (std::unique_lock<std::mutex> lock(mtx, std::try_to_lock); lock.owns_lock()) {
     //...
  } // mutex unlock
  // ... code
  return true;
}
This rule raises an issue when:
  -  a variable is declared just before a statement that allows variable declaration (if,switch),
-  this variable is used in the statement header, 
-  there are other statements after this statement where this variable might be used, 
-  yet, it is never used after the statement. 
Noncompliant code example
void handle(std::string_view s);
void ifStatement() {
  std::map<int, std::string> m;
  int key = 1;
  std::string value = "str1";
  auto [it, inserted] = m.try_emplace(key, value); // Noncompliant
  if (!inserted) {
    std::cout << "Already registered";
  } else {
    handle(it->second);
  }
  process(m);
}
enum class State { True, False, Maybe, MaybeNot };
std::pair<std::string, State> getStatePair();
void switchStatement() {
  auto state = getStatePair(); // Noncompliant
  switch (state.second) {
    case State::True:
    case State::Maybe:
      std::cout << state.first;
      break;
    case State::False:
    case State::MaybeNot:
      std::cout << "No";
      break;
  }
  std::cout << "\n";
}
Compliant solution
void handle(std::string_view s);
void ifStatement() {
  std::map<int, std::string> m;
  int key = 1;
  std::string value = "str1";
  if (auto [it, inserted] = m.try_emplace(key, value); !inserted) { // Compliant
    std::cout << "Already registered";
  } else {
    handle(it->second);
  }
  process(m);
}
enum class State { True, False, Maybe, MaybeNot };
std::pair<std::string, State> getStatePair();
void switchStatement() {
  switch (auto state = getStatePair(); state.second) { // Compliant
    case State::True:
    case State::Maybe:
      std::cout << state.first;
      break;
    case State::False:
    case State::MaybeNot:
      std::cout << "No";
      break;
  }
  std::cout << "\n";
}
Exceptions
While an if with both an initializer and a condition variable is valid, it is confusing. The rule does not raise an issue if the
if statement already has a condition variable:
void confusing() {
  if (int a = 42; std::optional<int> b = lookup(a)) { // Valid but confusing
    // ...
  }
}
void exception() {
  int a = 42; // Compliant by exception
  if (std::optional<int> b = lookup(a)) {
    // ...
  }
}