0

我只是想知道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;
}
4

5 回答 5

6

这就像for loop所有 3 个表达式都没有任何内容。它相当于:

while(true) { ... }

我只是想知道原因

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

于 2013-03-28T06:30:08.083 回答
5

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

于 2013-03-28T06:30:49.540 回答
4

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

  • 的使用break,专门用于中断循环的执行
  • return在函数上下文中使用
  • 用于throw抛出异常
  • 指某东西的用途goto
于 2013-03-28T06:36:15.710 回答
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));
}

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

这被称为循环半

使用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);
}

关于循环的表达,我更喜欢原来表达的代码,中间有exit。

于 2013-03-28T06:48:03.717 回答
2

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

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

在第 2 段中它说:

可以省略条件和表达式中的一个或两个。缺少条件使隐含的 while 子句等同于 while(true)。

于 2013-03-28T06:59:30.863 回答