I suspect the reason is that while substr() can just copy the data, whereas unpack() must recode, and check for more strange conditions. Also, try a (rather then A), Z, and z.
Warning: Unless otherwise stated, code is untested. Do not use without understanding. Code is posted in the hopes it is useful, but without warranty. All copyrights are relinquished into the public domain unless otherwise stated. I am not an angel. I am capable of error, and err on a fairly regular basis. If I made a mistake, please let me know (such as by replying to this node).