To complement Bruce Payette’s helpful answer:
Not all functions are created equal in PowerShell:
-
An advanced function is the written-in-PowerShell analog of a (binary) cmdlet (which, as stated, is compiled from a .NET language); decorating a function’s
param(...)
block with the[CmdletBinding()]
attribute or decorating at least one parameter with a[Parameter()]
attribute thanks, Ansgar Wiechers
is what makes it an advanced one; as such, it supports certain standard behaviors:-
You gain automatic support for common parameters such as
-Verbose
, and-OutVariable
and, on an opt-in basis, for-WhatIf
and-Confirm
. -
Arguments that cannot be bound to explicitly declared parameters result in an error on invocation.
-
Typically, but not necessarily, advanced functions support one-by-one pipeline-input processing via a
process { ... }
script block, via parameter-binding parameters decorated withValueFromPipeline
and/orValueFromPipelineByPropertyName
. -
Unfortunately, even advanced functions and cmdlets aren’t created fully equal:
-
Advanced functions run in a child variable scope, unlike cmdlets.
- However, it is possible to gain access to the caller’s variables even in functions placed in modules (to functions outside of modules the caller’s variables are visible by default, owing to PowerShell’s dynamic scoping), via the
$PSCmdlet.SessionState.PSVariable
object, as shown in this answer.
- However, it is possible to gain access to the caller’s variables even in functions placed in modules (to functions outside of modules the caller’s variables are visible by default, owing to PowerShell’s dynamic scoping), via the
-
Advanced functions apply culture-invariant parameter conversions, unlike cmdlets.
-
Advanced functions, in Windows PowerShell, handle
ValueFromRemainingArguments
differently than cmdlets.- This inconsistency was resolved in PowerShell Core, but, unfortunately, in a manner that created more problems than it solved – see GitHub issues #5955 and #6451.
-
-
-
A simple function, by contrast:
- is appropriate for script- and module-internal helper functions
- requires less “ceremony” (simpler syntax without parameter attributes, single-script-block body)
- Note: Unlike advanced functions and cmdlets, simple functions accept arbitrary arguments, whether they bind to declared parameters or not; unbound ones are reflected in the automatic
$args
variable; see this answer for how to prevent passing unbound (unexpected) arguments.
- Note: Unlike advanced functions and cmdlets, simple functions accept arbitrary arguments, whether they bind to declared parameters or not; unbound ones are reflected in the automatic
- can, however, still process pipeline input via automatic variable
$Input
or even via aprocess { ... }
block, if desired. - also runs in a child scope by default; outside of modules (which simple function shouldn’t be exported from anyway), the caller’s variables are visible owing to PowerShell’s dynamic scoping; modifying them (which should generally be avoided) requires calls to
Set-Variable
with-Scope 1
. - Note that there’s also a specialized, but rarely used variant of a simple function optimized for pipeline processing, defined with the
Filter
keyword. Its body is implicitly invoked for each pipeline input object, reflected in automatic variable$_
.
While exporting functions as part of a module – preferably via its module manifest (*.psd1
) – doesn’t enforce that functions are advanced ones, it is good practice to only export advanced functions.