243 lines
5.2 KiB
Markdown
243 lines
5.2 KiB
Markdown
|
<h2><center>Shell 并发控制</center></h2>
|
|||
|
|
|||
|
------
|
|||
|
|
|||
|
## 一:文件描述符
|
|||
|
|
|||
|
FD:文件描述符 / 文件句柄,进程使用文件描述符来管理进程打开的文件。
|
|||
|
|
|||
|
### 1. 查看当前进程打开的文件
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# ll /proc/$$/fd
|
|||
|
总用量 0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 0 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 1 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 2 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 19:49 255 -> /dev/pts/0
|
|||
|
```
|
|||
|
|
|||
|
### 2. 自定义当前进程用描述符号操作文件
|
|||
|
|
|||
|
手动定义文件描述符,只要没有被占用可以使用
|
|||
|
|
|||
|
自定义用当前进程打开一个文件
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# exec 6<> /file
|
|||
|
[root@wxin ~]# ll /proc/$$/fd
|
|||
|
总用量 0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 0 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 1 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 2 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 19:49 255 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 6 -> /file
|
|||
|
```
|
|||
|
|
|||
|
自定义用当前进程关闭一个文件
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# exec 6<&-
|
|||
|
[root@wxin ~]# ll /proc/$$/fd
|
|||
|
总用量 0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 0 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 1 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 2 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 19:49 255 -> /dev/pts/0
|
|||
|
```
|
|||
|
|
|||
|
### 3. 给文件写入内容
|
|||
|
|
|||
|
文件描述符fd方式
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# echo "wxin" >> /proc/$$/fd/6
|
|||
|
[root@wxin ~]# cat /file
|
|||
|
wxin
|
|||
|
```
|
|||
|
|
|||
|
文件名方式
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# echo "wxin" >> /file
|
|||
|
```
|
|||
|
|
|||
|
### 4. 文件描述符恢复文件
|
|||
|
|
|||
|
删除源文件
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# rm -rf /file
|
|||
|
```
|
|||
|
|
|||
|
新文件产生新的文件 inode
|
|||
|
|
|||
|
```shell
|
|||
|
[root@wxin ~]# cp /proc/$$/fd/6 /file
|
|||
|
```
|
|||
|
|
|||
|
查看状态为(deleted)
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# ll /proc/$$/fd
|
|||
|
总用量 0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 0 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 1 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 2 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 19:49 255 -> /dev/pts/0
|
|||
|
lrwx------. 1 root root 64 3月 26 17:21 6 -> /file (deleted)
|
|||
|
```
|
|||
|
|
|||
|
释放文件描述符,重新读取就显示正常了
|
|||
|
|
|||
|
### 二:管道
|
|||
|
|
|||
|
### 1. 匿名管道
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# rpm -qa |grep rpm
|
|||
|
```
|
|||
|
|
|||
|
### 2. 命令管道
|
|||
|
|
|||
|
创建命名管道文件(不是普通文件,管道内的文件只能读取一次,不会永久保存)
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# mkfifo /tmp/fifo1 f
|
|||
|
```
|
|||
|
|
|||
|
查看命名管道文件
|
|||
|
|
|||
|
```bash
|
|||
|
[root@wxin ~]# cat /tmp/fifo1
|
|||
|
|
|||
|
[root@wxin ~]# file /tmp/fifo1
|
|||
|
/tmp/fifo1: fifo (named pipe)
|
|||
|
```
|
|||
|
|
|||
|
## 三:并发控制
|
|||
|
|
|||
|
### 1. 基础并发:后台任务与`wait`
|
|||
|
|
|||
|
通过 `&` 将命令放到后台执行,再通过 `wait` 等待所有任务完成。
|
|||
|
|
|||
|
```shell
|
|||
|
#!/bin/bash
|
|||
|
|
|||
|
# 启动多个后台任务
|
|||
|
wget -q http://example.com/file1.jpg &
|
|||
|
wget -q http://example.com/file2.jpg &
|
|||
|
wget -q http://example.com/file3.jpg &
|
|||
|
|
|||
|
# 等待所有后台任务完成
|
|||
|
wait
|
|||
|
echo "All downloads completed!"
|
|||
|
```
|
|||
|
|
|||
|
### 2. 控制并发数量
|
|||
|
|
|||
|
同时运行过多任务可能导致资源耗尽。以下方法可限制并发数量:
|
|||
|
|
|||
|
**方法 1:循环计数 + 等待**
|
|||
|
|
|||
|
```shell
|
|||
|
#!/bin/bash
|
|||
|
max_workers=3 # 最大并发数
|
|||
|
task_list=(task1 task2 task3 task4 task5) # 任务列表
|
|||
|
|
|||
|
for task in "${task_list[@]}"; do
|
|||
|
(
|
|||
|
# 执行任务(此处替换为实际逻辑)
|
|||
|
echo "Processing $task..."
|
|||
|
sleep 1
|
|||
|
) &
|
|||
|
|
|||
|
# 控制并发数量
|
|||
|
if [[ $(jobs -r | wc -l) -ge $max_workers ]]; then
|
|||
|
wait -n # 等待任意一个后台任务完成(需要 bash 4.3+)
|
|||
|
fi
|
|||
|
done
|
|||
|
|
|||
|
wait # 等待剩余任务完成
|
|||
|
echo "All tasks done!"
|
|||
|
```
|
|||
|
|
|||
|
**方法 2:使用 `xargs` 的 `-P` 参数**
|
|||
|
|
|||
|
```shell
|
|||
|
# 使用 xargs 控制并行进程数
|
|||
|
echo task1 task2 task3 task4 task5 | xargs -n1 -P3 -I{} bash -c 'echo "Processing {}"; sleep 1'
|
|||
|
```
|
|||
|
|
|||
|
**方法 3:命名管道(FIFO)**
|
|||
|
|
|||
|
```shell
|
|||
|
#!/bin/bash
|
|||
|
max_workers=3
|
|||
|
task_list=(task1 task2 task3 task4 task5)
|
|||
|
|
|||
|
# 创建命名管道
|
|||
|
fifo=$(mktemp -u)
|
|||
|
mkfifo "$fifo"
|
|||
|
exec 3<>"$fifo"
|
|||
|
rm -f "$fifo"
|
|||
|
|
|||
|
# 初始化管道令牌
|
|||
|
for ((i=0; i<max_workers; i++)); do
|
|||
|
echo >&3
|
|||
|
done
|
|||
|
|
|||
|
# 处理任务
|
|||
|
for task in "${task_list[@]}"; do
|
|||
|
read -u3 # 获取令牌
|
|||
|
{
|
|||
|
echo "Processing $task..."
|
|||
|
sleep 1
|
|||
|
echo >&3 # 归还令牌
|
|||
|
} &
|
|||
|
done
|
|||
|
|
|||
|
wait
|
|||
|
exec 3>&-
|
|||
|
echo "All tasks done!"
|
|||
|
```
|
|||
|
|
|||
|
### 3. 示例
|
|||
|
|
|||
|
```shell
|
|||
|
#!/bin/bash
|
|||
|
max_workers=3
|
|||
|
urls=(
|
|||
|
"http://example.com/file1.jpg"
|
|||
|
"http://example.com/file2.jpg"
|
|||
|
"http://example.com/file3.jpg"
|
|||
|
"http://example.com/file4.jpg"
|
|||
|
)
|
|||
|
|
|||
|
download() {
|
|||
|
local url=$1
|
|||
|
local filename=$(basename "$url")
|
|||
|
if wget -q "$url" -O "$filename"; then
|
|||
|
echo "Downloaded: $filename"
|
|||
|
else
|
|||
|
echo "Failed: $filename" >&2
|
|||
|
return 1
|
|||
|
fi
|
|||
|
}
|
|||
|
|
|||
|
# 控制并发下载
|
|||
|
counter=0
|
|||
|
for url in "${urls[@]}"; do
|
|||
|
((counter++))
|
|||
|
( download "$url" ) &
|
|||
|
|
|||
|
if [[ $((counter % max_workers)) -eq 0 ]]; then
|
|||
|
wait # 每启动 max_workers 个任务后等待
|
|||
|
fi
|
|||
|
done
|
|||
|
|
|||
|
wait
|
|||
|
echo "All downloads processed!"
|
|||
|
```
|
|||
|
|