В C++, как система реализует буферизованный поток?

Может кто-нибудь написать пример кода, чтобы объяснить эту концепцию? Я знаю, для чего используется буферный поток, но я также хотел бы знать, как это реализовать.

Заранее спасибо!

3 ответа

Решение

Вы можете посмотреть на реализацию вашей платформы, стандарт C++ или "Стандартные потоки ввода-вывода C++ и локали" Анжелики Лангер и Клауса Крефта.

Будьте готовы к довольно кривой обучения. Ручьи старые и сложное дело. (Фрэнсис Глассбороу: "У меня очень мало сомнений в том, что библиотеки ввода-вывода являются одними из самых сложных аспектов любого языка".)

Схема вееее, для "входного" потока:

class BufferedInputStream
{
public:
  BufferedInputStream(SomeExternalDevice d)
  : m_Device(d),
    m_Current(m_Buffer),
    m_End(m_Buffer)
  {}

  char get(){
    if (!IsDataAvailableInBuffer()){
      ReadAChunkFromDiskAndPutItInBuffer();
    }
    return PopByteFromBuffer();
  }

private:

  bool IsDataAvailableInBuffer()const{
    return m_Current != m_End;
  }

  void ReadAChunkFromDiskAndPutItInBuffer(){
    // Buffer must be empty
    assert(!IsDataAvailableInBuffer());

    // Read bytes from the device
    bytes_read = Read(m_Device, m_Buffer, BufferSize);

    // Reposition the "begin" and "end" pointers
    m_Current = m_Buffer;
    m_End = m_Buffer + bytes_read;
  }

  char PopByteFromBuffer(){
    assert(IsDataAvailableInBuffer());
    return *m_Current++;
  }

  // For example, an OS file handle
  SomeExternalDevice m_Device;

  // The buffer itself
  char m_Buffer[BufferSize];

  // Begin of buffer contents
  char* m_Current;

  // End of buffer contents
  char* m_End;
};

Таким образом, данные считываются с диска кусками размера буфера, и большинство вызовов get() не должны заканчиваться вызовами ОС, поскольку они могут просто возвращать байт из буфера.

Взгляните на реализацию STL для sstream и sstream.tcc (ссылки на реализацию SGI STL).

База stringstream класс basic_stringstream, который реализует basic_iostream интерфейс.

  // [27.7.4] Template class basic_stringstream
  /**
   *  @brief  Controlling input and output for std::string.
   *
   *  This class supports reading from and writing to objects of type
   *  std::basic_string, using the inherited functions from
   *  std::basic_iostream.  To control the associated sequence, an instance
   *  of std::basic_stringbuf is used, which this page refers to as @c sb.
  */

Есть базовый класс basic_stringbuf который вытекает из basic_streambuf, Это держит буфер.

  // [27.7.1] template class basic_stringbuf
  /**
   *  @brief  The actual work of input and output (for std::string).
   *
   *  This class associates either or both of its input and output sequences
   *  with a sequence of characters, which can be initialized from, or made
   *  available as, a @c std::basic_string.  (Paraphrased from [27.7.1]/1.)
   *
   *  For this class, open modes (of type @c ios_base::openmode) have
   *  @c in set if the input sequence can be read, and @c out set if the
   *  output sequence can be written.
  */
Другие вопросы по тегам