C

Tutorial C: Funciones

C: Aprende sobre prototipos y definición de funciones en C. Mejora la gestión del código y evita inconsistencias con este tutorial detallado.

Aprende C GRATIS y certifícate

Prototipos y definición de funciones

Un prototipo de función especifica su nombre, tipo de retorno y parámetros, permitiendo al compilador conocer de antemano la firma de la función antes de que se encuentre su implementación en el código. Este enfoque evita advertencias sobre declaraciones implícitas y facilita la organización de archivos fuente cuando se trabaja en proyectos con múltiples módulos.

La definición de la función es el bloque de código que realmente ejecuta las instrucciones. Incluye el mismo tipo de retorno y los tipos de los parámetros declarados en el prototipo. En esta sección se ubican las sentencias que procesan los datos y devuelven un resultado. Mantener el prototipo igual a la definición garantiza la correcta vinculación durante la fase de enlazado.

Es habitual colocar los prototipos al inicio del fichero o en un archivo de encabezado. El compilador procesa primero estas declaraciones, asegurando que cuando el programa invoque la función, ya exista información sobre su firma y su tipo de retorno.

A continuación, se muestra un ejemplo sencillo que ilustra el uso de un prototipo y su definición:

#include <stdio.h>

/* Prototipo */
int sumar(int x, int y);

int main(void) {
    int resultado = sumar(10, 5);
    printf("La suma es: %d\n", resultado);
    return 0;
}

/* Definición */
int sumar(int x, int y) {
    return x + y;
}

En este ejemplo, el prototipo de la función sumar declara que la función recibe dos enteros y devuelve un entero. Posteriormente, la definición desarrolla el algoritmo que realiza la suma y devuelve el resultado. Usar prototipos claros evita inconsistencias entre llamada y definición, especialmente en proyectos de gran tamaño.

Paso de argumentos por valor

En C, todos los argumentos se transmiten a una función mediante paso de valores, lo que implica que la función recibe una copia de cada parámetro. Dichas copias se almacenan en áreas de memoria independientes dentro de la función, de modo que cualquier modificación de los argumentos no altera el contenido original en el programa que llamó a la función.

Un aspecto significativo de este mecanismo es que se evita la alteración accidental de variables externas. El compilador reserva espacio para los parámetros en la pila (stack) en el momento de la llamada y, al finalizar la función, esa memoria local se libera. Dicho flujo es especialmente ventajoso en situaciones donde se busca aislar comportamientos y proteger datos críticos.

A continuación, se muestra un ejemplo que ilustra el paso de argumentos por valor:

#include <stdio.h>

void incrementar(int numero) {
    numero++;
    printf("Dentro de la función: %d\n", numero);
}

int main(void) {
    int valor = 5;
    incrementar(valor);
    printf("Fuera de la función: %d\n", valor);
    return 0;
}

En este ejemplo, la función incrementar opera sobre una copia de la variable valor. Tras llamar a incrementar, el valor original permanece inalterado en el ámbito de main, demostrando así que las modificaciones efectuadas en la función no repercuten fuera de ella. Esta característica de C permite que cada función trabaje sobre datos temporales, manteniendo la coherencia y reduciendo efectos colaterales.

Tipos de retorno y ámbito (scope) de variables

Las funciones en C pueden devolver diferentes tipos de datos, como int, float o incluso void cuando no se requiere un resultado. El tipo de retorno debe declararse al inicio de la función, indicando claramente qué valor se devuelve. En caso de que la función retorne un objeto de un tipo compuesto, suele ser recomendable emplear estructuras o, si es necesario, punteros.

Para controlar el ámbito de una variable, es clave entender que en C existen diferentes niveles de visibilidad. Una variable declarada dentro de una función pertenece al ámbito local de dicha función; esto significa que solo puede usarse durante la ejecución de esa función y deja de existir al salir de ella. Por el contrario, una variable global, declarada fuera de cualquier función, puede emplearse en todo el fichero, y si se utiliza la palabra clave extern en otros ficheros, su alcance se extiende a distintos módulos.

El almacenamiento de las variables locales se reserva en la pila, mientras que las variables globales suelen residir en la memoria estática. Cuando una variable se declara con la palabra clave static dentro de una función, mantiene su valor entre llamadas sucesivas, aunque continúe siendo inaccesible desde fuera de la función. Este comportamiento se aprovecha en situaciones donde es útil conservar el estado interno sin exponerlo de manera global.

En el siguiente ejemplo, se ilustran distintos tipos de retorno y ámbitos de variables:

