くだすれC++/CLI(初心者用)part2

このエントリーをはてなブックマークに追加
847765:2014/06/21(土) 16:55:19.23 ID:BNLCZco1
久しぶりにVS開いたらWinFormが消えとる。もう捨てられるのか

>>845
これか
http://code.msdn.microsoft.com/windowsdesktop/CppHostCLR-e6581ee0

>>846
\でエスケープします
regex r("\[43\]");
848デフォルトの名無しさん:2014/06/21(土) 16:56:33.36 ID:BNLCZco1
regex r("\\[43\\]");
こうか
849デフォルトの名無しさん:2014/06/25(水) 18:40:03.79 ID:a/L3Ky6p
こんどCLI+API+STLで久々に開発するぜ
まともにWindowsアプリ作るの15年ぶりだぜ
850デフォルトの名無しさん:2014/06/25(水) 18:40:43.32 ID:jtRedncf
ご愁傷様
851デフォルトの名無しさん:2014/06/25(水) 19:43:48.20 ID:a/L3Ky6p
なんでご愁傷様?
852デフォルトの名無しさん:2014/06/29(日) 15:12:56.12 ID:74KQZ/Pg
WebBrowser::ActiveXInstanceのようなCOMポインターをObject^で返したいんですが
どう書いたらいいんでしょうか

Object^ get()
{
IUnknown* pU; //これを返したい
...
}

C#だと全部属性でやってて生ポインタをどう処理するかの参考にならないです。
853デフォルトの名無しさん:2014/06/30(月) 07:13:06.08 ID:AE59suyp
それを返した先でどうするのさ
例えばIntPtrを返しても一応Object^にはなるけど多分そういう事じゃないよね?
854デフォルトの名無しさん:2014/06/30(月) 08:29:40.70 ID:k6aB52Mb
Object^o=Marshal::GetTypedObjectForIUnknown(IntPtr(pU),Object::typeid);
とりあえずこうしました
addref,releaseが同じにならないといやだけど
855デフォルトの名無しさん:2014/07/30(水) 21:04:47.45 ID:4VkY0HvI
C++/CLI超楽しい
.Net、WinAPI、STL、Boost混在できて使いたい放題だし
何でみんな毛嫌いするの?
856デフォルトの名無しさん:2014/07/30(水) 22:46:45.65 ID:310IZqCv
C++とは似て異なるから
857デフォルトの名無しさん:2014/07/30(水) 23:48:31.72 ID:QJA3s1XV
最初から別言語だと思っとけばいいじゃん。
俺もC++/CLIは面白いと思うが、先がなさそうなのが残念だな。
WPFが使えるようになったらMFCのアプリを移行したいと思っていたんだが。
858デフォルトの名無しさん:2014/07/31(木) 13:36:46.98 ID:N7h43RT+
やりたい放題というのは1人でやってるだけなら楽しいが
足を引っ張る仲間がいると地獄巡り満腹コースになる
859デフォルトの名無しさん:2014/07/31(木) 20:54:03.92 ID:Ng6XWAlf
マネージ関数の中でlambda使えないとか、最近は混ざる利点みたいなものが少なく…。
860デフォルトの名無しさん:2014/08/05(火) 16:06:10.29 ID:8/O3Z/TB
genericとtemplate混ぜたら楽しいと思ってたけど、
あんまり混ざらないようにできてる・・
861デフォルトの名無しさん:2014/08/05(火) 21:42:31.92 ID:a3vuC94X
variadic template使って型パラメータの数が違うgeneric classを使えるかと思ったら、なんかエラーになるしね。
862デフォルトの名無しさん:2014/08/06(水) 12:55:33.48 ID:YJYkBYpf
genericクラスの中にtemplateは作れないし
その逆もできない。
できるのは、genericクラスをtemplateで
継承するくらい。

>>861 variadic genericあったらいいなぁ
863デフォルトの名無しさん:2014/09/26(金) 00:00:55.65 ID:Q/qK3xNe
C++/CLIに直接関わる質問ではないのですが
適切なスレが見当たらなかったのでここで質問させてください。

