0

boost::python でバイナリを Python から C++ に変換する必要があります。バイナリはイメージ ファイルまたはテキスト ファイルから取得されている可能性があります。ただし、イメージ ファイルのバイナリを C++ に変換するとエラーが発生します。以下はその例です。

C++

#include <boost/python.hpp>
#include <boost/python/module.hpp>
#include <boost/python/def.hpp>
#include <fstream>
#include <iostream>

using namespace boost::python;
void greet(char *name,char *ss)
{
    std::ofstream fout(name,std::ios::binary);
    std::cout<< "This length is:" << strlen(ss) <<std::endl;
    fout.write(ss.strlen);
    fout.close();
    return;
}

BOOST_PYTHON_MODULE(ctopy)
{
    def("greet",greet);
}

パイソン:

import ctopy
#It is right.
f=open("t1.txt","rb")
ctopy.greet("t2.txt",f.read())
f.close()

#Do a error.There isn't data in the file "p2.jpg".
f2=open("p1.jpg","rb")
ctopy.greet("p2.jpg",f2.read()) #error.the file "p2.jpg" will be a empty file.
f2.close()

画像のバイナリを C++ に変換するには?

4

2 に答える 2

1

バイナリ ファイルのエンコーディングは、通常、ファイルの種類、オペレーティング システムなど、プログラミング言語以外の要因に依存します。 C++ と Python の両方で同じように表されます。どちらの言語も、指定された形式に適切なエンコーディングを使用する必要があります。この場合、Python バイナリ ストリームを C++ バイナリ ストリームに変換する特別なプロセスはありません。これは、両方の言語で生のバイト ストリームであるためです。

元のコードのアプローチには 2 つの問題があります。

  • strlen()null で終わる文字列の長さを決定するために使用する必要があります。バイナリ ファイルに の値を持つバイトが含まれている場合、\0strlen()データ全体のサイズを返しません。
  • char*の代わりに が使用されているため、データのサイズが失われstd::stringます。std::stringサイズを提供しsize()、文字列自体にヌル文字を許可するため、使用を検討してください。別の解決策は、データと一緒にデータのサイズを明示的に提供することです。

完全な Boost.Python の例を次に示します。

#include <boost/python.hpp>
#include <fstream>
#include <iostream>

void write_file(const std::string& name, const std::string& data)
{
  std::cout << "This length is: " << data.size() << std::endl;
  std::ofstream fout(name.c_str(), std::ios::binary);
  fout.write(data.data(), data.size());
  fout.close();
}

BOOST_PYTHON_MODULE(example)
{
  using namespace boost::python;
  def("write", &write_file);
}

Python コードの例 ( test.py):

import example

with open("t1.txt", "rb") as f:
    example.write("t2.txt", f.read())

with open("p1.png", "rb") as f:
    example.write("p2.png", f.read())

そして、この画像をダウンロードして単純なテキスト ファイルを作成し、上記のコードでそれらのコピーを作成する使用法:

[twsansbury]$ wget http://www.boost.org/style-v2/css_0/get-boost.png -O p1.png >> /dev/null 2>&1
[twsansbury]$ echo "これはテストです" > t1.txt
[twsansbury]$ ./test.py
この長さ: 15
この長さ: 27054
[twsansbury]$ md5sum t1.txt t2.txt
e19c1283c925b3206685ff522acfe3e6 t1.txt
e19c1283c925b3206685ff522acfe3e6 t2.txt
[twsansbury]$ md5sum p1.png p2.png
fe6240bff9b29a90308ef0e2ba959173 p1.png
fe6240bff9b29a90308ef0e2ba959173 p2.png

md5 チェックサムが一致しており、ファイルの内容が同じであることを示しています。

于 2013-02-11T22:33:35.087 に答える
0

最小限の例を作成した後、実際のコードを提供してください。さらに、どのPythonバージョンを使用していますか?とにかく、あなたが提供したコードで間違っていることがいくつかあります:

  • constC ++ FAQでわかるように、を使用する必要があります。
  • strlen()を使用しているのは、ゼロで終了することが保証されていないものの、途中にゼロが含まれている可能性があるものです。
  • std :: stringを使用する必要があります。これは、内部null文字がある場合はbarfしません。
  • ファイルを閉じることは無意味です。それはdtorで自動的に行われます。ストリーム状態のフラッシュとチェックは、はるかに興味深いものです。失敗した場合は、例外をスローします。
  • トレーリングリターンを落としてください、それは傷つきませんが、それは不必要なノイズです。
  • PEP8をお読みください。
  • ファイルの読み取りにはwithステートメントを使用します。
于 2013-02-11T21:23:40.900 に答える