0

別の質問では、ユーザーが UDP 経由で QuartzComposer にデータを送信する方法を提供しました。QuartzComposer は、送信されたバイト データのパディングに非常にこだわっているようです。特に、送信される各文字の前に「\0\0\0」を付ける必要があります。

この問題を解決するために、簡単な Python プログラムをモックアップすることができました。

from socket import *

PORT = 50000
N = 3.14159265358

# connect to Quartz Composer on localhost port 50000
s = socket(AF_INET, SOCK_DGRAM)
s.bind(('', 0))
s.setsockopt(SOL_SOCKET, SO_BROADCAST, 1)

for c in str(N):
    out = out + '\0\0\0' + c
s.sendto(out, ('225.0.0.0', PORT))

C++ で実装する必要があるクリティカル セクションは、浮動小数点数をパディングするビットです。クリティカル セクションの簡単な Python モックアップを次に示します。

def padfloat(n):
    out = ''
    for c in str(n):
        out = out + '\0\0\0' + c
    return out

if __name__ == '__main__':
    print(padfloat(3.14159265358))

確かに、私の C++ チョップはそれほどタイトではありませんが、boost を使用して UDP を起動して実行し、単一のハードコードされたメッセージ「0.7」を送信することができました。

//
// Working C++ to Quartz Network Test
// 

#include <cstdlib>
#include <cstring>
#include <iostream>
#include <boost/asio.hpp>

using boost::asio::ip::udp;

enum { max_length = 1024 };

int main(int argc, char* argv[])
{
  try
  {
boost::asio::io_service io_service;

udp::socket s(io_service, udp::endpoint(udp::v4(), 0));

udp::resolver resolver(io_service);
udp::resolver::query query(udp::v4(), "225.0.0.0", "50000");
udp::resolver::iterator iterator = resolver.resolve(query);

using namespace std; // For strlen.

// TODO: alter code to take in any number

for (;;) {
  std::cout << "Press Any Key to send 0.7: ";
  char request[max_length];
  std::cin.getline(request, max_length);
  // size_t request_length = strlen(request);
  char test [] = {0,0,0,'0',0,0,0,'.',0,0,0,'7','\0'};
  s.send_to(boost::asio::buffer(test, 12), *iterator);
  std::cout << "Sent\n";
}
  }
  catch (std::exception& e)
  {
std::cerr << "Exception: " << e.what() << "\n";
  }
  return 0;
}

私が助けを必要とするかなり些細な問題は、浮動小数点数を取り込み、それを char 配列として適切にフォーマットして、すべての「\0\0\0」パディングを適切に先頭に追加して送信する方法です。私は何か醜いことをしようとしていますが、誰かが私にエレガントな修正を指摘できるなら、私は少し学ぶことができてとてもうれしいです.

4

1 に答える 1

1

手元にコンパイラがありませんが、これで十分です。

  1. 文字列変換を行います
  2. 0 に初期化された、サイズの 4 倍 (さらに null 終端の場合は 1) のバッファーを作成します。
  3. 文字列の 4 番目ごとの位置にコピーします。

asio が null ターミネータがそのサイズの一部であると想定しているかどうかわからないので、あなたの例に従いました。

float float_value = 4.2f;
std::string str = boost::lexical_cast<std::string>(float_value);

std::vector<char> char_buff((str.size() * 4) + 1, 0);

for (size_t i = 0; i < str.size(); i++)
  char_buff[(i * 4) + 3] = str[i];

s.send_to(boost::asio::buffer(&char_buff[0], char_buff.size()), *iterator);
于 2011-09-09T22:11:53.040 に答える