ネイティブのnative.dllをマネージのmanage.dllが参照しているとき
native.dllにパスが通っていない状態でmanage.dllを使用すると
FileNotFoundExceptionが発生してしまいますが、
その例外の値からはnative.dllが見つからないことが原因であることを
判断することができません。

そこでmanage.dllが参照しているdllの一覧のようなものを取得したいのですが
そのようなAPIは用意されていますでしょうか?
とりあえずdumpbin.exe /dependentsで出力される結果を用いる実装にしてみたのですが
やはり外部の実行ファイルに頼るのは微妙ですし、何より処理が重いです。
Win32APIでも構いませんので何か方法がありましたらご教示ください。
864デフォルトの名無しさん:2014/09/26(金) 08:53:54.22 ID:CyXmGE6J
開発フェーズなら
Dependency Walkerというツールがある
865デフォルトの名無しさん:2014/09/26(金) 08:56:02.56 ID:CyXmGE6J
あっ、実行時の話やったか
866デフォルトの名無しさん:2014/09/26(金) 11:29:52.72 ID:b/KTTsk9
FileNotFoundが発生したら
LoadLibrary("native.dll");
GetLastError()してみるってのはどうだろう?
867デフォルトの名無しさん:2014/10/08(水) 22:50:24.63 ID:T8t7hCtu
>>863です。亀ですいませんが解決しました。
APIとしてはImageHelpライブラリ辺りが使えそうだったのですが途中で詰まったので断念。
結局ファイルを直接読んでIMAGE_DOS_HEADER構造体からたどって
インポートデータを解析することで対応できました。
868デフォルトの名無しさん:2014/10/16(木) 17:29:15.53 ID:sLOsTk32
質問です。

MyCapsuleというクラスを作り、GetValue(参照渡しの引数)という仮想関数を
定義しておいて、このクラスから派生したクラスにおいてGetValueの実装を
するにあたり、与えられた引数がある特定の型に合致する場合にはその
引数に値を代入する、という動作を実現したいです。
例えばMyCapsuleIntという派生クラスでは、与えられた引数がint型だった
場合に、その引数に(何かしらメンバ変数などの)値を代入する、というもの
です。

こういう場合アンマネージドだと引数としてvoid*を使った実装になるかと
思うんですが、マネージドだとかわりにObject^%とかを使うのでしょうか?
ハンドルと追跡参照について未だによくわかっていないので、どうするのが
一般的な作法なのか見当がつきません。

よろしくお願いします。
869デフォルトの名無しさん:2014/10/16(木) 21:50:11.68 ID:rtOJWyFH
アンマネージからマネージの関数を呼んで
マネージドの中で作ったString^からchar*でアンマネージドに戻したいんです。

調べたところ
(char*)Marshal::StringToHGlobalAnsi("").ToPointer()ってのがあるのはわかったのです。
けど、これで戻してもアンマネージドからMarshal::FreeHGlobalが呼べないのでだめです。
一度StringToHGlobalAnsiで得たchar*を別のcharのバッファにコピーしてから戻す方法になるのでしょうか。
870デフォルトの名無しさん:2014/10/16(木) 22:01:55.82 ID:rvFRd6iT
Marshal::StringToHGlobal...で確保したのはLocalFreeすればいいけど
普通は呼び出し元がバッファ用意するなり、std::stringとかで返すなりするもんじゃね?
871デフォルトの名無しさん:2014/10/16(木) 22:15:47.60 ID:rtOJWyFH
なるほどLocalFreeで解放できるんですね。

>普通は呼び出し元がバッファ用意するなり
固定長になるからどうしようかなとおもって。

