> (int *)&print_number
> Address of print_number in my PC is 0x401630
Theory:
0x401630 is the human readable representation of the value of the pointer to int (the result of the explicit conversion from the function pointer to the object pointer type). The meaning of such a conversion (if supported by the implementation) is implementation-defined.
The only requirement that the standard makes is that if this conversion is supported, then the conversion of the result in the reverse direction should yield the original value. ie. if this compiles cleanly, (int*)&print_number
then ( void(*)() )(int*)&print_number must yield &print_number
In practice:
0x401630 typically is the address in some portion of storage (with at least execute access) starting from where the code for the function resides. For historical reasons, in many implementations, this is a section (aka segment) of memory with the name .text or .TEXT
the actual location varies by OS. Its in the big pile of memory allocated to the process by the OS. That memory can be divided into the classes you mention. The stack typically has global variables, function parameters and variables and calling function data so it can pop back to where it was, its called a stack for a reason. Code binary is the code segment. The heap is the whole thing, really, all the process's memory combined if I remember correctly (rarely use the term). Youll have to poke at the stack if you do any in-line assembly language. Most of the rest of it is behind the scenes unless writing an OS or driver or other low level things.
Its been a long time, but if you want to know about this stuff, study assembly language. It explicitly taps the memory locations and you have to unravel what goes where.