С именованными каналами чтение и запись являются (или могут быть) атомарными. В определенных пределах, если вы записываете, скажем, 1024 байта в канал, вызов чтения на другом конце, который ищет как минимум 1024 байта, фактически получит 1024 байта, даже если во время передачи в канале было больше данных. читать. Кроме того, и всегда, если в именованном канале всего 1024 байта, а чтение требует 4096 байтов, он получит 1024 байта с первой попытки и заблокирует только при следующей попытке.
Вы говорите:
Учитывая, что я должен предоставить буфер и размер буфера для чтения,
Вы делаете ...
и учитывая, что команда чтения блокирует (я полагаю),
Да, если вы не установите O_NONBLOCK в дескрипторе файла ...
У меня либо должен быть размер буфера, который гарантирует, что я никогда не получу недостаточную производительность,
Какие сообщения вы отправляете? С каким размером ты имеешь дело? Килобайт, мегабайт, больше?
или узнать размер сообщения заранее.
Нет особой проблемы с наличием, скажем, буфера 4 КБ в считывателе и чтением сообщения кусками. Проблема в том, что вы знаете, когда вы дойдете до конца сообщения. Большинству протоколов требуется длина заранее, потому что это позволяет легко писать код считывателя.
Если вы собираетесь сделать маркер «конца потока» (EOS), вы делаете «внутриполосную сигнализацию». И это вызывает проблемы. Какой персонаж вы собираетесь использовать? Что происходит, когда этот символ появляется в данных? Вам нужен механизм выхода, такой как символ, который означает «следующий символ не является маркером EOS». Например, в тексте, связанном с программированием, для этого используется обратный слеш. В терминале control-V часто служит цели.
Я не хочу, чтобы программа-отправитель знала размер буфера и дополняла его.
Почему отправителю трудно узнать размер буфера? И зачем это нужно "набирать"?
Если вы имеете дело с большими объемами данных (например, от килобайта и выше), односимвольное решение вряд ли даст приемлемую производительность. Я думаю, что было бы лучше, если бы отправитель мог определить размер пакета и сообщить об этом читателю, или разработать протокол так, чтобы были ограничения на размер пакета. Если вам нужно передать произвольные объемы данных, используйте протокол, который гласит:
- Большое количество данных неизвестного общего размера.
- Для каждого подпакета в сообщении говорится «это подпакет размером NN КБ».
- Для последнего подпакета размер может быть короче - это нормально и может указывать на «конец большого количества данных».
- Если последний подпакет имеет «полный размер», вы можете отправить пустой последний пакет, чтобы указать EOS.
- В качестве альтернативы, если подпакеты могут иметь переменный размер, вы всегда можете отправить явный пакет EOS.
Также подумайте о том, что произойдет в будущем, если вместо использования именованных каналов вы захотите обновить свою систему для работы через сокет-соединение с другим компьютером.
Я думаю, что вы должны проектировать свою систему с пакетами, в которых заголовки пакетов включают размер данных (как это делают большинство сетевых протоколов, таких как TCP / IP). И если есть поток данных более высокого уровня с неизвестным размером, обработайте его в соответствии с указанными выше линиями. Но даже там лучше, если вы сможете заранее определить общий размер.