Использование пространств имен в Powershell

Думал об этом, отвечая на этот вопрос.

Как вы можете избежать необходимости полностью квалифицировать каждый тип в пространстве имен?

Это действительно, очень утомительно писать System.Security.Cryptography.X509Certificates.X509Store вместо X509Store, или же [System.Security.Cryptography.X509Certificates.StoreName]::My вместо [StoreName]::My,

В C# у вас есть using директивы... как насчет Powershell?


РЕДАКТИРОВАТЬ 1 - Это работает для типов:

$ns = "System.Security.Cryptography.X509Certificates"
$store = New-Object "$ns.X509Store"(StoreName,StoreLocation)

New-Object принимает строковый литерал в качестве определения типа, поэтому он может быть построен программно.


РЕДАКТИРОВАТЬ 2 - Это работает для членов перечисления, используемых в качестве параметров:

$store = New-Object "$ns.X509Store"("My","LocalMachine")

Где "Мой" [System.Security.Cryptography.X509Certificates.StoreName]::My и "LocalMachine" является [System.Security.Cryptography.X509Certificates.StoreLocation]::LocalMachine,
Литеральные имена автоматически преобразуются в члены перечисления, если они размещены там, где ожидается член перечисления.

3 ответа

Решение

Для перечислений вам не нужно указывать полное имя типа. Например:

Вы можете сделать это:

New-Object System.DirectoryServices.ActiveDirectory.DirectoryContext([System.DirectoryServices.ActiveDirectory.DirectoryContextType]::Domain)

или гораздо более простая версия:

New-Object System.DirectoryServices.ActiveDirectory.DirectoryContext('Domain')

Вы можете использовать строки, чтобы идентифицировать перечисления, которые вы хотите использовать, не используя полностью оформленное имя. PowerShell обрабатывает преобразование типов, чтобы преобразовать строки в значения перечисления. Используя конкретный пример, который вы показали выше, это означает, что вы можете сделать это:

[System.Security.Cryptography.X509Certificates.OpenFlags]'ReadWrite'

И Powershell будет правильно конвертировать его (поэтому передача ReadWrite в параметр, который принимает значение перечисления OpenFlags, будет работать нормально). Если вы хотите передать несколько значений, вы можете сделать это следующим образом:

[System.Security.Cryptography.X509Certificates.OpenFlags]@('ReadWrite','IncludeArchived')

Обратите внимание, что я добавляю эти команды к имени типа, но если бы вы передавали их типизированному параметру, вы бы просто пропустили это.

Это должно сделать вас на один шаг ближе к возможности писать сценарии, которые работают с определенным пространством имен, без необходимости декорировать все имена.

Я знаю, уже немного поздно, но PowerShell v5 добавляет массу интересных языковых вещей. Одним из них является "использование пространства имен".

PS> using namespace System.Security.Cryptography.X509Certificates; [X509Store]


IsPublic IsSerial Name                                     BaseType                                     
-------- -------- ----                                     --------                                     
True     False    X509Store                                System.Object                                

Правильный путь?

$_m = [math]
$_m::sin((45*($_m::pi/180))) 

И это похоже на работу:

[Reflection.Assembly]::Load("System.Security, Version=2.0.0.0, Culture=Neutral, PublicKeyToken=b03f5f7f11d50a3a")

$stoopidnamespace = 'System.Security.Cryptography.X509Certificates.X509Store'
New-Object $stoopidnamespace($null)

Но это уродливо, когда вы делаете это:

$stoopidnamespace = 'System.Security.Cryptography.X509Certificates'
New-Object $stoopidnamespace'.X509Store'($null)
Другие вопросы по тегам