C
Tutorial C: Operadores y expresiones
Descubre cómo usar operadores aritméticos y lógicos en C, desde adiciones hasta jerarquías, esenciales para cualquier programador.
Aprende C GRATIS y certifícateOperadores aritméticos
En C, las operaciones aritméticas se implementan utilizando operadores que permiten sumar, restar, multiplicar, dividir y obtener el resto de una división. Estos operadores pueden aplicarse tanto a variables enteras como a variables de punto flotante, siempre respetando las reglas de conversión entre tipos.
- El operador
+
se utiliza para sumar valores de cualquier tipo numérico. Por ejemplo, dos variables de tipoint
se suman y devuelven otroint
, mientras que si al menos uno de los operandos es de tipofloat
, el resultado será flotante. Un uso frecuente consiste en acumular resultados en un bucle u operar con contadores. - El operador
-
funciona de forma similar a+
, pero para restar valores numéricos. Además de su uso binario, también se utiliza de manera unaria para indicar un número negativo. Con frecuencia, se combina con las demás operaciones para expresar cálculos más complejos. - Para la multiplicación, se emplea el operador
*
, que puede utilizarse tanto con enteros como con flotantes. Resulta útil en aplicaciones que requieran múltiples productos intermedios, y es común verlo junto a operaciones de incremento para optimizar cálculos de índices u offset en memoria. - El operador
/
realiza la división. Cuando ambos operandos son de tipo entero, el resultado se trunca a un valor entero sin redondeo; por este motivo, suele ser necesaria la conversión afloat
si se desea un resultado con decimales. Por ejemplo,3 / 2
produce1
en variables de tipo entero, mientras que con3.0 / 2
se obtiene1.5
. - Para obtener el resto de una división entera, se emplea el operador
%
. Este operador devuelve el residuo de dividir un primer operando entre un segundo operando, y se aplica exclusivamente a tipos enteros. Suele utilizarse para verificar divisibilidad o para extraer cifras en operaciones numéricas específicas.
La siguiente demostración ilustra los operadores aritméticos en un bloque de código conciso:
#include <stdio.h>
int main(void) {
int x = 10, y = 3;
float z = 3.5;
printf("Suma: %d\n", x + y); // 13
printf("Resta: %d\n", x - y); // 7
printf("Multiplicación: %d\n", x * y); // 30
printf("División entera: %d\n", x / y); // 3
printf("División flotante con dos decimales: %.2f\n", x / z); // 2.86
printf("Módulo: %d\n", x % y); // 1
return 0;
}
En los ejemplos anteriores, la precisión de cada operación depende del tipo de dato que interviene. Por ello, conviene prestar atención a las conversiones implicadas para evitar pérdidas de información cuando se mezclan tipos enteros y flotantes.
Operadores lógicos y relacionales
Los operadores lógicos y relacionales en C permiten evaluar expresiones que comparan valores y determinan condiciones para la toma de decisiones. Mediante estos operadores, se genera un resultado de tipo entero donde 0
suele representar que la comparación no se cumple y un valor distinto de cero indica que la comparación es verdadera.
- El operador relacional
==
comprueba si dos expresiones son iguales, mientras que!=
indica si son distintas. Por otro lado,<
y>
permiten verificar si un valor es menor o mayor que otro respectivamente. Estas comparaciones se aplican a variables de tipo numérico y resultan esenciales para controles y validaciones con rangos concretos. - Los operadores lógicos
&&
(AND) y||
(OR) combinan varias condiciones para producir un resultado unificado. En particular,&&
devuelve1
si todas las expresiones son verdaderas, y||
se satisface si al menos una expresión es verdadera. Es habitual encontrar estos operadores anidados para reflejar múltiples condiciones en una sola sentencia.
A continuación se muestra un ejemplo que aplica estos operadores en diferentes comparaciones:
#include <stdio.h>
int main(void) {
int x = 5, y = 2;
printf("x == 5: %d\n", x == 5); // 1
printf("x != 5: %d\n", x != 5); // 0
printf("x > y: %d\n", x > y); // 1
printf("(x > 0 && y > 0): %d\n", (x > 0 && y > 0)); // 1
printf("(x < 0 || y < 0): %d\n", (x < 0 || y < 0)); // 0
return 0;
}
En el ejemplo, cada comparación mostrará un entero (0 o 1) según se cumpla o no la condición evaluada. Es recomendable revisar cómo influyen los tipos de datos en estas operaciones cuando se trabaja con enteros y valores de punto flotante para evitar resultados inesperados.
Operadores de incremento y decremento
Los operadores de incremento(++
) y decremento(--
) se utilizan para modificar el valor de una variable entera de forma compacta. Mediante ++
, se aumenta el valor en una unidad; con --
, se disminuye una unidad. Estos operadores son frecuentes al trabajar con contadores o en contextos donde se requiere actualizar una variable de manera repetitiva.
La ubicación del operador (prefijo o sufijo) define el momento en el que se aplica el cambio. Usar ++x
o --x
(prefijo) afecta la variable antes de que se evalúe la expresión completa. Por su parte, x++
o x--
(sufijo) emplea primero el valor original de la variable y luego realiza el incremento o decremento al finalizar la evaluación. Esta distinción influye en el valor devuelto por la expresión que contiene el operador, por lo que conviene prestar atención a cómo se encadena con otras instrucciones.
Es recomendable usar la notación que aporte mayor claridad en cada situación. Por ejemplo, ++x
puede hacer el código más explícito si se desea actualizar el valor antes de emplearlo en otra parte de la expresión, mientras que x++
resulta útil cuando se precisa la versión previa de la variable en un cálculo intermedio.
A continuación se muestra un ejemplo ilustrativo:
#include <stdio.h>
int main(void) {
int x = 5;
printf("Valor inicial de x: %d\n", x); // 5
printf("Resultado de x++: %d\n", x++); // 5
printf("Después de x++: %d\n", x); // 6
printf("Resultado de ++x: %d\n", ++x); // 7
printf("Después de ++x: %d\n", x); // 7
return 0;
}
En este ejemplo, se puede observar cómo la diferencia entre x++
y ++x
afecta tanto a la evaluación inmediata de la expresión como al nuevo valor de la variable. Un criterio habitual es mantener la consistencia en todo el programa para evitar confusiones y mejorar la legibilidad.
Jerarquía y asociatividad de operadores
En C, la jerarquía de operadores determina qué partes de una expresión se evalúan primero cuando se combinan varios operadores. Esta organización evita ambigüedades y hace que ciertas operaciones tengan prioridad sobre otras, aunque en muchos casos se recurre a los paréntesis para dejar clara la intención del programador.
La asociatividad describe el orden en el que se evaluarán los operadores de la misma precedencia. Por convención, la mayoría de los operadores binarios en C tienen asociatividad de izquierda a derecha, de modo que las expresiones se resuelven en esa dirección. Por ejemplo, si se combina el operador -
varias veces sin paréntesis, se evaluará de izquierda a derecha de forma secuencial.
Las categorías de precedencia agrupan a operadores como *
y /
por encima de +
y -
, lo que asegura que las multiplicaciones y divisiones se realicen antes en una misma expresión sin paréntesis. La evaluación también se aplica a otros operadores que, aunque no se repasan aquí de forma detallada, siguen las mismas reglas de jerarquía.
Una práctica frecuente al programar es utilizar paréntesis para dejar clara la ordenación de las operaciones cuando la expresión resulte compleja. Esto no solo reduce errores, sino que mejora la legibilidad. Por ejemplo, en una expresión como a + b * c
, C primero evalúa b * c
y luego realiza a + (resultado)
, pero se pueden añadir paréntesis para dejar explícito: (a + b) * c
o a + (b * c)
.
Cuando se emplean operadores unarios como ++x
o --x
en expresiones más amplias, puede ser útil recordar que su precedencia es superior a la de las operaciones aritméticas de suma y resta. Además, el uso de estos operadores en forma prefijo o sufijo puede alterar el valor inmediato que se emplea en el cálculo, por lo que conviene confirmar el orden de evaluación consultando la tabla oficial de precedencias.
Precedencia | Operador | Descripción | Asociatividad |
---|---|---|---|
1 | () |
Paréntesis | Izquierda a derecha |
1 | [] |
Subíndice | Izquierda a derecha |
1 | . |
Acceso a miembro | Izquierda a derecha |
1 | -> |
Acceso a miembro a través de puntero | Izquierda a derecha |
2 | ++ |
Incremento | Derecha a izquierda |
2 | -- |
Decremento | Derecha a izquierda |
2 | + |
Unario más | Derecha a izquierda |
2 | - |
Unario menos | Derecha a izquierda |
2 | ! |
Negación lógica | Derecha a izquierda |
2 | ~ |
Complemento bit a bit | Derecha a izquierda |
2 | * |
Desreferencia | Derecha a izquierda |
2 | & |
Dirección de | Derecha a izquierda |
2 | sizeof |
Tamaño de | Derecha a izquierda |
2 | (type) |
Conversión de tipo | Derecha a izquierda |
3 | * |
Multiplicación | Izquierda a derecha |
3 | / |
División | Izquierda a derecha |
3 | % |
Módulo | Izquierda a derecha |
4 | + |
Suma | Izquierda a derecha |
4 | - |
Resta | Izquierda a derecha |
5 | << |
Desplazamiento a la izquierda | Izquierda a derecha |
5 | >> |
Desplazamiento a la derecha | Izquierda a derecha |
6 | < |
Menor que | Izquierda a derecha |
6 | <= |
Menor o igual que | Izquierda a derecha |
6 | > |
Mayor que | Izquierda a derecha |
6 | >= |
Mayor o igual que | Izquierda a derecha |
7 | == |
Igual a | Izquierda a derecha |
7 | != |
Diferente de | Izquierda a derecha |
8 | & |
AND bit a bit | Izquierda a derecha |
9 | ^ |
XOR bit a bit | Izquierda a derecha |
10 | ` | ` | OR bit a bit |
11 | && |
AND lógico | Izquierda a derecha |
12 | ` | ` | |
13 | ?: |
Operador ternario | Derecha a izquierda |
14 | = |
Asignación | Derecha a izquierda |
14 | += |
Asignación suma | Derecha a izquierda |
14 | -= |
Asignación resta | Derecha a izquierda |
14 | *= |
Asignación multiplicación | Derecha a izquierda |
14 | /= |
Asignación división | Derecha a izquierda |
14 | %= |
Asignación módulo | Derecha a izquierda |
14 | <<= |
Asignación desplazamiento a la izquierda | Derecha a izquierda |
14 | >>= |
Asignación desplazamiento a la derecha | Derecha a izquierda |
14 | &= |
Asignación AND bit a bit | Derecha a izquierda |
14 | ^= |
Asignación XOR bit a bit | Derecha a izquierda |
14 | ` | =` | Asignación OR bit a bit |
15 | , |
Coma | Izquierda a derecha |
El siguiente fragmento muestra una expresión con distintos operadores donde la importancia de la jerarquía y la asociatividad se hace evidente:
#include <stdio.h>
int main(void) {
int x = 2, y = 3, z = 4;
int resultado = x + y * z - ++y;
printf("Resultado: %d\n", resultado); // 10
printf("Nuevo valor de y: %d\n", y); // 4
return 0;
}
En este ejemplo, el operador *
tiene mayor prioridad que +
o -
, y el operador ++
(en modo prefijo) se aplica a y
antes de evaluar el resto de la expresión. Como norma general, es recomendable revisar la lista de precedencia y asociatividad en la documentación oficial cuando surjan dudas, especialmente si se combinan muchos operadores en la misma instrucción.
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
Introducción Y Entorno
Sintaxis
Sintaxis De Selectores Y Propiedades
Selectores Básicos
Sintaxis De Selectores Y Propiedades
Herencia Y Cascada
Sintaxis De Selectores Y Propiedades
Pseudo-clases Y Pseudo-elementos
Sintaxis De Selectores Y Propiedades
Estilos De Fuente
Estilización De Texto Y Fondo
Propiedades De Texto
Estilización De Texto Y Fondo
Sombras En Texto Y Cajas
Estilización De Texto Y Fondo
Propiedades De Fondo
Estilización De Texto Y Fondo
Modelo De Caja
Modelo Caja Y Posicionamiento
Propiedades De Posicionamiento
Modelo Caja Y Posicionamiento
Propiedad 'Display'
Modelo Caja Y Posicionamiento
Elementos 'Float' Y 'Clear'
Modelo Caja Y Posicionamiento
Flexbox Para Crear Layouts Y Estructuras
Flexbox Y Grid
Css Grid Para Crear Layouts Y Estructuras
Flexbox Y Grid
Animaciones Y Transiciones
Técnicas Modernas Y Metodologías
Variables En Css
Técnicas Modernas Y Metodologías
Diseño Responsive Con Media Queries
Técnicas Modernas Y Metodologías
Metodologías De Escritura En Css
Técnicas Modernas Y Metodologías
Introducción A Javascript
Sintaxis
Tipos De Datos
Sintaxis
Variables
Sintaxis
Operadores
Sintaxis
Estructuras De Control
Sintaxis
Funciones
Sintaxis
Funciones Cierre (Closure)
Sintaxis
Funciones Flecha
Programación Funcional
Filtrado Con Filter() Y Find()
Programación Funcional
Transformación Con Map()
Programación Funcional
Reducción Con Reduce()
Programación Funcional
Clases Y Objetos
Programación Orientada A Objetos
Excepciones
Programación Orientada A Objetos
Encapsulación
Programación Orientada A Objetos
Herencia
Programación Orientada A Objetos
Polimorfismo
Programación Orientada A Objetos
Array
Estructuras De Datos
Conjuntos Con Set
Estructuras De Datos
Mapas Con Map
Estructuras De Datos
Manipulación Dom
Dom
Selección De Elementos Dom
Dom
Modificación De Elementos Dom
Dom
Eventos Del Dom
Dom
Callbacks
Programación Asíncrona
Promises
Programación Asíncrona
Async / Await
Programación Asíncrona
Introducción A Typescript
Introducción Y Entorno
Variables Y Constantes
Sintaxis
Operadores
Sintaxis
Control De Flujo
Sintaxis
Funciones
Sintaxis
Funciones Flecha
Sintaxis
Clases Y Objetos
Programación Orientada A Objetos
Interfaces
Programación Orientada A Objetos
Encapsulación
Programación Orientada A Objetos
Herencia
Programación Orientada A Objetos
Polimorfismo
Programación Orientada A Objetos
Inmutabilidad
Programación Funcional
Funciones Puras
Programación Funcional
Funciones De Primera Clase
Programación Funcional
Funciones De Alto Orden
Programación Funcional
Tipos Literales
Tipos De Datos Avanzados
Tipos Genéricos
Tipos De Datos Avanzados
Tipos De Unión E Intersección
Tipos De Datos Avanzados
Tipos De Utilidad
Tipos De Datos Avanzados
Módulos
Namespaces Y Módulos
Namespaces
Namespaces Y Módulos
Resolución De Módulos
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
Introducción Y Entorno
Instalación De Tailwind Css
Introducción Y Entorno
Fundamentos Del Sistema De Utility-first
Fundamentos
Fundamentos Del Diseño Responsive
Fundamentos
Tipografía Y Fuentes En Tailwind Css
Clases De Utilidad
Clases De Tamaño De Tailwind Css
Clases De Utilidad
Utilidades De Espaciado Y Alineación De Tailwind Css
Clases De Utilidad
Clases De Colores Y Fondo De Tailwind Css
Clases De Utilidad
Clases De Bordes De Tailwind Css
Clases De Utilidad
Hover, Focus Y Estado De Tailwind Css
Clases De Utilidad
Transiciones Y Animaciones De Tailwind Css
Clases De Utilidad
Contenedores Y Columnas En Tailwind Css
Layout
Flexbox En Tailwind Css
Layout
Grid En Tailwind Css
Layout
Evaluación Test Tailwind Css
Evaluación
Evaluación Código Tailwind Css
Evaluación
Introducción A React Y Su Ecosistema
Introducción Y Entorno
Instalar React Y Crear Nuevo Proyecto
Introducción Y Entorno
Introducción A Jsx
Componentes
Introducción A Componentes
Componentes
Componentes Funcionales
Componentes
Eventos En React
Componentes
Props Y Manejo De Datos Entre Componentes
Componentes
Renderizado Condicional
Componentes
Renderizado Iterativo Con Bucles
Componentes
Manejo De Clases Y Estilos
Componentes
Introducción A Los Hooks
Hooks
Estado Y Ciclo De Vida De Los Componentes
Hooks
Hooks Estado Y Efectos Secundarios
Hooks
Hooks Para Gestión De Estado Complejo Y Contexto
Hooks
Hooks Optimización Y Concurrencia
Hooks
Introducción A React Router
Navegación Y Enrutamiento
Definición Y Manejo De Rutas
Navegación Y Enrutamiento
Rutas Anidadas Y Rutas Dinámicas
Navegación Y Enrutamiento
Navegación Programática Redirección
Navegación Y Enrutamiento
Nuevos Métodos Create De React Router
Navegación Y Enrutamiento
Solicitudes Http Con Fetch Api
Interacción Http Con Backend
Solicitudes Http Con Axios
Interacción Http Con Backend
Estado Local Con Usestate Y Usereducer
Servicios Y Gestión De Estado
Estado Global Con Context Api
Servicios Y Gestión De Estado
Estado Global Con Redux Toolkit
Servicios Y Gestión De Estado
Custom Hooks Para Servicios Compartidos
Servicios Y Gestión De Estado
Evaluación Test React
Evaluación
Aprendizaje Automático
Introducción Y Entorno
Introducción E Instalación
Introducción Y Entorno
Introducción Al Preprocesamiento De Datos
Preprocesamiento De Datos
Identificación Y Tratamiento De Valores Faltantes
Preprocesamiento De Datos
Escalado De Datos
Preprocesamiento De Datos
Normalización De Datos
Preprocesamiento De Datos
Codificación De Variables Categóricas
Preprocesamiento De Datos
Ingeniería De Características
Preprocesamiento De Datos
Selección De Características
Preprocesamiento De Datos
Extracción De Características
Preprocesamiento De Datos
Particionamiento De Datos
Preprocesamiento De Datos
Preprocesamiento De Datos Desbalanceados
Preprocesamiento De Datos
Introducción A La Regresión
Regresión
Regresión Lineal
Regresión
Regresión Knn Kneighborsregressor
Regresión
Regresión Svm Con Svr
Regresión
Regresión Con Árboles Decisiontreeregressor
Regresión
Regresión Con Algoritmos De Conjunto
Regresión
Introducción A La Clasificación
Clasificación
Clasificación Con Regresión Logística
Clasificación
Clasificación Knn Kneighborsclassifier
Clasificación
Clasificación Svm Con Svc
Clasificación
Clasificación Con Árboles Decisiontreeclassifier
Clasificación
Clasificación Con Algoritmos De Conjunto
Clasificación
Reducción De La Dimensionalidad Con Pca
Aprendizaje No Supervisado
Clustering Con Kmeans
Aprendizaje No Supervisado
Clustering Jerárquico
Aprendizaje No Supervisado
Clustering De Densidad Con Dbscan
Aprendizaje No Supervisado
Preprocesamiento De Textos Para Nlp
Nlp
Representación De Texto Y Extracción De Características
Nlp
Clasificación De Texto Con Scikit Learn
Nlp
Análisis De Sentimiento
Nlp
Técnicas Avanzadas De Extracción De Características
Nlp
Introducción Al Análisis De Series Temporales
Series Temporales
Preprocesamiento De Datos De Series Temporales
Series Temporales
Ingeniería De Características Para Series Temporales
Series Temporales
Transformación Y Escalado De Series Temporales
Series Temporales
Validación Y Evaluación De Modelos En Series Temporales
Series Temporales
Validación Y Evaluación De Modelos
Validación De Modelos
Técnicas De Validación Cruzada
Validación De Modelos
Métricas De Regresión
Validación De Modelos
Métricas De Clasificación
Validación De Modelos
Ajuste De Hiperparámetros
Validación De Modelos
Introducción A Pipelines
Pipelines Y Despliegue
Creación De Pipelines Básicos
Pipelines Y Despliegue
Preprocesamiento De Datos Con Pipelines
Pipelines Y Despliegue
Pipelines Y Validación Cruzada
Pipelines Y Despliegue
Pipelines Con Columntransformer
Pipelines Y Despliegue
Exportar E Importar Pipelines
Pipelines Y Despliegue
Introducción E Instalación De Opencv
Introducción Y Entorno
Carga Y Visualización De Imágenes
Manipulación Imágenes
Operaciones Básicas En Imágenes
Manipulación Imágenes
Detección De Bordes Y Contornos
Procesamiento Y Análisis
Histograma Y Ecualización
Procesamiento Y Análisis
Preprocesamiento Para Machine Learning
Aprendizaje Automático
Clasificación De Imágenes Con Ml
Aprendizaje Automático
Introducción A Docker
Introducción Y Entorno Docker
Instalación De Docker
Introducción Y Entorno Docker
Descargar Imágenes De Hub.docker.com
Imágenes Docker
Crear Imágenes Con Dockerfile
Imágenes Docker
Contenedores Docker
Contenedores Docker
Volúmenes Docker
Volúmenes Docker
Redes Docker
Redes Docker
Creación De Archivos Docker Compose
Docker Compose
Docker Compose Para Varios Servicios
Docker Compose
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.
Control de versiones con Git
5h 0m
Programación con HTML
10h 0m
Programación con CSS
10h 0m
Programación con JavaScript
20h 0m
Programación con TypeScript
20h 0m
Programación con Python
20h 0m
Programación con Java
20h 0m
Programación con Kotlin
20h 0m
Programación con C#
20h 0m
Programación con Go
20h 0m
Consola de comandos Linux Bash
15h 0m
Bases de datos con SQL
20h 0m
Frontend con Bootstrap CSS
10h 0m
Frontend con Tailwind CSS
10h 0m
Frontend con Angular
30h 0m
Frontend con React
30h 0m
Frontend con Vuejs
30h 0m
Frontend básico: HTML, CSS y JavaScript
40h 0m
Frontend avanzado: HTML, CSS, JS, TS, Angular
90h 0m
Backend NestJS
30h 0m
Backend Spring Boot
30h 0m
Backend Hibernate ORM
30h 0m
Backend: Java y Spring Boot
50h 0m
Backend: TypeScript y NestJS
50h 0m
Full Stack: Spring Boot y Angular
120h 0m
Full Stack: NestJS y Angular
100h 0m
Ciencia de datos con Numpy con Python
15h 0m
Ciencia de datos con Pandas con Python
20h 0m
Análisis de datos y visualización con Matplotlib
15h 0m
Análisis de datos y visualización con Seaborn
10h 0m
Docker y Docker Compose
10h 0m
Objetivos de aprendizaje de esta lección
- Comprender el uso de operadores aritméticos en C.
- Aplicar operativos lógicos y relacionales en expresiones.
- Diferenciar entre prefijo y sufijo en incrementos y decrementos.
- Identificar jerarquía y asociatividad de operadores.