папке - получить путь к файлу c#




Как я могу получить путь приложения в консольном приложении.NET? (18)

Как найти путь приложения в консольном приложении?

В Windows Forms я могу использовать Application.StartupPath для поиска текущего пути, но это, похоже, не доступно в консольном приложении.


Вероятно, немного поздно, но это стоит упомянуть:

Environment.GetCommandLineArgs()[0];

Или, вернее, получить только путь к каталогу:

System.IO.Path.GetDirectoryName(Environment.GetCommandLineArgs()[0]);

Редактировать:

Многие люди указали, что GetCommandLineArgs не гарантирует возврата имени программы. См . Первое слово в командной строке - это имя программы только по соглашению . В статье говорится, что «хотя очень немногие программы для Windows используют эту причуду (я сам не знаю)». Таким образом, можно «обманывать» GetCommandLineArgs , но мы говорим о консольном приложении. Консольные приложения обычно бывают быстрыми и грязными. Таким образом, это соответствует моей философии KISS.


Возможно, вы захотите сделать это:

System.IO.Path.GetDirectoryName(
    System.Reflection.Assembly.GetExecutingAssembly().GetName().CodeBase)

Вы можете использовать следующий код, вы получите полный путь приложения:

 class Program
{
    static void Main(string[] args)
    {
        string AppPath = System.Reflection.Assembly.GetExecutingAssembly().Location;
        WriteLine($"ApplicationPath ---{AppPath}");
        //OutPut// //ApplicationPath---C:\Users\TestUser\source\repos\ThreadStack\ThreadStack\bin\Debug\ThreadStack.exe
        ReadLine();
    }
}  

Вы можете использовать следующий код, чтобы получить текущий каталог приложения.

AppDomain.CurrentDomain.BaseDirectory

Вы можете создать имя папки как Ресурсы в проекте с помощью Обозревателя решений, затем вы можете вставить файл в Ресурсы.

private void Form1_Load(object sender, EventArgs e) {
    string appName = Environment.CurrentDirectory;
    int l = appName.Length;
    int h = appName.LastIndexOf("bin");
    string ll = appName.Remove(h);                
    string g = ll + "Resources\\sample.txt";
    System.Diagnostics.Process.Start(g);
}

Для всех, кто интересуется веб-приложениями asp.net. Вот мои результаты из 3 разных методов

protected void Application_Start(object sender, EventArgs e)
{
  string p1 = System.IO.Path.GetDirectoryName(System.Reflection.Assembly.GetExecutingAssembly().Location);
  string p2 = System.Web.Hosting.HostingEnvironment.ApplicationPhysicalPath;
  string p3 = this.Server.MapPath("");
  Console.WriteLine("p1 = " + p1);
  Console.WriteLine("p2 = " + p2);
  Console.WriteLine("p3 = " + p3);
}

результат

p1 = C:\Windows\Microsoft.NET\Framework64\v4.0.30319\Temporary ASP.NET Files\root\a897dd66\ec73ff95\assembly\dl3\ff65202d\29daade3_5e84cc01
p2 = C:\inetpub\SBSPortal_staging\
p3 = C:\inetpub\SBSPortal_staging

приложение физически работает от «C: \ inetpub \ SBSPortal_staging», поэтому первое решение, безусловно, не подходит для веб-приложений.


Если вы ищете совместимый с .NET Core способ, используйте

System.AppContext.BaseDirectory

Это было внедрено в .NET Framework 4.6 и .NET Core 1.0 (и .NET Standard 1.3). См .: Свойство AppContext.BaseDirectory .

Согласно этой странице ,

Это предпочтительная замена для AppDomain.CurrentDomain.BaseDirectory в .NET Core


Ни один из этих методов не работает в особых случаях, таких как использование символической ссылки на exe, они возвращают местоположение ссылки, а не фактическую exe.

Так можно использовать QueryFullProcessImageName чтобы обойти это:

using System;
using System.IO;
using System.Runtime.InteropServices;
using System.Text;
using System.Diagnostics;

internal static class NativeMethods
{
    [DllImport("kernel32.dll", SetLastError = true)]
    internal static extern bool QueryFullProcessImageName([In]IntPtr hProcess, [In]int dwFlags, [Out]StringBuilder lpExeName, ref int lpdwSize);

    [DllImport("kernel32.dll", SetLastError = true)]
    internal static extern IntPtr OpenProcess(
        UInt32 dwDesiredAccess,
        [MarshalAs(UnmanagedType.Bool)]
        Boolean bInheritHandle,
        Int32 dwProcessId
    );
}

public static class utils
{