>std::string
StringToHGlobalAnsiで領域を確保したのに
stringでまた確保されるのがあれかなとか。
気にしすぎといわれればそうです・・・。
872デフォルトの名無しさん:2014/10/22(水) 13:38:11.49 ID:5iN2ViqP
C#みたいにCodeProviderで動的な文字列をコンパイルして
実行したいんですが、もしかしたらC++/CLIにはCodeProviderが
ないですか?
873デフォルトの名無しさん:2014/10/22(水) 13:43:12.37 ID:5iN2ViqP
すいません、抜けてました。
CodeDomProviderを継承したCodeProviderです
874デフォルトの名無しさん:2014/10/22(水) 17:09:24.88 ID:LQAHBIBx
まあ混合型使えないからそれならC#でいいじゃんってことになるよね
875デフォルトの名無しさん:2014/11/11(火) 14:49:35.49 ID:TsA8Knwq
C++のDLLをC#で使いたいのですがC++/CLIで吸収したほうがいいのかな?
C++の構造体のポインタとかあってわけわからん
876デフォルトの名無しさん:2014/11/11(火) 16:50:02.76 ID:9mz6Ds+P
構造体 マーシャリング でぐぐる
877デフォルトの名無しさん:2014/11/12(水) 00:51:06.13 ID:pdAxKpZn
>>875の件ですがC#でなんとかがんばることにしました
お騒がせしました
878デフォルトの名無しさん:2014/12/09(火) 00:10:16.60 ID:xpR1dfQS
C++/cli からC++のプロジェクトに定義している関数を呼び出したいんです。
C++プロジェクトのlibファイルを参照し、includeファイルをC++/cli側に
持ってきて#include すると、物凄い数のエラーがでます。

(C2011:C2079:C2504:C3395:C3699)

[1] C++/cliで参照できるよう、ビルドエラーを直すしかない

[2] 別の方法でC++の関数をビルドエラー無しに参照できる。

ちなみに、C++の関数を参照しないで自作・・・という回答は無しでお願いします。
879デフォルトの名無しさん:2014/12/09(火) 10:30:47.48 ID:sSNN3cut
物凄い数のエラーじゃなくてエラーの内容をチェックしろよ。
出力ログのエラーの行をダブルクリックすると問題の箇所に飛ぶから
何で再定義だとか定義がないと言われてるか考えろ。
何でその程度のレベルでC++/CLIとかやってんの。
880デフォルトの名無しさん:2014/12/09(火) 10:30:56.11 ID:Xzj5HeMN
1.C++から呼び出す
2.C#に移植する

C++/CLIでなきゃいけない理由なんてあるん?
881878:2014/12/09(火) 22:59:58.60 ID:ygQnozb4
すみません、初心者なりにエラーの原因を調べてきました。

a.h

class A : public B {}

b.h

class B {}



この時に Class B で再定義のエラーが発生していました。

a.h が先にincludeされているのが問題で、b.h を先にinclude するよう
ソース修正すれば良い・・・・・ということでしょうか??
882デフォルトの名無しさん:2014/12/11(木) 12:05:51.55 ID:WAwY/dPu
各ヘッダの先頭に#pragma onceと書く

a.hでb.hのクラスを使っているなら
a.h内でb.hをinludeしろ

cpp側でincludeの順を気にしなきゃならないのは糞
883デフォルトの名無しさん:2014/12/15(月) 20:00:30.77 ID:JoQajMNA
質問です。

任意の型のマネージド配列(cli::array)を、任意の個数、引数として受け取って
それらの配列としての長さの最小値を返す関数を書きたいと思っています。

イメージとしては、
array<String^> arrStr = gcnew array<String^>(6);
array<double> arrDob = gcnew array<double>(8);
array<int> arrInt = gcnew array<int>(3);
に対して、
Function(arrStr, arrDob) = 6
Function(arrStr, arrDob, arrInt) = 3
となるような関数Functionを書きたいと思っています。

しかし、この場合の引数リストの書き方がよくわかりません。
単純に可変長の引数リストだとFunction(... array<Object^>^ args)みたいな
書き方になると思うんですが、これをマネージド配列の配列だからといって
真似てFunction(... array<array<Object^>^>^ arrays)みたいな書き方をしても、
呼び出し側(例えばFunction(arrStr, arrDob)とか)で引数リストが一致しない旨
怒られます。このような場合の適切な引数リストの書き方ってありますか?

