19 KiB
Capítulo 13: El comando global
Hasta ahora has aprendido cómo repetir el último cambio con el comando del punto (.
), repetir acciones con macros (q
) y almacenar textos en los registros ("
).
En este capítulo, aprenderás cómo repetir un comando ejecutado en la línea de comandos de Vim con el comando global. Ejecútalo una vez, aplícalo en todas partes (dentro de un buffer).
Un vistazo al comando global
El comando global de Vim es utilizado para ejecutar un comando de la línea de comandos en múltiples líneas de manera simultánea.
Por cierto, puede que hayas leído anteriormente el témino "comandos Ex". En este libro, yo me refiero a ellos como comandos de la línea de comandos, pero ambos tanto los comandos Ex como los comandos de la línea de comandos son lo mismo. Hay comandos que comienzan con el símbolo de dos puntos (:
). En el capítulo anterior aprendiste sobre los comandos de sustitución. Fue un ejemplo de los comandos Ex. Son llamados Ex porque originalmente provenían del editor de texto Ex. Me seguiré refiriendo a ellos en este libro como comandos de la línea de comandos. Para obtener una lista completa de los comandos Ex, echa un vistazo en Vim a :h ex-cmd-index
.
El comando global tiene la siguiente sintaxis:
:g/patrón/comando
El patrón
busca todas las líneas que contengan ese patrón, de manera similar al patrón en el comando de sustitución. El comando
puede ser cualquier comando de la línea de comandos. El comando global funciona al ejecutar comando
en cada línea que coincida con el patrón
.
Si tienes las siguiente expresiones:
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
Para eliminar todas las líneas que contengan la palabra "console", puedes ejecutar:
:g/console/d
Este sería el resultado:
const one = 1;
const two = 2;
const three = 3;
El comando global ejecuta el comando eliminar (d
) en todas las líneas que coincidan con el patrón "console".
Al ejecutar el comando g
, Vim realiza dos escaneos del archivo. En el primero, escanea cada línea y marca las líneas que coinciden con el patrón /console/
. Una vez que todas las líneas que coinciden están marcadas, vuelve a hacer otra pasada por todo el archivo, donde ejecuta el comando d
en las líneas marcadas.
Si en vez de eso, quieres eliminar todas las líneas que contienen "const", ejecuta:
:g/const/d
Este sería el resultado:
console.log("one: ", one);
console.log("two: ", two);
console.log("three: ", three);
Invertir las coincidencias
Para ejecutar el comando global en las líneas que no cumplan la coincidencia con el patrón, puedes ejecutar:
:g!/{patrón}/{comando}
o
:v/{patrón}/{comando}
Si ejecutas :v/console/d
, esto eliminará todas las líneas que no contengan la palabra "console".
Patrón
El comando global utiliza el mismo sistema de patrones que el comando de sustitución, así que esta sección servirá como repaso. Si lo deseas, sientete libre de saltar a la siguiente sección o sigue leyendo.
Si tienes las siguientes expresiones:
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
Para eliminar las líneas que contienen tanto "one" o "two", ejecuta:
:g/one\|two/d
Para eliminar las líneas que contienen algún número de una cifra, ejecuta una de estas dos opciones:
:g/[0-9]/d
o
:g/\d/d
Si tienes las siguientes expresiones:
const oneMillion = 1000000;
const oneThousand = 1000;
const one = 1;
Para encontrar las líneas que contienen entre tres a seis ceros, ejecuta:
:g/0\{3,6\}/d
Pasando un rango
Puedes pasar un rango antes del comando g
. Aquí tienes algunas maneras de cómo lo puedes hacer:
:1,5/g/console/d
encuentra la cadena "console" entre las líneas 1 y 5 y las elimina.:,5/g/console/d
si no se especifica un número delante de la coma, entonces esto indica que comience el comando desde la línea actual. Busca la cadena "console" entre la línea actual y la línea 5 y las elimina.:3,/g/console/d
si no se especifica un número después de la coma, entonces indica que el comando finaliza en la línea actual. Busca la cadena "console" entre la línea 3 y la línea actual y las elimina.:3g/console/d
si solo se le pasa un número sin el símbolo de la coma, esto ejecuta el comando solo en la línea 3. Busca en la línea 3 y la elimina si existe la cadena "console".
Además de los números, también se pueden utilizar esto símbolos como rango:
.
significa la línea actual. Un rango de.,3
significa entre la línea actual y la línea 3.$
significa la última línea del archivo.3,$
es un rango entre la línea 3 y la última línea del archivo.+n
significa n líneas después de la línea actual. Lo puedes utilizar con.
o sin ese símbolo.3,+1
o3,.+1
ambas significa, entre la línea 3 y la línea después de la línea actual.
Si no pasas ningún rango, de manera predeterminada se entiende que el comando se aplica al archivo completo. Esto realmente no es la norma general. La mayoría de los comandos para la línea de comandos de Vim se ejecutan solo en la línea actual si no se le pasa ningún rango. Las dos excepciones más notables son los comandos globales (:g
) y el comando de guardar (:w
).
El comando normal
Puedes ejecutar un comando normal dentro del comando global con el comando :normal
de la línea de comandos.
Si tienes el siguiente texto:
const one = 1
console.log("one: ", one)
const two = 2
console.log("two: ", two)
const three = 3
console.log("three: ", three)
Para añadir un símbolo ";" al final de cada línea, ejecuta:
:g/./normal A;
Diseccionesmos el comando:
:g
es el comando global././
es el patrón para "todas las líneas que no estén vacías". Recuerda que el punto (.
) es un registro que representa cualquier caracter. Lo que hace es buscar las líneas con al menos un caracter, así que en este caso marcará las líneas con las palabras "const" y "console". Ignorando las líneas vacías.normal A;
ejecuta el comando de la línea de comando:normal
.A;
es el comando normal para insertar un ";" al final de la línea.
Ejecutar una macro
También puedes ejecutar una macro con el comando global. Una macroes simplemente un modo normal de operación, así que es posible ejecutarlo con :normal
. Si tienes las siguiente expresiones:
const one = 1
console.log("one: ", one);
const two = 2
console.log("two: ", two);
const three = 3
console.log("three: ", three);
Ten en cuenta que las líneas con "const" no tienen un símbolo de punto y coma al final de la línea. Vamos a crear una macro, que almacenaremos en el registro "a", que añada un símbolo de punto y coma al final de esas líneas:
qa0A;<esc>q
Si necesitas refrescar conocimientos, echa un vistazo al capítulo que trata sobre las macros. Ahora ejecuta:
:g/const/normal @a
Ahora todas las líneas con "const" tendrán un ";" al final de la línea.
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
Comando recursivo global
El comando global pro sí mismo es un tipo de comando de la línea de comandos, así que técnicamente puedes ejecutar el comando global dentro de un comando global.
Dadas las siguientes expresiones:
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
Si ejecutas:
:g/console/g/two/d
Primero, g
buscará las líneas que contengan el patrón "console" y encontrará 3 coincidencias. El segundo g
buscará la línea que contenga el patrón "two" para esas tres coincidencias encontradas anteriormente. Finalmente, eliminará esas coincidencias.
Puedes combinar g
con v
para encontrar patrones positivos o negativos. Por ejemplo:
:g/console/v/two/d
En vez de buscar las líneas que contienen "two", buscará las líneas que no contienen el patrón "two".
Cambiando el símbolo del delimitador del comando global
Puedes cambiar el delimitador del comando global de igual manera que se podía hacer en el comando de sustitución. Las reglas son las mismas: puedes utilizar cualquier caracter de un único byte excepto las letras del alfabeto, números, "
, |
, y \
.
Para eliminar las líneas que contienen la palabra "console":
:g@console@d
Si estás utilizando el comando de sustitución con el comando global, puedes utilizar dos delimitadores diferentes:
g@one@s+const+let+g
Aquí el comando global buscará todas laslíneas que contienen "one". El comando de sustitución, sustituirá de esas líneas encontradas, la cadena "const" con "let".
El comando predeterminado
¿Qué pasa si no especificas ningún comando de la línea de comandos en el comando global?
El comando global utilizará el comando print o mostrar por pantalla (:p
) para mostrar la línea actual. Si ejecutas:
:g/console
Esto mostrará en la parte inferior de la pantalla todas las líneas que contengan "console".
Por cierto, aquí hay un hecho curioso. Debido a que el comando predeterminado utilizado por el comando global es p
, esto hace que la sintaxis de g
sea:
:g/re/p
g
= el comando globalre
= la expresión regular del patrón de búsquedap
= el comando imprimir por pantalla
Esto forma "grep"*, el mismo grep
que se utiliza en la línea de comandos. Esto no es una coincidencia. El comando g/re/p
procede originalmente del editor Ed, uno de los primeros editores de texto. El comando grep
toma su nombre de Ed.
Tu equipo probablemente todavía tenga un editor Ed instalado. Ejecuta ed
en una terminal (pista: para salir, escribe q
).
Más ejemplos
Invertir el buffer completo
Para invertir un archivo por completo, ejecuta:
:g/^/m 0
^
es un patrón que significa "el comienzo de una línea". Utiliza ^
para seleccionar todas las líneas, incluyendo las líneas vacías.
Si solo necesitas invertir unas cuantas líneas y no el archivo por completo, puedes añadir un rango. Para invertir las líneas de la cinco a la diez, ejecuta:
:5,10g/^/m 0
Para aprender más sobre el comando de mover, echa un vistazo a :h :move
.
Agregar todas las tareas pendientes o TODO
Mientras estás escribiendo código, a veces te vienen a la cabeza diferentes ideas aleatoria brillantes. Si no quieres perder la concentración de lo que estás haciendo, puedes anotarlas en el propio archivo que estás editando, y etiquetarlas como TODO o PENDIENTE, por ejemplo:
const one = 1;
console.log("one: ", one);
// TODO: dar de comer a las gatas
const two = 2;
// TODO: dar de comer a las gatas automáticamente
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
// TODO: crear un invento que sirva para dar de comer a las gatas automáticamente
Puede ser complicado llevar la cuenta de todas las tareas TODO que hemos ido creando. Vim tiene un método para copiar todas las coincidencias encontradas a una dirección con el comando :t
. Para aprender más sobre este método de copiado, echa un vistazo a :h :copy
.
Para copiar todas las tareas TODOs al final del archivo para darles un repaso, ejecuta:
:g/TODO/t $
Este sería el resultado:
const one = 1;
console.log("one: ", one);
// TODO: dar de comer a las gatas
const two = 2;
// TODO: dar de comer a las gatas automáticamente
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
// TODO: crear un invento que sirva para dar de comer a las gatas automáticamente
// TODO: dar de comer a las gatas
// TODO: dar de comer a las gatas automáticamente
// TODO: crear un invento que sirva para dar de comer a las gatas automáticamente
Ahora puede revisar fácilmente todas las tareas TODO que he ido creando, y encontrar el tiempo para realizarlas o delegarlas a otra persona y continuar en mi trabajo con mis siguientes tareas.
Otra alternativa es utilizar m
:
:g/TODO/m $
Y este sería el resultado:
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
// TODO: dar de comer a las gatas
// TODO: dar de comer a las gatas automáticamente
// TODO: crear un invento que sirva para dar de comer a las gatas automáticamente
Puedo simplemente eliminar la lista una vez que decida qué hacer con ella.
Mandar el contenido borrado a un agujero negro
Recuerda del capítulo de los registros que los textos eliminados son almacenados dentro de los registros numerados (aunque son suficientemente grandes). Cada vez que ejecutas :g/console/d
, Vim almacena las líneas eliminadas en los registros numerados. Si eliminas muchas líneas, puedes rápidamente llenar el contenido de esos registros numerados. Para eliminar esto, siempre puedes utilizar el registro del agujero negro ("_
) para no almacenar las líneas eliminadas en los registros. Ejecuta:
:g/console/d _
Al pasar al comando el símbolo _
después del comando de borrar d
, Vim no almacenará las líneas eliminadas en ningún registro.
Reducir múltiples líneas vacías en una única línea vacía
Si tienes un archivo con múltiples líneas vacías como este:
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
Puedes reducir rápidamente cada conjunto de líneas vacías consecutivas a una única línea vacía. Ejecuta:
:g/^$/,/./-1j
Este es el resultado:
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
Vamos a verlo en detalle:
:g
es el comando global./^$/
es el patrón para una línea vacía. Recuerda que^
significa el comienzo de una línea y$
el final de una línea.^$
encuentra una línea vacía (una línea con una longitud de cero caracteres).,/./-1
es el rango para el comandoj
. Como no se le pasa ningún valor para el comienzo del rango, este comienza en la línea actual. Ya aprendiste anteriormente que/./
es el patrón para una línea no vacía.,/./
es un rango desde la línea actual hasta la siguiente línea no vacía. El rango del comando global,/^$/
, te llevará hasta la primera coincidencia en la línea inferiorconsole.log("one: ", one);
. Esta es la línea actual././
encuentra la primera línea no vacía, la líneaconst two = 2;
. Finalmente,-1
cambia eso por una línea. El rango efectivo para la primera concurrencia es la línea vacía debajo deconsole.log("one: ", one);
y la línea vacía encima deconst two = 2;
.j
es el comando Para unir líneas:j
. Puedes unir todas las líneas dadas como su rango. Por ejemplo,:1,5j
une las líneas de la uno a la cinco.
Ten en cuenta que estás pasando un rango (,/./-1
) antes del comando j
. Solo porqie estás utilizando un comando para la línea de comandos de Vim con el comando global, no significa que no puedas pasarla un rango. En este código, estás pasando al comando j
su propio rango para ejecutarlo. Puedes pasar un rango para cualquier comando mientras estás ejecutando el comando global.
Por cierto, si quieres reducir múltiples líneas vacías en una línea, en vez de utilizar ,/./-1
como rango para el comando j
, simplemente utiliza ,/./
como rango en vez del anterior.
:g/^$/,/./j
O simplificado:
:g/^$/-j
Tu texto ahora es reducido a:
const one = 1;
console.log("one: ", one);
const two = 2;
console.log("two: ", two);
const three = 3;
console.log("three: ", three);
Advanced sort
Vim has a :sort
command to sort the lines within a range. For example:
d
b
a
e
c
You can sort them by running :sort
. If you give it a range, it will sort only the lines within that range. For example, :3,5sort
sorts only between lines three and five.
If you have the following expressions:
const arrayB = [
"i",
"g",
"h",
"b",
"f",
"d",
"e",
"c",
"a",
]
const arrayA = [
"h",
"b",
"f",
"d",
"e",
"a",
"c",
]
If you need to sort the elements inside the arrays, but not the arrays themselves, you can run this:
:g/\[/+1,/\]/-1sort
Result:
const arrayB = [
"a",
"b",
"c",
"d",
"e",
"f",
"g",
"h",
"i",
]
const arrayA = [
"a"
"b",
"c",
"d",
"e",
"f",
"h",
]
This is great! But the command looks complicated. Let's break it down. The command consists of three main parts: the global command pattern, the sort command range, and the sort command.
:g/\[/
is the global command pattern.
:g
is the global command./\[/
is the pattern used by the global command.\[
looks for a literal "[" string.
+1,/\]/-1
is the range for the sort command.
- A range can have a starting and an ending addresses. In this case,
+1
is the starting address and/\]/-1
is the ending address. +1
represents the line after the current line, which is the line that matches the pattern "[" from the global command.+1
offsets the current line by one line. So in the first match, the range actually starts one line after theconst arrayB = [
text./\]/-1
is the ending address.\]
represents a literal closing square bracket "]".-1
offsets it by one line. The ending address is the line above the "]".
sort
is the sort command-line command. It sorts everything within the given range. Everything after the "[" to the line above "]" is getting sorted.
If you are still confused by the command, do not worry. It took me a long time to grasp it. Take a break, leave the screen, and come back again with a fresh mind.
Learn the Global Command the Smart Way
The global command executes the command-line command against all matching lines. With it, you only need to run a command once and Vim will do the rest for you. To become proficient at the global command, two things are required: a good vocabulary of command-line commands and a knowledge of regular expressions. As you spend more time using Vim, you will naturally learn more command-line commands. A regular expression knowledge will require a more active approach. But once you become comfortable with regular expressions, you will be ahead of many.
Some of the examples here are complicated. Do not be intimidated. Really take your time to understand them. Learn to read the patterns. Make sure you know what each letter in each command represent. Do not give up.
Whenever you need to apply a command in several locations, pause and see if you can use the g
command. Look for the best command for the job and write a pattern to target as many things at once. Then repeat it until you can do it without thinking. The next time, see if there is even a faster and more efficient way to do it.
Now that you know how powerful the global command is, let's learn how to use the external commands to increase your tool arsenals.