#include <stdio.h>

/* Variable global */
static int contadorGlobal = 0;

/* Función que retorna un valor entero */
int incrementarGlobal(void) {
    contadorGlobal++;
    return contadorGlobal;
}

/* Función sin retorno que utiliza ámbito local */
void imprimirLocal(void) {
    int valorLocal = 100;
    printf("Valor local dentro de la función: %d\n", valorLocal);
}

int main(void) {
    printf("Contador global: %d\n", incrementarGlobal());
    imprimirLocal();
    return 0;
}

En este código, se observa que contadorGlobal conserva su valor a lo largo de múltiples llamadas debido a su almacenamiento estático global, mientras que valorLocal solo existe dentro de la función imprimirLocal y se destruye al finalizar dicha función. El retorno de la función incrementarGlobal es de tipo int, por lo que se especifica con claridad el valor devuelto en la instrucción return.

Recursividad

La recursividad en C consiste en que una función se invoque a sí misma, ya sea de forma directa (llamándose de manera explícita) o indirecta (mediante otra función intermedia). Este mecanismo se basa en resolver un problema dividiéndolo en subproblemas de la misma naturaleza, lo cual puede simplificar la lógica en ciertos casos. Sin embargo, cada llamada recursiva consume espacio en la pila y, si las llamadas son muy profundas o no se definen condiciones base correctamente, se puede agotar la memoria disponible y producir errores en tiempo de ejecución.

Para implementar la recursividad, es esencial establecer un caso base que detenga la cadena de llamadas. Sin una condición clara de terminación, la función provocaría llamadas infinitas. Por otro lado, cada llamada reduce la complejidad del problema, avanzando paso a paso hacia el caso base y retornando los resultados parciales hasta reconstruir la solución completa.

A modo de ejemplo, se muestra una implementación recursiva del cálculo de la serie de Fibonacci, donde cada término depende de los dos anteriores:

#include <stdio.h>

int fibonacci(int n) {
    if (n <= 1) {
        return n;
    }
    return fibonacci(n - 1) + fibonacci(n - 2);
}

int main(void) {
    int resultado = fibonacci(6);
    printf("El término en la posición 6 es: %d\n", resultado);
    return 0;
}

En este ejemplo, la función fibonacci llama de nuevo a fibonacci con valores decrecientes hasta llegar a los casos n <= 1, retornando así la sucesión. Cada llamada se apila en la memoria hasta que se alcanza el caso base, momento en el que se van resolviendo las sumas pendientes.

La recursividad también puede darse de manera indirecta, donde la llamada recursiva no se produce de forma inmediata, sino a través de otra función que eventualmente retorna el control a la función inicial. Aunque menos frecuente, este patrón puede darse en algoritmos complejos que involucran pasos intermedios. En cualquier caso, es recomendable analizar la eficiencia y considerar alternativas no recursivas, sobre todo cuando el tamaño del problema pueda comprometer el espacio en la pila.

Aprende C GRATIS online

Todas las lecciones de C

Accede a todas las lecciones de C y aprende con ejemplos prácticos de código y ejercicios de programación con IDE web sin instalar nada.

Introducción A Css

CSS3

Introducción Y Entorno

Sintaxis

CSS3

Sintaxis De Selectores Y Propiedades

Selectores Básicos

CSS3

Sintaxis De Selectores Y Propiedades

Herencia Y Cascada

CSS3

Sintaxis De Selectores Y Propiedades

Pseudo-clases Y Pseudo-elementos

CSS3

Sintaxis De Selectores Y Propiedades

Estilos De Fuente

CSS3

Estilización De Texto Y Fondo

Propiedades De Texto

CSS3

Estilización De Texto Y Fondo

Sombras En Texto Y Cajas

CSS3

Estilización De Texto Y Fondo

Propiedades De Fondo

CSS3

Estilización De Texto Y Fondo

Modelo De Caja

CSS3

Modelo Caja Y Posicionamiento

Propiedades De Posicionamiento

CSS3

Modelo Caja Y Posicionamiento

Propiedad 'Display'

CSS3

Modelo Caja Y Posicionamiento

Elementos 'Float' Y 'Clear'

CSS3

Modelo Caja Y Posicionamiento

Flexbox Para Crear Layouts Y Estructuras

CSS3

Flexbox Y Grid

Css Grid Para Crear Layouts Y Estructuras

CSS3

Flexbox Y Grid

Animaciones Y Transiciones

CSS3

Técnicas Modernas Y Metodologías

Variables En Css

CSS3

