The part about filenames is correct, the rest is not. For traditional COM dlls, you can register multiple versions of them. The will reside in different physical locations, but an application requesting a specific version of a dll can get it easily. For the managed programming model: You can have every version of the .NET framework running alongside each other. This is also true for your own assemblies, you can have multiple versions of the same assembly in the GAC. And my point is: it is handled directly by the operating system, the developers don't have to do anything about it.
That way, multiple libraries can be on a single system, and the right one is selected for each program. However, if a program breaks on an update to a library that kept the same soname, you can force it to use the older library version by copying the old library back somewhere, renaming the program (say to the old name plus ``.orig''), and then create a small ``wrapper'' script that resets the library to use and calls the real (renamed) program. You could place the old library in its own special area, if you like, though the numbering conventions do permit multiple versions to live in the same directory. The wrapper script could look something like this:
Given this lengthy list, developers of C++ libraries in particular must plan for more than occasional updates that break binary compatibility. Fortunately, on Unix-like systems (including Linux) you can have multiple versions of a library loaded at the same time, so while there is some disk space loss, users can still run ``old'' programs needing old libraries.