    private const UInt32 PROCESS_QUERY_INFORMATION = 0x400;
    private const UInt32 PROCESS_VM_READ = 0x010;

    public static string getfolder()
    {
        Int32 pid = Process.GetCurrentProcess().Id;
        int capacity = 2000;
        StringBuilder sb = new StringBuilder(capacity);
        IntPtr proc;

        if ((proc = NativeMethods.OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, false, pid)) == IntPtr.Zero)
            return "";

        NativeMethods.QueryFullProcessImageName(proc, 0, sb, ref capacity);

        string fullPath = sb.ToString(0, capacity);

        return Path.GetDirectoryName(fullPath) + @"\";
    }
}

Попробуйте эту простую строку кода:

 string exePath = Path.GetDirectoryName( Application.ExecutablePath);

Следующая строка даст вам путь к приложению:

var applicationPath = Path.GetDirectoryName(Process.GetCurrentProcess().MainModule.FileName)

Выше решение работает правильно в следующих ситуациях:

  • простое приложение
  • в другом домене, где Assembly.GetEntryAssembly () вернет значение null
  • DLL загружается из встроенных ресурсов в виде массива байтов и загружается в AppDomain как Assembly.Load (byteArrayOfEmbeddedDll)

У вас есть два варианта поиска каталога приложения, который вы выбрали, будет зависеть от вашей цели.

// to get the location the assembly is executing from
//(not necessarily where the it normally resides on disk)
// in the case of the using shadow copies, for instance in NUnit tests, 
// this will be in a temp directory.
string path = System.Reflection.Assembly.GetExecutingAssembly().Location;

//To get the location the assembly normally resides on disk or the install directory
string path = System.Reflection.Assembly.GetExecutingAssembly().CodeBase;

//once you have the path you get the directory with:
var directory = System.IO.Path.GetDirectoryName(path);

Я имею в виду, почему бы не ap / invoke метод?

    using System;
    using System.IO;
    using System.Runtime.InteropServices;
    using System.Text;
    public class AppInfo
    {
            [DllImport("kernel32.dll", CharSet = CharSet.Auto, ExactSpelling = false)]
            private static extern int GetModuleFileName(HandleRef hModule, StringBuilder buffer, int length);
            private static HandleRef NullHandleRef = new HandleRef(null, IntPtr.Zero);
            public static string StartupPath
            {
                get
                {
                    StringBuilder stringBuilder = new StringBuilder(260);
                    GetModuleFileName(NullHandleRef, stringBuilder, stringBuilder.Capacity);
                    return Path.GetDirectoryName(stringBuilder.ToString());
                }
            }
    }

Вы бы использовали его так же, как Application.StartupPath:

    Console.WriteLine("The path to this executable is: " + AppInfo.StartupPath + "\\" + System.Diagnostics.Process.GetCurrentProcess().ProcessName + ".exe");

Я использую это, если exe предполагается вызывать двойным щелчком

var thisPath = System.IO.Directory.GetCurrentDirectory();

Я не видел, чтобы кто-то конвертировал LocalPath, предоставляемый отражением .Net Core, в полезный путь System.IO, вот моя версия.

public static string GetApplicationRoot()
{
   var exePath = new Uri(System.Reflection.
   Assembly.GetExecutingAssembly().CodeBase).LocalPath;

   return new FileInfo(exePath).DirectoryName;

}

Это вернет полный путь «C: \ xxx \ xxx» в том месте, где находится ваш код.


вы можете использовать это вместо этого.

System.Environment.CurrentDirectory

я использовал

System.AppDomain.CurrentDomain.BaseDirectory

когда я хочу найти путь относительно папки приложений. Это работает как для приложений ASP.Net, так и для winform. Он также не требует ссылки на сборки System.Web.


Assembly.GetEntryAssembly().Location или Assembly.GetExecutingAssembly().Location

Используйте в сочетании с System.IO.Path.GetDirectoryName() чтобы получить только каталог.

Пути из GetEntryAssembly() и GetExecutingAssembly() могут быть разными, хотя в большинстве случаев каталог будет таким же.

С помощью GetEntryAssembly() вы должны знать, что это может возвращать значение null если модуль ввода неуправляемый (т.е. исполняемый файл C ++ или VB6). В этих случаях можно использовать GetModuleFileName из Win32 API:

[DllImport("kernel32.dll", CharSet = CharSet.Auto)]
public static extern int GetModuleFileName(HandleRef hModule, StringBuilder buffer, int length);

AppDomain.CurrentDomain.BaseDirectory

Будет разрешена проблема для ссылки на файлы сторонних ссылок с установочными пакетами.





console-application