问题描述


在服务器上提交任务时,需要限制运行的核的数目。程序本身是单线程的,但是不同的输入参数需要跑很多组,粗暴的方法是开多个终端,不断地去提交任务。但这比较麻烦,可以用 shell 实现。

基础


首先看第一种 shell 写法

 #! /bin/bash

 start=`date +%s`

 for i in {..}
do
echo sucess
sleep
done end=`date +%s`
echo "time: `expr $end - $start`"
  • 如何提取当前时间,date +%s,返回值是从 1970 年至今的秒数。
  • for 循环的写法。
  • shell 中数学运算,注意 start 与 end 之间需要空格。

该程序的运行结果是 8 s,如果放在后台运行,则只需要 2 s,代码如下

 #! /bin/bash

 start=`date +%s`

 for i in {..}
do
{
echo sucess
sleep
} &
done
wait end=`date +%s`
echo "time: `expr $end - $start`"
  • 需要使用 {} 将主执行程序包括,用 & 将其放入后台。
  • 需要 wait 命令等待所有命令执行完成,不然系统会直接继续往下走。

这样做实现了多线程并发,但是问题是不能控制使用的进程数。因此引入了管道和文件操作符。

管道

  • 比如 hexdump Run0035.bin | less 中的 | 就是管道,用于连接两个命令间的数据。
  • 管道的特点是,如果管道中没有数据,那么取管道数据的操作就会停滞,直到管道中有数据;同样写管道的操作也是如此,如果没有读取操作,那么写操作也会停滞。意思就是管道需要既有读操作,也有写操作。
  • 使用 mkfifo 命令可以创建有名管道

文件操作符

  • 通过文件操作符,可以控制 linux 系统中 /proc 文件夹下的内容,从而控制 linux 中进程的运行。

代码


 1 #!/usr/bin/env bash

 #设置变量
beta=0.25
gammaList=(22.0 24.0 26.0 28.0 30.0 32.0 34.0 36.0 38.0)
moiList=(4.0 5.0 6.0 7.0 8.0 9.0 10.0 11.0 12.0 13.0 14.0 15.0 16.0 17.0 18.0 19.0 20.0 21.0 22.0 23.0 24.0 25.0 26.0 27.0 28.0 29.0) start=`date +%s`
thread_num= trap "exec 1000>&-;exec 1000<&-;exit 0"
tmpfifo=$$.fifo
mkfifo $tmpfifo
exec <>$tmpfifo
rm -rf $tmpfifo for (( i=; i<$thread_num; i++ ))
do
echo >&
done for gamma in ${gammaList[@]}
do
if [ ! -d "gamma"$gamma ]; then
mkdir gamma$gamma
fi
outfile=`printf "terminal-out-%s" "$gamma"`
for moi in ${moiList[@]}
do
read -u1000
{
sh auto-run.sh $beta $gamma $moi > $outfile
echo >&
}&
done
done wait
end=`date +%s`
echo "times: `expr $end - $start`"
  • 第 9 行,设置使用进程数.
  • 第 11 行,表示脚本在运行过程中,如果接收到 ctrl+c,则则关闭文件描述符 1000 的读写,并正常退出。其中 exec >1000&- 代表关闭对文件描述符 1000 的写,exec <1000&- 代表关闭对文件描述符 1000 的读,trap 用于接受中断命令,2 代表 ctrl+c.
  • 12 ~ 14 行,创建一个管道文件 PID.fifo,并与文件描述符 1000 做读写的绑定。如此一来,对文件描述符 1000 的操作就相当于对管道文件的操作,且该文件描述符又有管道的特性,即读写操作必须同时存在,否则会停滞。
  • 第 15 行,将管道文件删除(不明白)
  • 17 ~ 20 行,向文件描述符 1000 写入相应进程数的空行,用于控制进程数。
  • 22 ~ 36 行,主体部分。第 30 行,从文件描述符 1000 读取 1 行,第 31 ~ 34 行,后台执行其中的命令,当第 32 行执行结束后,再向文件描述符 1000 中写入一个空行。结果就是,先读取了文件描述符 1000 中所有的行,读取完后因为没有写入,由于管道性质,该行为被阻塞,因此不会再读取,即不会再执行 read -u1000,但后台若有第 32 行的命令执行结束,便会执行第 33 行的写入命令,写入后 read -u1000 操作便不会阻塞,于是每当跑完一个后台,便可以提交下一个后台。
  • 38 ~ 40 行,等待所有命令执行完成,输出耗时。