Técnicas Modernas Y Metodologías

Diseño Responsive Con Media Queries

CSS3

Técnicas Modernas Y Metodologías

Metodologías De Escritura En Css

CSS3

Técnicas Modernas Y Metodologías

Introducción A Javascript

JavaScript

Sintaxis

Tipos De Datos

JavaScript

Sintaxis

Variables

JavaScript

Sintaxis

Operadores

JavaScript

Sintaxis

Estructuras De Control

JavaScript

Sintaxis

Funciones

JavaScript

Sintaxis

Funciones Cierre (Closure)

JavaScript

Sintaxis

Funciones Flecha

JavaScript

Programación Funcional

Filtrado Con Filter() Y Find()

JavaScript

Programación Funcional

Transformación Con Map()

JavaScript

Programación Funcional

Reducción Con Reduce()

JavaScript

Programación Funcional

Clases Y Objetos

JavaScript

Programación Orientada A Objetos

Excepciones

JavaScript

Programación Orientada A Objetos

Encapsulación

JavaScript

Programación Orientada A Objetos

Herencia

JavaScript

Programación Orientada A Objetos

Polimorfismo

JavaScript

Programación Orientada A Objetos

Array

JavaScript

Estructuras De Datos

Conjuntos Con Set

JavaScript

Estructuras De Datos

Mapas Con Map

JavaScript

Estructuras De Datos

Manipulación Dom

JavaScript

Dom

Selección De Elementos Dom

JavaScript

Dom

Modificación De Elementos Dom

JavaScript

Dom

Eventos Del Dom

JavaScript

Dom

Callbacks

JavaScript

Programación Asíncrona

Promises

JavaScript

Programación Asíncrona

Async / Await

JavaScript

Programación Asíncrona

Introducción A Typescript

TypeScript

Introducción Y Entorno

Variables Y Constantes

TypeScript

Sintaxis

Operadores

TypeScript

Sintaxis

Control De Flujo

TypeScript

Sintaxis

Funciones

TypeScript

Sintaxis

Funciones Flecha

TypeScript

Sintaxis

Clases Y Objetos

TypeScript

Programación Orientada A Objetos

Interfaces

TypeScript

Programación Orientada A Objetos

Encapsulación

TypeScript

Programación Orientada A Objetos

Herencia

TypeScript

Programación Orientada A Objetos

Polimorfismo

TypeScript

Programación Orientada A Objetos

Inmutabilidad

TypeScript

Programación Funcional

Funciones Puras

TypeScript

Programación Funcional

Funciones De Primera Clase

TypeScript

Programación Funcional

Funciones De Alto Orden

TypeScript

Programación Funcional

Tipos Literales

TypeScript

Tipos De Datos Avanzados

Tipos Genéricos

TypeScript

Tipos De Datos Avanzados

Tipos De Unión E Intersección

TypeScript

Tipos De Datos Avanzados

Tipos De Utilidad

TypeScript

Tipos De Datos Avanzados

Módulos

TypeScript

Namespaces Y Módulos

Namespaces

TypeScript

Namespaces Y Módulos

Resolución De Módulos

TypeScript

Namespaces Y Módulos

Introducción A C#

Sintaxis

Creación De Proyecto C#

Sintaxis

Variables Y Constantes

Sintaxis

Tipos De Datos

Sintaxis

Operadores

Sintaxis

Control De Flujo

Sintaxis

Funciones

Sintaxis

Clases Y Encapsulación

Programación Orientada A Objetos

Objetos

Programación Orientada A Objetos

Constructores Y Destructores

Programación Orientada A Objetos

Herencia

Programación Orientada A Objetos

Polimorfismo

Programación Orientada A Objetos

Excepciones

Excepciones

Arrays Y Listas

Colecciones Y Linq

Diccionarios

Colecciones Y Linq

Conjuntos, Colas Y Pilas

Colecciones Y Linq

Uso De Consultas Linq

Colecciones Y Linq

Delegados

Programación Asíncrona

Eventos

Programación Asíncrona

Lambdas

Programación Asíncrona

Uso De Async Y Await

Programación Asíncrona

Tareas

Programación Asíncrona

Introducción A C

Introducción Y Entorno

Primer Programa En C

Introducción Y Entorno

Estructura Básica De Un Programa En C

Sintaxis

Operadores Y Expresiones

Sintaxis

Control De Flujo

Sintaxis

Arrays Y Manejo De Cadenas

Sintaxis

Funciones

Funciones Y Punteros

Punteros

Funciones Y Punteros

Gestión De Memoria Dinámica