あるいは回避策として、引数はFunction(... array<Object^>^ args)としておき
この関数の中でfor each (Object^ arg in args) { argが配列かどうか }という
コードもありえますが、肝心の、「argが配列かどうか」チェックするコードが
わかりません。

上記いずれか一方で構いませんので、解決策をご教授いただけないでしょうか?
884デフォルトの名無しさん:2014/12/15(月) 20:51:05.44 ID:oSpn/+/y
... System::Array^ args
arg->GetLength

... System::Collections::IList^ args
arg->Count
885デフォルトの名無しさん:2014/12/15(月) 22:28:59.04 ID:JoQajMNA
>>884
試してみましたが、いずれの場合もコンパイルエラーC3132が発生します。

そもそもargの型を何に指定すべきなのか、というのがよくわかっていません。
例えば
int Function(... array<Object^>^ args)
{
  int MinLength = -1;
  for each(array<Object^>^ arg in args)
  {
    if ( (MinLength < 0) || (arg->Length < MinLength) )
    {
      MinLength = arg->Length;
    }
  }
  return MinLength;
}
みたいなコードだと、Function(arrStr, arrInt)を実行したとき、それら引数を
System.Object[]にキャストできない旨を怒られます。
886デフォルトの名無しさん:2014/12/15(月) 22:43:22.71 ID:oSpn/+/y
ごめんちょっとぼけてた
... array<System::Array^>^ args
for each (auto arg in args) { int len = arg->GetLength(0);
... array<System::Collections::IList^>^ args
for each (auto arg in args) { int len = arg->Count;
887デフォルトの名無しさん:2014/12/15(月) 22:49:54.98 ID:JoQajMNA
>>886
うまくいきました!
動作版のコードは以下の通りです。

int Function(... array<System::Array^>^ args)
{
  int MinLength = -1;
  for each(auto arg in args)
  {
    if ( (MinLength < 0) || (arg->Length < MinLength) )
    {
      MinLength = arg->Length;
    }
  }
  return MinLength;
}

ありがとうございました。
888デフォルトの名無しさん:2015/01/28(水) 22:20:24.42 ID:NY0MHdg2
String^ selectedEmployee = (String^)(ComboBox1->SelectedItem);

^ってどういう意味?
889デフォルトの名無しさん:2015/01/28(水) 22:43:41.32 ID:CFOgWxxn
トラッキングハンドルって意味
890デフォルトの名無しさん:2015/01/30(金) 16:37:12.12 ID:w20faKmH
こらこら。
普通のC++でいうところの*だよ。
891デフォルトの名無しさん:2015/02/06(金) 15:15:17.14 ID:n2QXEqei
>>888
参照型
892デフォルトの名無しさん:2015/03/01(日) 13:44:47.71 ID:M4V41Yry
既存の native C++ のプログラムからC#で作ったdllを呼び出す必要が出てきたんで
/clrを使おうと思うんだが、.NETを使う部分以外のソースを全部 #pragma unmanaged
しておけば既存のnativeのプログラムと変わらないと考えていいのかな?
スタートアップに時間がかかったりするかもしれないけど。
それ以外になにか、純粋なnativeアプリと比べてのデメリットとかあるだろうか?
893デフォルトの名無しさん:2015/03/03(火) 10:49:44.41 ID:2xjYWNh7
拡張する度にpragma追加するのが面倒くさいからCOM exportのがいいよ
894デフォルトの名無しさん:2015/03/03(火) 22:20:54.28 ID:QODsipDB
COMはregistryやGACに登録しないとならないのが。それがなければ使いたいんだけど。
pragamはまぁ、stdafx.hに仕込んどけばいいだけなんで。
で、試してみて一応ビルドは通って、DependencyWalkerで見てもほとんど同じに見えたけど、
起動時に例外が出てうまく立ち上がらなかった。
どちらにしても、ビルドに時間がかかるんで普通にwrapperだけC++/CLIで作った方が楽そう。
895デフォルトの名無しさん:2015/03/03(火) 23:37:18.31 ID:TZKBUrl9
Register freeのcomが良い
896デフォルトの名無しさん
ほー、Interopでも登録なしで使えたのか。知らんかった。