Perhaps you are confusing /dev/fd/63, which is simply a string whose characters spell out a path, with the file or pipe that provides the data you want to read. As a string, /dev/fd/63 is not the data. Neither as a path. In order to read the data cat really needs the file/pipe to exist while cat is reading.
In you first post, your first example: cat <(echo a)
For that bash creates a file, names it "63" located in /dev/fd, and gives cat a string "/dev/fd/63" to chew. Cat chews the string by opening the file that string refers to, etc.
Then your second example:
set -- <(echo a)
cat $1 => error file not found
For that bash creates a file ... and gives set -- the strings "/dev/fd/63" to chew. Set chews the string by storing it as positional parameter $1. At this point the line is fully executed, so bash destroy the file--not the string "/dev/fd/63" but the file.
On the next line, before starting cat, bash expands $1 to the string it is, "/dev/fd/63", then it gives cat that string to chew. Cat chews the string by opening the file that string refers to... But wait, there is no such file anymore because bash destroyed it one line ago. So cat reports an error: cat: /dev/fd/63: No such file or directory.
In your last post, your example:
function catw(){ cat "$1"; }
catw <(echo a)
bash rewrites it to
cat <(echo a)