Funciones Y Punteros

Estructuras En C

Estructuras, Uniones Y Tipos

Uniones Y Enumeraciones

Estructuras, Uniones Y Tipos

Introducción A Tailwind Css

Tailwind CSS

Introducción Y Entorno

Instalación De Tailwind Css

Tailwind CSS

Introducción Y Entorno

Fundamentos Del Sistema De Utility-first

Tailwind CSS

Fundamentos

Fundamentos Del Diseño Responsive

Tailwind CSS

Fundamentos

Tipografía Y Fuentes En Tailwind Css

Tailwind CSS

Clases De Utilidad

Clases De Tamaño De Tailwind Css

Tailwind CSS

Clases De Utilidad

Utilidades De Espaciado Y Alineación De Tailwind Css

Tailwind CSS

Clases De Utilidad

Clases De Colores Y Fondo De Tailwind Css

Tailwind CSS

Clases De Utilidad

Clases De Bordes De Tailwind Css

Tailwind CSS

Clases De Utilidad

Hover, Focus Y Estado De Tailwind Css

Tailwind CSS

Clases De Utilidad

Transiciones Y Animaciones De Tailwind Css

Tailwind CSS

Clases De Utilidad

Contenedores Y Columnas En Tailwind Css

Tailwind CSS

Layout

Flexbox En Tailwind Css

Tailwind CSS

Layout

Grid En Tailwind Css

Tailwind CSS

Layout

Evaluación Test Tailwind Css

Tailwind CSS

Evaluación

Evaluación Código Tailwind Css

Tailwind CSS

Evaluación

Introducción A React Y Su Ecosistema

React

Introducción Y Entorno

Instalar React Y Crear Nuevo Proyecto

React

Introducción Y Entorno

Introducción A Jsx

React

Componentes

Introducción A Componentes

React

Componentes

Componentes Funcionales

React

Componentes

Eventos En React

React

Componentes

Props Y Manejo De Datos Entre Componentes

React

Componentes

Renderizado Condicional

React

Componentes

Renderizado Iterativo Con Bucles

React

Componentes

Manejo De Clases Y Estilos

React

Componentes

Introducción A Los Hooks

React

Hooks

Estado Y Ciclo De Vida De Los Componentes

React

Hooks

Hooks Estado Y Efectos Secundarios

React

Hooks

Hooks Para Gestión De Estado Complejo Y Contexto

React

Hooks

Hooks Optimización Y Concurrencia

React

Hooks

Introducción A React Router

React

Navegación Y Enrutamiento

Definición Y Manejo De Rutas

React

Navegación Y Enrutamiento

Rutas Anidadas Y Rutas Dinámicas

React

Navegación Y Enrutamiento

Navegación Programática Redirección

React

Navegación Y Enrutamiento

Nuevos Métodos Create De React Router

React

Navegación Y Enrutamiento

Solicitudes Http Con Fetch Api

React

Interacción Http Con Backend

Solicitudes Http Con Axios

React

Interacción Http Con Backend

Estado Local Con Usestate Y Usereducer

React

Servicios Y Gestión De Estado

Estado Global Con Context Api

React

Servicios Y Gestión De Estado

Estado Global Con Redux Toolkit

React

Servicios Y Gestión De Estado

Custom Hooks Para Servicios Compartidos

React

Servicios Y Gestión De Estado

Evaluación Test React

React

Evaluación

Aprendizaje Automático

scikit-learn

Introducción Y Entorno

Introducción E Instalación

scikit-learn

Introducción Y Entorno

Introducción Al Preprocesamiento De Datos

scikit-learn

Preprocesamiento De Datos

Identificación Y Tratamiento De Valores Faltantes

scikit-learn

Preprocesamiento De Datos

Escalado De Datos

scikit-learn

Preprocesamiento De Datos

Normalización De Datos

scikit-learn

Preprocesamiento De Datos

Codificación De Variables Categóricas

scikit-learn

Preprocesamiento De Datos

Ingeniería De Características

scikit-learn

Preprocesamiento De Datos

Selección De Características

scikit-learn

Preprocesamiento De Datos

Extracción De Características

scikit-learn

Preprocesamiento De Datos

Particionamiento De Datos

scikit-learn

Preprocesamiento De Datos

Preprocesamiento De Datos Desbalanceados

scikit-learn

Preprocesamiento De Datos

Introducción A La Regresión

scikit-learn

Regresión

Regresión Lineal

scikit-learn

Regresión

Regresión Knn Kneighborsregressor

scikit-learn

