質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.50%
C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

Q&A

解決済

2回答

1108閲覧

Cドライブでないパスは8.3形式に変換できますか?

j_t

総合スコア13

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

0グッド

0クリップ

投稿2019/06/18 00:58

編集2019/06/19 04:42

使用言語:C#
フレームワーク:.Net Framework 4.6.1

###8.3形式が必要な理由
InstallShield製のSetUp.exeの実行時にインストールフォルダを引数で指定するとき、フォルダパスに空白が入るとエラーが発生するため、空白がない形で渡したい。(エスケープすれば、パスに空白が含まれていても大丈夫でした。2019/06/19)

###質問本文
パス文字列を8.3形式に変換するためにGetShortPathNameを使っているのですが、引数のlongPathに渡すのがCドライブ以外のドライブのパス(例えばD:\tera tail\など)だと8.3形式になりません。

そもそも8.3形式はCドライブにのみ有効なのでしょうか?
もしも、Cドライブ以外のパスを変換できる方法があれば教えていただけないでしょうか?

以下はGetShortPathNameを使用できるように呼び出しているコードです。

C#

1 /// <summary> 2 /// Win32Api呼び出し kernel32.dll 3 /// </summary> 4 internal class Kernel32 5 { 6 [SuppressUnmanagedCodeSecurity] 7 internal static class NativeMethods 8 { 9 [DllImport("kernel32.dll", CharSet = CharSet.Unicode)] 10 public static extern int GetShortPathName(string longPath, StringBuilder shortPathBuffer, int bufferSize); 11 } 12 }

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

Zuishin

2019/06/18 01:26

そもそも 8.3 がまだ必要ですか? 何に使うのでしょう?
Zuishin

2019/06/18 01:39

なぜか質問編集によって消されましたが、変換できるかどうかは GetShortPathName の戻り値と GetLastError によってわかります。長いファイル名と短いファイル名は NTFS 独自のものなので、その他のファイルシステムには使えません。NTFS でもレジストリを書き換えることで無効にできます。
j_t

2019/06/18 01:42

8.3が必要な理由を追記しました。 ご指摘ありがとうございます。
Zuishin

2019/06/18 01:44

了解しました。InstallShield は作りが良くないので昔から嫌いでしたが、まだそんな仕様なんですね。
j_t

2019/06/19 04:38

今までエラーが出ていたのはエスケープの方法が悪かったせいで、空白が入っていたら必ず8.3が必要なわけではありませんでした。
guest

回答2

0

ベストアンサー

そもそも8.3形式はCドライブにのみ有効なのでしょうか?

Cドライブかどうかに限らず、8.3形式のファイル名を生成しないフォーマットあるいは設定も存在しますので、そのような場合にはGetShortPathNameは有効な値を返しません。

投稿2019/06/18 01:00

maisumakun

総合スコア145121

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

j_t

2019/06/18 01:15

ご回答ありがとうございます。 設定があるとは初めて知りました。
guest

0

こんにちは。

fsutil 8dot3nameコマンドで8.3形式のファイル名を設定できます。
状況的に、「3 - システム ボリュームを除くすべてのボリュームで 8dot3 名の作成を無効にする」が設定されているようです。
デフォルトは「2 - 8dot3 名の作成をボリューム単位で設定する」のようです。
記憶では3がデフォルトだった時があったと思うのですが、手元のWindows 10は2でした。私は設定していないのでデフォルトの筈です。(PCショップが設定変更していない限り)

InstallShield製のSetUp.exeの実行時にインストールフォルダを引数で指定するとき、フォルダパスに空白が入るとエラーが発生するため、空白がない形で渡したい。

""で括って対応できないでしょうか?
既に試されてうまく行っていない場合、エスケープが必要なのかもしれません。
適切なエスケープに超苦労する場合もあります。

外部コマンドを呼び出す時、内部ではcmd.exe /Cで呼び出しているケースも多いのですが、「空白がなく、()がある」パスで以前ハマったことがあります。
インストーラから"C:\Program Files(x86)\foo(1)\baz.bat"は起動できるのですが、
"C:\Users\user_name\AppData\Local\foo(1)\baz.bat"は起動できないのです。
"C:\Users\user_name\AppData\Local\foo"を起動できないと叱られるのです。

最終的に"\""+BatchPath+"\""のようにエスケープして両方のケースの起動に成功しました。
(この辺のエスケープの仕方はインストーラ開発ツール毎に異なると思います。)

投稿2019/06/18 02:15

Chironian

総合スコア23272

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

j_t

2019/06/18 04:33

ご回答ありがとうございます。 エスケープは試してみたのですが、うまくいきませんでした。InstallSieldだと違うようです。
j_t

2019/06/19 04:18

エスケープ文字もエスケープするようにしたら上手く行きました。 アドバイスありがとうございます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.50%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問