If you aren't careful, what you are proposing is a security hole in the making.  So be very cafeful - especially when trusting network input.
You didn't specify TCP or UDP, so I'll just give you some general guidance.
For TCP or UDP, just allocate a single buffer of size N, where N is the biggest message size you could possibly have sent from the remote player or server.  For UDP, I'd recommend keeping this under 1500 bytes.  For TCP, you could have larger sizes.
For your socket, whether UDP or TCP, make it non-blocking.  This is so you don't hang the game loop while waiting for data.
Add a size and crc-hash to the header of any message.  When you receive the messsage as a UDP packet, if the size in the packet header is bigger than N (which means you've already got a truncated packet) or the hash doesn't match what you compute on the receive size, just reject the packet.  I call this out because without additional integrity checks, hackers and cheaters will exploit your packet structure to win.
For TCP, you essentially frame your messages as the way you describe.  Each message has a header dictating the number of bytes that are to follow.  But do the same stuff as I call out for UDP - add a header of your own for integrity checking.  Close the socket if you ever get message corruption and ASSERT.
In TCP, because of TCP segmentation, you may not receive the entire message in the same "recv" call.  That is, if you only received a partial message, store it in  temporary buffer.  On a subsequent call to recv (on the next game frame), append to this buffer.  You'll have to keep variables to track the expected end of the message so you don't accidentally read into the next message that may have been sent.
Good luck.