File descriptors are basically a system's way of keeping track of all the open files (and some other types of I/O channels) within a running program. They are represented as integers, and you can think of them as a kind of "ID" or "handle" that your program uses to work with files.
Let's consider a real-life analogy. Imagine you're a librarian, and every time someone checks out a book, you give them a ticket with a unique number. Later, when they want to return the book or ask about it, they show you the ticket, and you can immediately know which book they're talking about. In this analogy, the file descriptors are the numbers on the tickets.
When you open a file in a program, the operating system gives you a file descriptor – the smallest unused number – that you can use to refer to that file in the future. When you read from or write to the file, you use that file descriptor. When you're done with the file, you "close" it, which tells the system you're done with that file descriptor, and it can use that number for something else.
The exact limit of fd depends on your system, but it's typically quite high (thousands or even tens of thousands). Each open file, pipe, or other resource consumes one file descriptor.
Here are some standard file descriptor values:
0: Standard input (stdin)
1: Standard output (stdout)
2: Standard error (stderr)
These are the default file descriptors that are available in every process. When you open a file using the open function, you get a file descriptor that is always the lowest-numbered file descriptor not currently open for the process. This is usually 3 for the first file opened, 4 for the second, and so on.
A file descriptor remains open until it is explicitly closed, or until the process terminates. It's important to always close file descriptors when you're done with them to prevent a file descriptor leak, which can cause a process to exhaust the number of file descriptors available to it.
Example:
write(1, "a", 1); //First argument represent the Standard Output.
write(4, "a", 1); //4 is not currently open or valid, then the write function will fail, and it will return -1 to indicate an error.
int fd;
fd = open("42", O_RDWR | O_CREAT | O_APPEND, 0777); //Create file "42"
printf("%d\n", fd); // => fd value is 3 because it takes the next smallest int available (0, 1 and 2) are always taken.
fd can go up to thousands depending on your system.