C ++:for(;;)的目的?

时间:2013-03-28 06:29:02

标签: c++ for-loop boost-asio

我只是想知道以下代码中for(;;)行的原因。

//
// blocking_tcp_echo_server.cpp
// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
//
// Copyright (c) 2003-2012 Christopher M. Kohlhoff (chris at kohlhoff dot com)
//
// Distributed under the Boost Software License, Version 1.0. (See accompanying
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
//

#include <cstdlib>
#include <iostream>
#include <boost/bind.hpp>
#include <boost/smart_ptr.hpp>
#include <boost/asio.hpp>
#include <boost/thread/thread.hpp>

using boost::asio::ip::tcp;

const int max_length = 1024;

typedef boost::shared_ptr<tcp::socket> socket_ptr;

void session(socket_ptr sock)
{
  try
  {
    for (;;)
    {
      char data[max_length];

      boost::system::error_code error;
      size_t length = sock->read_some(boost::asio::buffer(data), error);
      if (error == boost::asio::error::eof)
        break; // Connection closed cleanly by peer.
      else if (error)
        throw boost::system::system_error(error); // Some other error.

      boost::asio::write(*sock, boost::asio::buffer(data, length));
    }
  }
  catch (std::exception& e)
  {
    std::cerr << "Exception in thread: " << e.what() << "\n";
  }
}

void server(boost::asio::io_service& io_service, short port)
{
  tcp::acceptor a(io_service, tcp::endpoint(tcp::v4(), port));
  for (;;)
  {
    socket_ptr sock(new tcp::socket(io_service));
    a.accept(*sock);
    boost::thread t(boost::bind(session, sock));
  }
}

int main(int argc, char* argv[])
{
  try
  {
    if (argc != 2)
    {
      std::cerr << "Usage: blocking_tcp_echo_server <port>\n";
      return 1;
    }

    boost::asio::io_service io_service;

    using namespace std; // For atoi.
    server(io_service, atoi(argv[1]));
  }
  catch (std::exception& e)
  {
    std::cerr << "Exception: " << e.what() << "\n";
  }

  return 0;
}

5 个答案:

答案 0 :(得分:6)

就像for loop一样,所有3个表达都没有。它相当于:

while(true) { ... }
  

我只是想知道推理

在这种情况下,代码作者希望指定的代码块“永远”运行,直到他告诉它不要运行(通过使用break明确退出,returnthrow等等。)

答案 1 :(得分:5)

永远循环,因此您需要让breakreturn退出循环。

答案 2 :(得分:4)

for (;;)它永远运行的目的。它相当于while (true)。它将需要以下条件之一来打破:

  • 使用break,专门用于打破循环执行
  • 在函数上下文中使用return
  • 使用throw抛出异常
  • 使用goto

答案 3 :(得分:4)

有问题的for循环,

for (;;)
{
  char data[max_length];
  boost::system::error_code error;
  size_t length = sock->read_some(boost::asio::buffer(data), error);
  if (error == boost::asio::error::eof)
    break; // Connection closed cleanly by peer.
  else if (error)
    throw boost::system::system_error(error); // Some other error.
  boost::asio::write(*sock, boost::asio::buffer(data, length));
}

在中间实现了一个正常退出的循环。

这被称为 loop-and-a-half

使用for(;;)代替while(true)可以避免使用至少一个编译器(即Visual C ++)发出愚蠢警告。它也更惯用。如果while循环意味着扮演这样一个角色,那么它将具有默认条件(如for循环所有)。

重写这样一个循环以避免中间退出的一种方法是使循环体的第二部分成为条件,如下所示:

for ( bool finished = false; !finished;)
{
  char data[max_length];
  boost::system::error_code error;
  size_t length = sock->read_some(boost::asio::buffer(data), error);
  if (error == boost::asio::error::eof)
    finished = true; // Connection closed cleanly by peer.
  else
  {
    if (error)
      throw boost::system::system_error(error); // Some other error.
    boost::asio::write(*sock, boost::asio::buffer(data, length));
  }
}

表达它的另一种方式,但具有相当不正确的代码冗余,是在循环之前重复循环的上半部分,一半展开循环,如下所示:

char data[max_length];
boost::system::error_code error;
size_t length = sock->read_some(boost::asio::buffer(data), error);
for (; error != boost::asio::error::eof; )
{
  if (error)
    throw boost::system::system_error(error); // Some other error.

  boost::asio::write(*sock, boost::asio::buffer(data, length));
  size_t length = sock->read_some(boost::asio::buffer(data), error);
}

关于表达循环,我更喜欢最初表达的代码,退出中间。

答案 4 :(得分:2)

ISO / IEC 14882:2003在第6.5.3节第1段中陈述了for循环的结构:

for ( for-init-statement condition_opt ; expression_opt ) statement

在第2段中,它说:

  

可以省略条件和表达式中的任何一个或两个。一个   缺失条件使隐含的while子句等效于   而(真)。