Отправка данных сокетов с ведущим значением длины

Я хочу отправлять сообщения JSON из PHP-скрипта в приложение C# через сетевое соединение с использованием PHP-сокетов.

Обычно для двоичных протоколов первые 4 байта каждого сообщения должны быть целым числом, которое представляет длину (сколько байтов) сообщения.

В C# я добавляю к каждому сообщению целое число, которое сообщает длину сообщения следующим образом:

byte[] msgBytes = UTF8Encoding.UTF8.GetBytes("A JSON msg");            
byte[] prefixBytes = BitConverter.GetBytes(msgBytes.Length);
byte[] msgToSend = new byte[prefixBytes.Length + msgBytes.Length];
Buffer.BlockCopy(prefixBytes, 0, msgToSend, 0, prefixBytes.Length);
Buffer.BlockCopy(msgBytes, 0, msgToSend, prefixBytes.Length, msgBytes.Length);

Как я понимаю, в PHP функция socket_send принимает только строки. Итак, как я могу сделать такой же префикс в PHP 5.x?

Обновление: я разместил дополнительный вопрос о том, как обрабатывать такие префиксные данные при получении из сетевого сокета.

2 ответа

Решение

В PHP строки являются двоичными.

Таким образом, вам необходимо закодировать значение длины целого как двоичное представление целого числа без знака в виде строки из 4 символов (4 октета; 32 бита). Увидеть pack:

# choose the right format according to your byte-order needs:

l   signed long (always 32 bit, machine byte order)
L   unsigned long (always 32 bit, machine byte order)
N   unsigned long (always 32 bit, big endian byte order)
V   unsigned long (always 32 bit, little endian byte order)

$string = pack('l', $length);

Я думаю, вы могли бы использовать pack() для преобразования количества байтов в двоичную строку. Когда вы отправляете свои данные по сети, вам, вероятно, необходимо выполнить преобразование в формате "N" (длинный без знака, всегда 32-битный, порядок байтов с прямым порядком байтов).

Вот пример:

$s="Hello World";
$length=pack("N",strlen($s));
socket_send($sock,$length.$s,4+strlen($s));
Другие вопросы по тегам