shell 设置进程数运行的更多相关文章

  1. curl命令,curl实现post,curl监控网页shell脚本,curl多进程实现并控制进程数,

    cURL > Docs > Tutorial:  http://curl.haxx.se/docs/httpscripting.html 下载单个文件,默认将输出打印到标准输出中(STDO ...

  2. Shell 脚本进程并发&进程数控制

    Shell 都以串行的方式自上而下执行命令,不适用需要大量作业的场景. 学习此篇shell脚本进程并发,能够大大提高工作效率~ 通过wait 和 & 后台符号 可以实现并行,但无法控制进程数. ...

  3. IIS应用程序池最大进程数设置

    1.当工作进程数>1时,如果有多个连接请求就会启动多个工作进程实例来处理,所启动的最多工作进程数就是你设置的最大进程数,后续更多的连接请求会循环的发送至不同的工作进程来处理.每个工作进程都能承担 ...

  4. oracle连接进程数设置

    SQL> select count(*) from v$session #连接数SQL> Select count(*) from v$session where status='ACTI ...

  5. linux shell编程,先等10秒再判断是否有进程存在,存在就再等10秒再杀了进程才运行

    linux shell编程,先等10秒再判断是否有进程存在,存在就再等10秒再杀了进程才运行 crontab每分钟执行一次,但5秒以上才有更新数据,有时候一分钟可能跑不完上一个进程,需要先等10秒再判 ...

  6. linux最大进程数、最大打开文件数

    ulimit 是一种 linux 系统的内键功能,它具有一套参数集,用于为由它生成的 shell 进程及其子进程的资源使用设置限制.本文将在后面的章节中详细说明 ulimit 的功能,使用以及它的影响 ...

  7. Linux 进程资源用量监控和按用户设置进程限制

    每个 Linux 系统管理员都应该知道如何验证硬件.资源和主要进程的完整性和可用性.另外,基于每个用户设置资源限制也是其中一项必备技能. 在这篇文章中,我们会介绍一些能够确保系统硬件和软件正常工作的方 ...

  8. linux上限制用户进程数、cpu占用率、内存使用率

    限制进程CPU占用率的问题,给出了一个shell脚本代码如下: renice +10 `ps aux | awk '{ if ($3 > 0.8 && id -u $1 > ...

  9. shell学习三十八天----运行顺序和eval

    运行顺序和eval shell从标准输入或脚本中读取的每一行称为管道,它包括了一个或多个命令,这些命令被一个或多个管道字符(|)隔开. 其实嗨哟非常多特殊符号可用来切割单个的命令:分号(;),管道(| ...

随机推荐

  1. JavaScript map+parseInt 容易产生的误区

    map /** * 语法: * var new_array = arr.map(function callback(currentValue[,index[,array]]){ * // return ...

  2. HTML基础-04

    定位 定位:通过定位可以将元素摆放在页面中任意位置 语法:position属性设置元素的定位 可选值:static:默认值,开启定位 relative开启相对定位 absolute开启绝对定位 fix ...

  3. java的线程、创建线程的 3 种方式、静态代理模式、Lambda表达式简化线程

    0.介绍 线程:多个任务同时进行,看似多任务同时进行,但实际上一个时间点上我们大脑还是只在做一件事情.程序也是如此,除非多核cpu,不然一个cpu里,在一个时间点里还是只在做一件事,不过速度很快的切换 ...

  4. windows下的Redis安装:

    windows下的Redis安装: 百度网盘地址:https://pan.baidu.com/s/1yYED2pXLWolPXvWaABtF2Q 提取密码:xshu 1.解压文件并且创建start.b ...

  5. python中操作csv文件

    python中操作csv文件 读取csv improt csv f = csv.reader(open("文件路径","r")) for i in f: pri ...

  6. storcli 命令(更新Ing)

    help [root@centos7]# storcli -h Storage Command Line Tool Ver 007.0606.0000.0000 Mar , (c)Copyright ...

  7. 金题大战Vol.0 A、凉宫春日的叹息

    金题大战Vol.0 A.凉宫春日的叹息 题目描述 给定一个数组,将其所有子区间的和从小到大排序,求第 \(k\) 小的是多少. 输入格式 第一行两个数\(n\),$ k\(,表示数组的长度和\)k$: ...

  8. xshell乱码解决办法

    https://jingyan.baidu.com/article/7908e85c758a58af481ad2ae.html

  9. 浅析XML和JSON的区别

    前言 今天做接口对接时,发现对方竟然是通过XML进行数据传输,当时冒出的第一个想法就是:WTF,这都什么年代了,还在用XML,是来搞笑的吧,JSON它不香吗? 想法归想法,但对接还是要完成的是吧?然后 ...

  10. java应用中的日志介绍

    日志在应用程序中是非常非常重要的,好的日志信息能有助于我们在程序出现 BUG 时能快速进行定位,并能找出其中的原因. 但是,很多介绍 AOP 的地方都采用日志来作为介绍,实际上日志要采用切面的话是极其 ...