Merge remote-tracking branch 'LCTT/master'

This commit is contained in:
Xingyu.Wang 2019-03-16 11:02:16 +08:00
commit c17fd2b9c9
3 changed files with 417 additions and 239 deletions

View File

@ -1,239 +0,0 @@
[#]: collector: (lujun9972)
[#]: translator: (HankChow)
[#]: reviewer: ( )
[#]: publisher: ( )
[#]: url: ( )
[#]: subject: (All about {Curly Braces} in Bash)
[#]: via: (https://www.linux.com/blog/learn/2019/2/all-about-curly-braces-bash)
[#]: author: (Paul Brown https://www.linux.com/users/bro66)
All about {Curly Braces} in Bash
======
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/curly-braces-1920.jpg?itok=cScRhWrX)
At this stage of our Bash basics series, it would be hard not to see some crossover between topics. For example, you have already seen a lot of brackets in the examples we have shown over the past several weeks, but the focus has been elsewhere.
For the next phase of the series, well take a closer look at brackets, curly, curvy, or straight, how to use them, and what they do depending on where you use them. We will also tackle other ways of enclosing things, like when to use quotes, double-quotes, and backquotes.
This week, we're looking at curly brackets or _braces_ : `{}`.
### Array Builder
You have already encountered curly brackets before in [The Meaning of Dot][1]. There, the focus was on the use of the dot/period (`.`), but using braces to build a sequence was equally important.
As we saw then:
```
echo {0..10}
```
prints out the numbers from 0 to 10. Using:
```
echo {10..0}
```
prints out the same numbers, but in reverse order. And,
```
echo {10..0..2}
```
prints every second number, starting with 10 and making its way backwards to 0.
Then,
```
echo {z..a..2}
```
prints every second letter, starting with _z_ and working its way backwards until _a_.
And so on and so forth.
Another thing you can do is combine two or more sequences:
```
echo {a..z}{a..z}
```
This prints out all the two letter combinations of the alphabet, from _aa_ to _zz_.
Is this useful? Well, actually it is. You see, arrays in Bash are defined by putting elements between parenthesis `()` and separating each element using a space, like this:
```
month=("Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec")
```
To access an element within the array, you use its index within brackets `[]`:
```
$ echo ${month[3]} # Array indexes start at [0], so [3] points to the fourth item
Apr
```
You can accept all those brackets, parentheses, and braces on faith for a moment. We'll talk about them presently.
Notice that, all things being equal, you can create an array with something like this:
```
letter_combos=({a..z}{a..z})
```
and `letter_combos` points to an array that contains all the 2-letter combinations of the entire alphabet.
You can also do this:
```
dec2bin=({0..1}{0..1}{0..1}{0..1}{0..1}{0..1}{0..1}{0..1})
```
This last one is particularly interesting because `dec2bin` now contains all the binary numbers for an 8-bit register, in ascending order, starting with 00000000, 00000001, 00000010, etc., until reaching 11111111. You can use this to build yourself an 8-bit decimal-to-binary converter. Say you want to know what 25 is in binary. You can do this:
```
$ echo ${dec2bin[25]}
00011001
```
Yes, there are better ways of converting decimal to binary as we saw in [the article where we discussed & as a logical operator][2], but it is still interesting, right?
### Parameter expansion
Getting back to
```
echo ${month[3]}
```
Here the braces `{}` are not being used as apart of a sequence builder, but as a way of generating _parameter expansion_. Parameter expansion involves what it says on the box: it takes the variable or expression within the braces and expands it to whatever it represents.
In this case, `month` is the array we defined earlier, that is:
```
month=("Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec")
```
And, item 3 within the array points to `"Apr"` (remember: the first index in an array in Bash is `[0]`). That means that `echo ${month[3]}`, after the expansion, translates to `echo "Apr"`.
Interpreting a variable as its value is one way of expanding it, but there are a few more you can leverage. You can use parameter expansion to manipulate what you read from variable, say, by cutting a chunk off the end.
Suppose you have a variable like:
```
a="Too longgg"
```
The command:
```
echo ${a%gg}
```
chops off the last two gs and prints " _Too long_ ".
Breaking this down,
* `${...}` tells the shell to expand whatever is inside it
* `a` is the variable you are working with
* `%` tells the shell you want to chop something off the end of the expanded variable ("Too longgg")
* and `gg` is what you want to chop off.
This can be useful for converting files from one format to another. Allow me to explain with a slight digression:
[ImageMagick][3] is a set of command line tools that lets you manipulate and modify images. One of its most useful tools ImageMagick comes with is `convert`. In its simplest form `convert` allows you to, given an image in a certain format, make a copy of it in another format.
The following command takes a JPEG image called _image.jpg_ and creates a PNG copy called _image.png_ :
```
convert image.jpg image.png
```
ImageMagick is often pre-installed on most Linux distros. If you can't find it, look for it in your distro's software manager.
Okay, end of digression. On to the example:
With variable expansion, you can do the same as shown above like this:
```
i=image.jpg
convert $i ${i%jpg}png
```
What you are doing here is chopping off the extension `jpg` from `i` and then adding `png`, making the command `convert image.jpg image.png`.
You may be wondering how this is more useful than just writing in the name of the file. Well, when you have a directory containing hundreds of JPEG images, you need to convert to PNG, run the following in it:
```
for i in *.jpg; do convert $i ${i%jpg}png; done
```
... and, hey presto! All the pictures get converted automatically.
If you need to chop off a chunk from the beginning of a variable, instead of `%`, use `#`:
```
$ a="Hello World!"
$ echo Goodbye${a#Hello}
Goodbye World!
```
There's quite a bit more to parameter expansion, but a lot of it makes sense only when you are writing scripts. We'll explore more on that topic later in this series.
### Output Grouping
Meanwhile, let's finish up with something simple: you can also use `{ ... }` to group the output from several commands into one big blob. The command:
```
echo "I found all these PNGs:"; find . -iname "*.png"; echo "Within this bunch of files:"; ls > PNGs.txt
```
will execute all the commands but will only copy into the _PNGs.txt_ file the output from the last `ls` command in the list. However, doing
```
{ echo "I found all these PNGs:"; find . -iname "*.png"; echo "Within this bunch of files:"; ls; } > PNGs.txt
```
creates the file _PNGs.txt_ with everything, starting with the line " _I found all these PNGs:_ ", then the list of PNG files returned by `find`, then the line "Within this bunch of files:" and finishing up with the complete list of files and directories within the current directory.
Notice that there is space between the braces and the commands enclosed within them. Thats because `{` and `}` are _reserved words_ here, commands built into the shell. They would roughly translate to " _group the outputs of all these commands together_ " in plain English.
Also notice that the list of commands has to end with a semicolon (`;`) or the whole thing will bork.
### Next Time
In our next installment, we'll be looking at more things that enclose other things, but of different shapes. Until then, have fun!
Read more:
[And, Ampersand, and & in Linux][4]
[Ampersands and File Descriptors in Bash][5]
[Logical & in Bash][2]
--------------------------------------------------------------------------------
via: https://www.linux.com/blog/learn/2019/2/all-about-curly-braces-bash
作者:[Paul Brown][a]
选题:[lujun9972][b]
译者:[译者ID](https://github.com/译者ID)
校对:[校对者ID](https://github.com/校对者ID)
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
[a]: https://www.linux.com/users/bro66
[b]: https://github.com/lujun9972
[1]: https://www.linux.com/blog/learn/2019/1/linux-tools-meaning-dot
[2]: https://www.linux.com/blog/learn/2019/2/logical-ampersand-bash
[3]: http://www.imagemagick.org/
[4]: https://www.linux.com/blog/learn/2019/2/and-ampersand-and-linux
[5]: https://www.linux.com/blog/learn/2019/2/ampersands-and-file-descriptors-bash

View File

@ -0,0 +1,182 @@
[#]: collector: (oska874)
[#]: translator: ( )
[#]: reviewer: ( )
[#]: publisher: ( )
[#]: url: ( )
[#]: subject: (Getting started with PiFlash: Booting your Raspberry Pi on Linux)
[#]: via: (https://opensource.com/article/19/3/piflash?utm_campaign=intrel)
[#]: author: (Ian Kluft https://opensource.com/users/ikluft)
Getting started with PiFlash: Booting your Raspberry Pi on Linux
============================================================
### Linux users can say goodbye to manually creating bootable SD cards for Raspberry Pi with PiFlash.
![Vector, generic Raspberry Pi board](https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/raspberrypi_board_vector_red.png?itok=yaqYjYqI "Vector, generic Raspberry Pi board")
Most people install some form of Linux when they set up a Raspberry Pi computer. Until recently, the installation tools for creating a bootable SD card were only available for Windows and MacOS desktops. If you were running Linux on your desktop, you got a page of instructions on doing it manually from the command line.
That works. But if you experiment with multiple Raspberry Pi boards, over time this gets tedious. As an engineer and a coder, I started thinking about automating it to make it easier and, in the usual way open source projects get started, I came away with [PiFlash][10].
I wrote PiFlash just for myself in 2016\. By February 2017, it had evolved far enough that I introduced it in a presentation at SVPerl (Silicon Valley Perl) about using Perl on the Raspberry Pi, then was invited to write two articles for Opensource.com:  _[Getting Started with Perl on the Raspberry Pi][7]_  and  _[How to Secure Your Raspberry Pi][8]._
### PiFlash features
PiFlash has features that appeal to beginners and experts.
Like most other open source software tools, you don't need to see the language it's written in, if you're not into that. But the source code is available for power users and participants.
For expert users, PiFlash simplifies the process of writing an SD card. When you download a bootable OS "image" file, it's usually in a ZIP archive or compressed. All the distributions package them a little differently. With PiFlash, you don't have to unpack or decompress the image. Just specify the downloaded file as the input and PiFlash will do the tedious part of extracting it.
For beginners, there's an important safety feature: PiFlash will write  _only_  to an SD card and refuse to write to any other type of device. Since you have to use root permissions to write the card, the system will do anything you tell it to. Therefore, it's possible to accidentally erase the wrong device, maybe a hard drive you want to keep, when trying to flash an SD card manually for a new Raspberry Pi.
This is where PiFlash protects you from danger. Internally, it finds device attributes with the **lsblk** command from the **util-linux** package, which is part of all Linux distributions. It can recognize SD cards using various drivers. It will refuse to write to a block device if it isn't an SD card.
Fortunately, the Etcher GUI tool that Raspberry Pi Foundation uses in its instructions for Windows and MacOS users has been expanded to Linux, so there is now a GUI option on Linux for those who prefer one. But if you want to automate the process, or if you want power-user levels of customization, only a command-line tool will do the job.
The latest version of PiFlash adds plugin modules that can modify the SD card's filesystem after installing the OS image, so you can start to explore new options for automation.
### Installing PiFlash
[PiFlash is available][11] from [CPAN][12], the Comprehensive Perl Archive Network—but before you proceed, make sure you have all the dependency packages installed. To install the dependencies:
On RPM-based Linux systems (Red Hat Enterprise, Fedora, CentOS, etc.):
```
sudo dnf install coreutils util-linux perl file-libs perl-File-LibMagic perl-IO perl-Exception-Class perl-Try-Tiny perl-Module-Pluggable perl-File-Path perl-YAML-LibYAML gzip unzip xz e2fsprogs dosfstools
```
On Debian-based Linux systems (Debian, Ubuntu, Raspbian, etc.):
```
sudo apt-get install coreutils util-linux klibc-utils perl-base libmagic1 libfile-libmagic-perl libio-all-perl libexception-class-perl libtry-tiny-perl libmodule-pluggable-perl libyaml-libyaml-perl gzip xz-utils e2fsprogs dosfstools
```
For source-based distributions or other packaging systems, see the CPAN documentation for the dependency list.
Next, install PiFlash using the CPAN tool:
```
cpan PiFlash
```
I have the [Dist:][13][:Zilla][14]-based build set up to make DEB and RPM packages, but it isn't in any of the major package archives yet. That's possible in the future.
### Running PiFlash
If you just run the **piflash** command without any arguments, it will print usage information.
```
usage: piflash [--verbose] [--resize] [--config conf-file] input-file output-device
       piflash [--verbose] [--config conf-file] --SDsearch
       piflash --version
```
Scan the system for SD cards to get the exact device name, which you'll need for the Pi-Flash output-device parameter below.
```
piflash --sdsearch
```
If no SD cards are found, it says it can't find anything.
```
no SD cards found on system
```
More on Raspberry Pi
* [What is Raspberry Pi?][1]
* [Getting started with Raspberry Pi][2]
* [Getting started with Raspberry Pi cheat sheet][3]
* [Our latest on Raspberry Pi][4]
* [Send us your Raspberry Pi projects and tutorials][5]
By the way, if you have an SD card writer that PiFlash doesn't know about, please let me know by filing a report on GitHub. For problem reports and troubleshooting, please collect the program's internal information by using the **--verbose** option so I can see what driver your system has that PiFlash didn't recognize.
Your exact device name may vary by drivers and the names used by other devices on your system. If you have a USB-based SD reader/writer, it may say something like this:
```
SD cards found: /dev/sdb
```
Or if you have a built-in SD card slot, it may use a different driver and have a name that indicates its an SD card using the MMC (MultiMediaCard) driver:
```
SD cards found: /dev/mmcblk0
```
Next, download a system image to install. The Raspberry Pi Foundation has an old [list of possibilities][15] that is no longer updated. Since Raspbian is the official Linux distribution for the Raspberry Pi, driver support goes there first. But others work: Ubuntu is on the list, but Fedora isn't because ARM and Raspberry Pi support came after the list was made, however, you can [download it][16].
The command to flash the SD card is **piflash <input-file> <output-device **. Here are some examples of commands to flash it, depending upon the system you downloaded and which block device has your SD card:
```
piflash 2018-11-13-raspbian-stretch-full.zip /dev/mmcblk0
piflash Fedora-LXDE-armhfp-29-1.2-sda.raw.xz /dev/sdb
```
PiFlash now has a **--resize** option, which will resize the root partition on the SD card to fill the rest of the device. It only works if the root is a Linux ext2/3/4 filesystem. Most distributions make that available as an option you can do manually after installation, but this automates it to one step.
### Plugins, automation, and future development
PiFlash 0.3.1 is current at the time of this writing.
PiFlash 0.1.0 introduced plugin modules, which means Perl modules named like **PiFlash::Plugin::Something** will be available to the program. As a security precaution, because PiFlash can run commands as **sudo-root**, plugins are not automatically loaded. Instead, they have to be enabled via **--plugin** on the command line or in a configuration file. The name of the plugin is the Perl module name without the **PiFlash::Plugin:: prefix**. For more than one plugin, it's a comma-delimited list of names. Enabled plugins can use a hook that is called when the boot and root filesystems are mounted after writing the SD card. More on plugins can be found at the [**PiFlash::Plugin**][17] module documentation.
Here are some ideas on my own to-do list for upcoming plugin modules...
* Change the password of the user and root accounts on the SD card before it boots
* Set the network parameters and hostname
* Modify system configuration parameters that would usually require manually running **raspi-config** or editing **config.txt**
* Set up a custom Linux environment around ARM kernels you build
My ideas are, of course, influenced by what I want to do. You may want to go in other directions, so please consider contributing patches and plugins.
The source code repository for PiFlash is [on GitHub][18]. Please file problem reports and code/documentation contributions there, also.
### About the author
Ian Kluft - Ian has had parallel interests since grade school in computing and flight. He was coding on Unix before there was Linux, and started on Linux 6 months after the kernel was posted. He has a masters degree in Computer Science and is a CSSLP (Certified Secure Software Lifecycle Professional). On the side he's a pilot and a certified flight instructor. As a licensed Ham Radio operator for over 25 years, experimentation with electronics has evolved in recent years to include the Raspberry Pi and...
--------------------------------------------------------------------------------
via: https://opensource.com/article/19/3/piflash?utm_campaign=intrel
作者:[ Ian Kluft ][a]
选题:[ezio][b]
译者:[译者ID](https://github.com/译者ID)
校对:[校对者ID](https://github.com/校对者ID)
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
[a]:https://opensource.com/users/ikluft
[b]:https://github.com/oska874
[1]:https://opensource.com/resources/what-raspberry-pi?src=raspberry_pi_resource_menu1&intcmp=701f2000000h4RcAAI
[2]:https://opensource.com/article/16/12/getting-started-raspberry-pi?src=raspberry_pi_resource_menu2&intcmp=701f2000000h4RcAAI
[3]:https://opensource.com/downloads/getting-started-raspberry-pi-cheat-sheet?src=raspberry_pi_resource_menu3&intcmp=701f2000000h4RcAAI
[4]:https://opensource.com/tags/raspberry-pi?src=raspberry_pi_resource_menu4&intcmp=701f2000000h4RcAAI
[5]:https://opensource.com/article/17/2/raspberry-pi-submit-your-article?src=raspberry_pi_resource_menu5&intcmp=701f2000000h4RcAAI
[6]:https://opensource.com/article/19/3/piflash?rate=uzXbDSgdiRC1mnuHSCo-WHnHP7-lSyoYkTjdfMcFcUI
[7]:https://opensource.com/article/17/3/perl-raspberry-pi
[8]:https://opensource.com/article/17/3/iot-security-raspberry-pi
[9]:https://opensource.com/user/120171/feed
[10]:https://github.com/ikluft/piflash
[11]:https://metacpan.org/release/PiFlash
[12]:http://www.cpan.org/
[13]:https://metacpan.org/pod/Dist::Zilla
[14]:https://metacpan.org/pod/Dist::Zilla
[15]:https://www.raspberrypi.org/downloads/
[16]:https://arm.fedoraproject.org/
[17]:https://metacpan.org/pod/PiFlash::Plugin
[18]:https://github.com/ikluft/piflash
[19]:https://opensource.com/users/ikluft
[20]:https://opensource.com/users/ikluft
[21]:https://opensource.com/tags/raspberry-pi
[22]:https://opensource.com/tags/linux

View File

@ -0,0 +1,235 @@
[#]: collector: (lujun9972)
[#]: translator: (HankChow)
[#]: reviewer: ( )
[#]: publisher: ( )
[#]: url: ( )
[#]: subject: (All about {Curly Braces} in Bash)
[#]: via: (https://www.linux.com/blog/learn/2019/2/all-about-curly-braces-bash)
[#]: author: (Paul Brown https://www.linux.com/users/bro66)
浅析 Bash 中的 {花括号}
======
![](https://www.linux.com/sites/lcom/files/styles/rendered_file/public/curly-braces-1920.jpg?itok=cScRhWrX)
在前面的 Bash 基础系列文章中,我们或多或少地使用了一些还没有讲到的符号。在之前文章的很多例子中,我们都使用到了括号,但并没有重点讲解关于括号的内容。
这个系列接下来的文章中,我们会研究括号们的用法:如何使用这些括号?将它们放在不同的位置会有什么不同的效果?除了圆括号、方括号、花括号以外,我们还会接触另外的将一些内容“包裹”起来的符号,例如单引号、双引号和反引号。
在这周,我们先来看看花括号 `{}`
### 构造序列
花括号在之前的《[点的含义][1]》这篇文章中已经出现过了,当时我们只对点号 `.` 的用法作了介绍。但在构建一个序列的过程中,同样不可以缺少花括号。
我们使用
```
echo {0..10}
```
来顺序输出 0 到 10 这 11 个数。使用
```
echo {10..0}
```
可以将这 11 个数倒序输出。更进一步,可以使用
```
echo {10..0..2}
```
来跳过其中的奇数。
```
echo {z..a..2}
```
则从倒序输出字母表,并跳过其中的第奇数个字母。
以此类推。
还可以将两个序列进行组合:
```
echo {a..z}{a..z}
```
这个命令会将从 aa 到 zz 的所有双字母组合依次输出。
这是很有用的。在 Bash 中,定义一个数组的方法是在圆括号 `()` 中放置各个元素并使用空格隔开,就像这样:
```
month=("Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec")
```
如果需要获取数组中的元素,就要使用方括号 `[]` 并在其中填入元素的索引:
```
$ echo ${month[3]} # 数组索引从 0 开始,因此 [3] 对应第 4 个元素
Apr
```
先不要过分关注这里用到的三种括号,我们等下会讲到。
注意,像上面这样,我们可以定义这样一个数组:
```
letter_combos=({a..z}{a..z})
```
其中 `letter_combos` 变量指向的数组依次包含了从 aa 到 zz 的所有双字母组合。
因此,还可以这样定义一个数组:
```
dec2bin=({0..1}{0..1}{0..1}{0..1}{0..1}{0..1}{0..1}{0..1})
```
在这里,`dec2bin` 变量指向的数组按照升序依次包含了所有 8 位的二进制数,也就是 00000000、00000001、00000010……11111111。这个数组可以作为一个十进制数到 8 位二进制数的转换器。例如将十进制数 25 转换为二进制数,可以这样执行:
```
$ echo ${dec2bin[25]}
00011001
```
对于进制转换,确实还有更好的方法,但这不失为一个有趣的方法。
### <ruby>参数展开<rt>parameter expansion</rt></ruby>
再看回前面的
```
echo ${month[3]}
```
在这里,花括号的作用就不是构造序列了,而是用于参数展开。顾名思义,参数展开就是将花括号中的变量展开为这个变量实际的内容。
我们继续使用上面的 `month` 数组来举例:
```
month=("Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec")
```
注意Bash 中的数组索引从 0 开始,因此 3 代表第 4 个元素 `"Apr"`。因此 `echo ${month[3]}` 在经过参数展开之后,相当于 `echo "Apr"`
像上面这样将一个数组展开成它所有的元素,只是参数展开的其中一种用法。另外,还可以通过参数展开的方式读取一个字符串变量,并对其进行处理。
例如对于以下这个变量:
```
a="Too longgg"
```
如果执行:
```
echo ${a%gg}
```
可以输出“too long”也就是去掉了最后的两个 g。
在这里,
* `${...}` 告诉 shell 展开花括号里的内容
* `a` 就是需要操作的变量
* `%` 告诉 shell 需要在展开字符串之后从字符串的末尾去掉某些内容
* `gg` 是被去掉的内容
这个特性在转换文件格式的时候会比较有用,我来举个例子:
[ImageMagick][3] 是一套可以用于操作图像文件的命令行工具,它有一个 `convert` 命令。这个 `convert` 命令的作用是可以为某个格式的图像文件制作一个另一格式的副本。
下面这个命令就是使用 `convert` 为 JPEG 格式图像 `image.jpg` 制作一个 PNG 格式的图像副本 `image.png`
```
convert image.jpg image.png
```
在很多 Linux 发行版中都预装了 ImageMagick如果没有预装一般可以在发行版对应的软件管理器中找到。
继续来看,在对变量进行展开之后,就可以批量执行相类似的操作了:
```
i=image.jpg
convert $i ${i%jpg}png
```
这实际上是将变量 `i` 末尾的 `"jpg"` 去掉,然后加上 `"png"`,最终将整个命令拼接成 `convert image.jpg image.png`
如果你觉得并不怎么样,可以想象一下有成百上千个图像文件需要进行这个操作,而仅仅运行:
```
for i in *.jpg; do convert $i ${i%jpg}png; done
```
就瞬间完成任务了。
如果需要去掉字符串开头的部分,就要将上面的 `%` 改成 `#` 了:
```
$ a="Hello World!"
$ echo Goodbye${a#Hello}
Goodbye World!
```
参数展开还有很多用法,但一般在写脚本的时候才会需要用到。在这个系列以后的文章中就继续提到。
### 合并输出
最后介绍一个花括号的用法,这个用法很简单,就是可以将多个命令的输出合并在一起。首先看下面这个命令:
```
echo "I found all these PNGs:"; find . -iname "*.png"; echo "Within this bunch of files:"; ls > PNGs.txt
```
以分号分隔开的几条命令都会执行,但只有最后的 `ls` 命令的结果输出会被重定向到 `PNGs.txt` 文件中。如果将这几条命令用花括号包裹起来,就像这样:
```
{ echo "I found all these PNGs:"; find . -iname "*.png"; echo "Within this bunch of files:"; ls; } > PNGs.txt
```
执行完毕后,可以看到 `PNGs.txt` 文件中会包含两次 `echo` 的内容、`find` 命令查找到的 PNG 文件以及最后的 `ls` 命令结果。
需要注意的是,花括号与命令之间需要有空格隔开。因为这里的花括号 `{``}` 是作为 shell 中的保留字shell 会将这两个符号之间的输出内容组合到一起。
另外,各个命令之间要用分号 `;` 分隔,否则命令无法正常运行。
### 下期预告
在后续的文章中,我会介绍其它“包裹”类符号的用法,敬请关注。
相关阅读:
[And, Ampersand, and & in Linux][4]
[Ampersands and File Descriptors in Bash][5]
[Logical & in Bash][2]
--------------------------------------------------------------------------------
via: https://www.linux.com/blog/learn/2019/2/all-about-curly-braces-bash
作者:[Paul Brown][a]
选题:[lujun9972][b]
译者:[HankChow](https://github.com/HankChow)
校对:[校对者ID](https://github.com/校对者ID)
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
[a]: https://www.linux.com/users/bro66
[b]: https://github.com/lujun9972
[1]: https://www.linux.com/blog/learn/2019/1/linux-tools-meaning-dot
[2]: https://www.linux.com/blog/learn/2019/2/logical-ampersand-bash
[3]: http://www.imagemagick.org/
[4]: https://www.linux.com/blog/learn/2019/2/and-ampersand-and-linux
[5]: https://www.linux.com/blog/learn/2019/2/ampersands-and-file-descriptors-bash