It’s useful for interop with unmanaged code. Any pointers passed to unmanaged functions need to be fixed (aka. pinned) to prevent the garbage collector from relocating the underlying memory.
If you are using P/Invoke, then the default marshaller will pin objects for you. Sometimes it’s necessary to perform custom marshalling, and sometimes it’s necessary to pin an object for longer than the duration of a single P/Invoke call.