Regresión

Regresión Svm Con Svr

scikit-learn

Regresión

Regresión Con Árboles Decisiontreeregressor

scikit-learn

Regresión

Regresión Con Algoritmos De Conjunto

scikit-learn

Regresión

Introducción A La Clasificación

scikit-learn

Clasificación

Clasificación Con Regresión Logística

scikit-learn

Clasificación

Clasificación Knn Kneighborsclassifier

scikit-learn

Clasificación

Clasificación Svm Con Svc

scikit-learn

Clasificación

Clasificación Con Árboles Decisiontreeclassifier

scikit-learn

Clasificación

Clasificación Con Algoritmos De Conjunto

scikit-learn

Clasificación

Reducción De La Dimensionalidad Con Pca

scikit-learn

Aprendizaje No Supervisado

Clustering Con Kmeans

scikit-learn

Aprendizaje No Supervisado

Clustering Jerárquico

scikit-learn

Aprendizaje No Supervisado

Clustering De Densidad Con Dbscan

scikit-learn

Aprendizaje No Supervisado

Preprocesamiento De Textos Para Nlp

scikit-learn

Nlp

Representación De Texto Y Extracción De Características

scikit-learn

Nlp

Clasificación De Texto Con Scikit Learn

scikit-learn

Nlp

Análisis De Sentimiento

scikit-learn

Nlp

Técnicas Avanzadas De Extracción De Características

scikit-learn

Nlp

Introducción Al Análisis De Series Temporales

scikit-learn

Series Temporales

Preprocesamiento De Datos De Series Temporales

scikit-learn

Series Temporales

Ingeniería De Características Para Series Temporales

scikit-learn

Series Temporales

Transformación Y Escalado De Series Temporales

scikit-learn

Series Temporales

Validación Y Evaluación De Modelos En Series Temporales

scikit-learn

Series Temporales

Validación Y Evaluación De Modelos

scikit-learn

Validación De Modelos

Técnicas De Validación Cruzada

scikit-learn

Validación De Modelos

Métricas De Regresión

scikit-learn

Validación De Modelos

Métricas De Clasificación

scikit-learn

Validación De Modelos

Ajuste De Hiperparámetros

scikit-learn

Validación De Modelos

Introducción A Pipelines

scikit-learn

Pipelines Y Despliegue

Creación De Pipelines Básicos

scikit-learn

Pipelines Y Despliegue

Preprocesamiento De Datos Con Pipelines

scikit-learn

Pipelines Y Despliegue

Pipelines Y Validación Cruzada

scikit-learn

Pipelines Y Despliegue

Pipelines Con Columntransformer

scikit-learn

Pipelines Y Despliegue

Exportar E Importar Pipelines

scikit-learn

Pipelines Y Despliegue

Introducción E Instalación De Opencv

OpenCV

Introducción Y Entorno

Carga Y Visualización De Imágenes

OpenCV

Manipulación Imágenes

Operaciones Básicas En Imágenes

OpenCV

Manipulación Imágenes

Detección De Bordes Y Contornos

OpenCV

Procesamiento Y Análisis

Histograma Y Ecualización

OpenCV

Procesamiento Y Análisis

Preprocesamiento Para Machine Learning

OpenCV

Aprendizaje Automático

Clasificación De Imágenes Con Ml

OpenCV

Aprendizaje Automático

Introducción A Docker

Docker

Introducción Y Entorno Docker

Instalación De Docker

Docker

Introducción Y Entorno Docker

Descargar Imágenes De Hub.docker.com

Docker

Imágenes Docker

Crear Imágenes Con Dockerfile

Docker

Imágenes Docker

Contenedores Docker

Docker

Contenedores Docker

Volúmenes Docker

Docker

Volúmenes Docker

Redes Docker

Docker

Redes Docker

Creación De Archivos Docker Compose

Docker

Docker Compose

Docker Compose Para Varios Servicios

Docker

Docker Compose

Accede GRATIS a C y certifícate

Certificados de superación de C

Supera todos los ejercicios de programación del curso de C y obtén certificados de superación para mejorar tu currículum y tu empleabilidad.

Objetivos de aprendizaje de esta lección

  • Comprender la importancia de los prototipos en C.
  • Diferenciar entre prototipos y definiciones de funciones.
  • Entender el concepto de paso de argumentos por valor.
  • Identificar diferentes tipos de retorno.
  • Aprender sobre el ámbito de variables dentro y fuera de funciones.
  • Aplicar recursividad en funciones, entendiendo su uso y limitaciones.