前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >创建ortools的Dockerfile

创建ortools的Dockerfile

原创
作者头像
不知名站长
修改于 2021-04-06 03:44:29
修改于 2021-04-06 03:44:29
1.1K00
代码可运行
举报
文章被收录于专栏:python深度学习python深度学习
运行总次数:0
代码可运行

技术背景

基于已有的Docker容器镜像,去创建一个本地的镜像,有两种方法:一种是在之前的博客中提到过的,使用docker commit的方案,也就是先进去基础系统镜像内部完成所需的修改,然后commit到一个新的容器内部;还有另外一种也非常常用的方法,就是写一个Dockerfile,在本文中会作简单介绍。

另外我们在上一篇博客中介绍了如何部署与使用IBM主导的Cplex线性规划求解器的一些基本使用方法。在本文中我们会介绍另外一套由Google主导的开源线性规划求解器ortools的部署与基本使用方法。

Dockerfile的创建

对于简单的场景而言,尤其是别人已经把基础容器镜像做的比较完善的情况下,使得我们减少了大量的工作量。比如这里我们直接使用一个别人做好的python3.7的基础镜像,而获得该镜像的方法在上一篇博客中也作了介绍。那么我们在dockerfile里面只需要安装好我们所需要的ortools的python包即可:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
3
4
[dechin-root ortools]# cat Dockerfile 
FROM rackspacedot/python37
RUN python3 -m pip install pip --upgrade \
        && python3 -m pip install ortools

这里FROM后面所跟随的基础镜像是必须在本地所具备的,可以在docker images里面看到的才行。最好也在本地通过运行docker run your_iamge来测试一下这个容器镜像是否正常工作,因为有些容器镜像必须要跟随版本号才能正常使用。在上述dockerfile中我们先对pip管理工具做了一个升级,mp3歌曲免费下载然后才安装ortools工具包。有一个需要注意的点是,我们也可以选择使用多次的RUN来制作一个dockerfile,但是这会导致添加了多层的镜像,因此最好我们是可以用命令拼接的方式一次性完成所有的任务,这样只会增加一层镜像(截图来自于参考链接2):

按照上述流程编写好dockerfile之后,我们就可以使用docker build来构建一个本地的容器镜像:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
[dechin-root ortools]# docker build -t dechin/ortools:v1 .
Sending build context to Docker daemon  2.048kB
Step 1/2 : FROM rackspacedot/python37
 ---> ab7083b6c7c4
Step 2/2 : RUN python3 -m pip install pip --upgrade     && python3 -m pip install ortools
 ---> Running in ff6b1971bdc9
Requirement already satisfied: pip in /usr/local/lib/python3.7/site-packages (20.3.1)
Collecting pip
  Downloading pip-21.0.1-py3-none-any.whl (1.5 MB)
Installing collected packages: pip
  Attempting uninstall: pip
    Found existing installation: pip 20.3.1
    Uninstalling pip-20.3.1:
      Successfully uninstalled pip-20.3.1
Successfully installed pip-21.0.1
Collecting ortools
  Downloading ortools-8.2.8710-cp37-cp37m-manylinux1_x86_64.whl (14.2 MB)
Collecting protobuf>=3.14.0
  Downloading protobuf-3.15.6-cp37-cp37m-manylinux1_x86_64.whl (1.0 MB)
Collecting absl-py>=0.11
  Downloading absl_py-0.12.0-py3-none-any.whl (129 kB)
Requirement already satisfied: six in /usr/local/lib/python3.7/site-packages (from absl-py>=0.11->ortools) (1.15.0)
Installing collected packages: protobuf, absl-py, ortools
Successfully installed absl-py-0.12.0 ortools-8.2.8710 protobuf-3.15.6
Removing intermediate container ff6b1971bdc9
 ---> b9ff988385a5
Successfully built b9ff988385a5
Successfully tagged dechin/ortools:v1

我们可以看到2条dockerfile的指令的运行结果都在屏幕上输出,显示是成功安装了的。在执行完build之后,我们可以在本地的images仓库里面看到这个新的容器镜像:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
3
[dechin-root ortools]# docker images
REPOSITORY                                                 TAG       IMAGE ID       CREATED          SIZE
dechin/ortools                                             v1        b9ff988385a5   35 seconds ago   1.09GB

我们也可以测试一下这个容器镜像的功能是否正常:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
3
4
5
6
7
[dechin-root ortools]# docker run -it dechin/ortools:v1 /bin/bash
root@198255eacb30:/# python3
Python 3.7.9 (default, Nov 18 2020, 14:29:12) 
[GCC 6.3.0 20170516] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> import ortools
>>> 

通过执行一个简单的python指令我们可以看到ortools这个工具已经被成功的部署在容器镜像内,在下一个章节中我们会介绍如何使用ortools来解决一个实际问题。当然,上述测试方案也有一个更加简单的方法,使用-c标签来运行代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
[dechin-root ortools]# docker run dechin/ortools:v1 python3 -c "import ortools;print('hello')"
hello

这里再补充介绍一下在docker中如何删除一个容器镜像的方法,那就是使用rmirm指令。这两个指令也容易区分,如果是在docker images指令下找到的容器镜像,那就用rmi来进行删除,如果是在docker ps里面看到的容器,那就用rm来删除,以下是两个示例:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
3
4
5
[dechin-root ortools]# docker rmi cplex-py37
Error response from daemon: conflict: unable to remove repository reference "cplex-py37" (must force) - container 7ce9dbee3e55 is using its referenced image 34e272969701
[dechin-root ortools]# docker rmi -f cplex-py37
Untagged: cplex-py37:latest
Deleted: sha256:34e2729697010b1320c2f7dbfd1fc45004e9ffae6a1d26ffb8748b5627cb2224

上面这个用例是表示我们在docker images中有一个名为cplex-py37的容器镜像,其实也是在上一篇博客中制作的产物。假如我们需要删除这个镜像,就用删除的rmi指令。当我们第一次尝试删除的时候,我们收到一个提示,关于一些冲突的提示。假如我们认定了这个冲突并不影响我们的操作,那么我们可以强制删除,也就是加上-f指令。

另外也展示一下rm指令的使用场景。假如我们使用docker ps -n 5查看过去执行的最近5条指令,并且需要删除第一条镜像id为2df3的容器:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
[dechin-root ortools]# docker ps -n 5
CONTAINER ID   IMAGE                   COMMAND       CREATED        STATUS                      PORTS     NAMES
2df3fcc803e0   34e272969701            "/bin/bash"   20 hours ago   Exited (0) 20 hours ago               bold_colden
c766ed62d149   rackspacedot/python37   "/bin/bash"   20 hours ago   Exited (0) 20 hours ago               xenodochial_ardinghelli
af037db88540   cplex                   "/bin/bash"   21 hours ago   Exited (0) 12 minutes ago             magical_cori
e8c49c211039   cplex                   "/bin/bash"   21 hours ago   Exited (0) 21 hours ago               gracious_babbage
1e053a1b6330   cplex                   "/bin/bash"   21 hours ago   Exited (0) 21 hours ago               suspicious_ride
[dechin-root ortools]# docker rm 2df3
2df3
[dechin-root ortools]# docker ps -n 5
CONTAINER ID   IMAGE                   COMMAND                  CREATED        STATUS                      PORTS     NAMES
c766ed62d149   rackspacedot/python37   "/bin/bash"              20 hours ago   Exited (0) 20 hours ago               xenodochial_ardinghelli
af037db88540   cplex                   "/bin/bash"              21 hours ago   Exited (0) 12 minutes ago             magical_cori
e8c49c211039   cplex                   "/bin/bash"              21 hours ago   Exited (0) 21 hours ago               gracious_babbage
1e053a1b6330   cplex                   "/bin/bash"              21 hours ago   Exited (0) 21 hours ago               suspicious_ride
de7f22ac259b   cplex                   "python3 -m pip inst…"   21 hours ago   Exited (0) 21 hours ago               hardcore_meitner

可以看到用rm删除之后就不会在最新的结果查询中出现这个容器,这也方便我们释放不需要的容器资源给本地环境。

ortools案例

这里我们还是使用上一篇博客中所提到的单背包问题(Knapsack Problem)来进行测试。相关问题的定义如下:

当然在ortools的案例中我们不需要写lp文件,只是借用这个lp文件来展示一下我们的约束条件和目标函数。这个问题的含义也在上一篇博客中介绍过了,这里我们直接截图引用:

ortools求解器的使用

在了解清楚问题的背景之后,现在我们就可以开始写测试代码了,首先我们也是从进入docker容器开始,然后出于方便我们直接在python指令中执行相关的测试(这里的测试代码我们参考了官方文档,也就是本文的参考链接1):

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
[dechin-root ortools]# docker run -it dechin/ortools:v1 /bin/bash
root@3882b83959c8:/# python3
Python 3.7.9 (default, Nov 18 2020, 14:29:12) 
[GCC 6.3.0 20170516] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> from ortools.linear_solver import pywraplp
>>> solver = pywraplp.Solver.CreateSolver('SCIP') # 这里使用了第三方后端SCIP
>>> x1 = solver.IntVar(0.0, 1.0, 'x1')
>>> x2 = solver.IntVar(0.0, 1.0, 'x2')
>>> x3 = solver.IntVar(0.0, 1.0, 'x3')
>>> print ('Number of variables = ', solver.NumVariables()) # 参数数量
Number of variables =  3
>>> solver.Add(3 * x1 + 4 * x2 + 5 * x3 <= 8)
<ortools.linear_solver.pywraplp.Constraint; proxy of <Swig Object of type 'operations_research::MPConstraint *' at 0x7f9013411de0> >
>>> solver.Maximize(2 * x1 + 3 * x2 + 4 * x3)
>>> status = solver.Solve()
>>> print('Number of constraints =', solver.NumConstraints()) # 约束条件数量
Number of constraints = 1
>>> print('Objective value =', solver.Objective().Value()) # 最终解的函数值
Objective value = 6.0
>>> print('x1 =', x1.solution_value())
x1 = 1.0
>>> print('x2 =', x2.solution_value())
x2 = 0.0
>>> print('x3 =', x3.solution_value())
x3 = 1.0
>>> print (status == pywraplp.Solver.OPTIMAL) # 是否最优解?
True

在这个案例中我们使用了一个第三方的求解器后端来进行计算,叫SCIP。我们得到的最终解已经达到了最优解,这个我们在上一篇博客中也分析过了。到这里为止,我们就成功的使用ortools提供的框架求解了一个实际的背包问题。321无损音乐网

总结概要

在本地构建基于Docker的编程环境是一个兼容性和可用性非常强的解决方案,这里我们介绍了一个使用Dockerfile来构建Docker容器镜像的简单实例。同时也用谷歌所主导的开源线性规划求解器ortools来测试这个容器化的编程环境解决方案,最终我们用ortools成功的求解了一个单背包问题,并且跟前面一篇博客中所介绍的IBM主导的cplex一样都得到了问题的最优解。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
作者已关闭评论
暂无评论
推荐阅读
进程组、会话、控制终端概念,如何创建守护进程?
守护进程,也就是通常所说的Daemon进程,是Linux中的后台服务进程。周期性的执行某种任务或等待处理某些发生的事件。
睡魔的谎言
2020/11/25
1.5K0
什么是守护进程?
在了解守护进程之前,需要先知道什么是什么是终端?什么是作业?什么是进程组?什么是会话?
全栈程序员站长
2022/09/07
1.1K0
Linux内核编程--进程组和守护进程
进程组:进程组是多个进程的集合, 接收同一个终端的各类信号信息。进程调用setpgid(pid, pgid)可以加入一个现有的进程组或者创建一个新的进程组。
Coder-ZZ
2022/05/09
3K0
Linux内核编程--进程组和守护进程
Linux 守护进程|应急响应
通常我们都是通过以上两种方式来获得一个shell,之后运行程序的,此时我需要纠正一个概念,我们通常都说获得一个shell,本质上来说,我们获取了一个session(会话,以下session都是会话)
意大利的猫
2021/03/18
3.9K0
Linux 守护进程|应急响应
linux系统编程之进程(五):终端、作业控制与守护进程
该文介绍了如何在Linux系统中通过fork函数创建守护进程,并给出了具体的示例代码。同时,文章还介绍了守护进程的一些常见用途,如保证程序在后台运行、处理控制台输入输出等。
s1mba
2018/01/03
2.7K0
linux系统编程之进程(五):终端、作业控制与守护进程
【计算机网络】日志与守护进程
一般使用cout进行打印,但是cout打印是不规范的 实际上 是采用日志进行打印的
lovevivi
2023/11/17
1820
【计算机网络】日志与守护进程
【在Linux世界中追寻伟大的One Piece】进程间关系与守护进程
其实每一个进程除了有一个进程ID(PID)之外,还属于一个进程组。进程组是一个或者多个进程的集合, 一个进程组可以包含多个进程。 每一个进程组也有一个唯一的进程组ID(PGID), 并且这个 PGID类似于进程ID, 同样是一个正整数, 可以存放在pid_t数据类型中。
枫叶丹
2024/09/24
610
【在Linux世界中追寻伟大的One Piece】进程间关系与守护进程
Linux守护进程
进程组,也叫做作业。BSD于1980年前后向Unix中增加的一个新特性,代表一个或者多个进程的集合,每个进程都属于一个进程组。操作系统设计进程组的概念主要就是为了简化对多个进程的管理。
mindtechnist
2024/08/08
1930
Linux守护进程
守护进程「建议收藏」
在UNIX系统中, 用户通过终端登录系统后得到一个Shell进程, 这个终端成为Shell进程的控制终端(Controlling Terminal), 进程中, 控制终端是保存在PCB中的信息, 而fork会复制PCB中的信息, 因此由Shell进程启动的其它进程的控制终端也是这个终端. 默认情况下(没有重定向), 每个进程的标准输入, 标准输出和标准错误输出都指向控制终端, 进程从标准输入读也就是读用户的键盘输入, 进程往标准输出或标准错误输出写也就是输出到显示器上. 信号中还讲过, 在控制终端输入一些特殊的控制键可以给前台进程发信号, 例如Ctrl-C表示SIGINT,Ctrl-\表示SIGQUIT。
全栈程序员站长
2022/09/16
5980
13(守护进程)
守护进程是一种纯粹的后台进程,与运行前环境完全隔离,包括未关闭的文件描述符、控制终端、会话、进程组、工作目录以及文件创建掩码等 很多守护进程是父进程 fork 产生,所以会继承所有的父进程地址空间中的环境,所以必须在守护进程诞生之初,断绝这些相关环境,当然,守护进程也可以在 linux 系统启动时从启动脚本 /etc/rc.d 中启动,也可以由 crontab 启动 事实上,守护进程与普通进程的编写并没有特别大的区别
提莫队长
2019/02/21
8030
Python守护进程daemon实现
守护进程是系统中生存期较长的一种进程,常常在系统引导装入时启动,在系统关闭时终止,没有控制终端,在后台运行。守护进程脱离于终端是为了避免进程在执行过程中的信息在任何终端上显示并且进程也不会被任何终端所产生的终端信息所打断。 在这里,我们在Linux2.6内核的centos中,ps -ef |awk '{print $1"\t "$2"\t "$3"\t  "$8}'看到:PPID=0的进程有两个,分别是PID=1的/sbin/init进程和PID=2的[kthreadd]进程。
py3study
2020/01/07
7.7K0
Linux守护进程
守护进程在 Linux 系统中极为重要,它们是许多服务器的核心组成部分,例如 Internet 服务器 inetd 和 Web 服务器 httpd。这些进程不仅负责提供网络服务,还执行各种系统任务,例如作业调度进程 crond。
不脱发的程序猿
2024/11/26
1770
Linux守护进程
Linux - 请允许我静静地后台运行
枕边书
2018/01/04
1.7K0
Linux - 请允许我静静地后台运行
守护进程
在Linux中,session(会话)通常指的是与用户交互的一个环境,它是系统中与某个用户交互的一系列活动的集合。会话在Linux系统中有多种用途,下面是几种常见的会话类型及其相关概念:
ljw695
2025/01/03
780
守护进程
python 守护进程(daemon)
通常,我们执行服务端程序的时候都会通过终端连接到服务器,成功连接后会加载shell环境,终端盒shell都是进程,shell进程是终端进程的子进程,通过ps命令可以很容易的查看到,在这个shell环境下一开始执行的程序都是shell进程的子进程,自然会受到shell进程的影响,在程序里fork子进程后,父进程退出,对于shell进程来说,这个父进程就算执行完毕,而产生的子进程会被init进程接管,从而也就脱离了终端控制。
py3study
2020/01/10
1.1K0
守护进程Xinted和日志记录Syslogd
调用fork函数创建子进程后,使父进程立即退出。这样,产生的子进程将变成孤儿进程,并被init进程接管,同时,所产生的新进程将变为在后台运行。
星哥玩云
2022/07/04
8920
Python实现守护进程
專 欄 ❈汤英康,Python程序员,负责设计和开发大数据监控平台的相关产品。 PyCon China2016 深圳 讲师。 博客:http://blog.tangyingkang.com/ ❈— Daemon场景 考虑如下场景:你编写了一个python服务程序,并且在命令行下启动,而你的命令行会话又被终端所控制,python服务成了终端程序的一个子进程。因此如果你关闭了终端,这个命令行程序也会随之关闭。 要使你的python服务不受终端影响而常驻系统,就需要将它变成守护进程。 守护
Python中文社区
2018/01/31
2K0
PHP中的会话
2、当执行php xxx.php 时,默认系统会把当前的进程设置为会话首进程(使用strace查看),所以当前会话首进程不能使用posix_setsid 创建为会话首进程,只能使用子进程调用此函数
北溟有鱼QAQ
2021/06/08
1.2K0
Linux之守护进程理解(2)
1、屏蔽一些有关控制终端操作的信号 防止在守护进程没有正常运转起来时,控制终端受到干扰退出或挂起。 2、脱离控制终端,登录会话和进程组 登录会话可以包含多个进程组,这些进程组共享一个控制终端,这个控制终端通常是创建进程的登录终端。控制终端,登录会话和进程组通常是从父进程继承下来的。我们的目的就是要摆脱它们,使之不受它们的影响。 其方法是在fork()的基础上,调用setsid()使进程成为会话组长。调用成功后,进程成为新的会话组长和新的进程组长,并与原来的登录会话和进程组脱离,由于会话过程对控制终端的独占性,进程同时与控制终端脱离。 setsid()实现了以下效果: (a) 成为新对话期的首进程 (b) 成为一个新进程组的首进程 (c) 没有控制终端。 3、禁止进程重新打开控制终端 现在,进程已经成为无终端的会话组长,但它可以重新申请打开一个控制终端。可以通过使进程不再成为会话组长来禁止进程重新打开控制终端,再fork()一次。 4、关闭打开的文件描述符 进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,造成进程所在地文件系统无法卸下以及无法预料的错误。一般来说, 必要的是关闭0、1、2三个文件描述符,即标准输入、标准输出、标准错误。因为我们一般希望守护进程自己有一套信息输出、输入的体系,而不是把所有的东西 都发送到终端屏幕上。 5、改变当前工作目录 将当前工作目录更改为根目录。从父进程继承过来的当前工作目录可能在一个装配的文件系统中。因为守护进程通常在系统重启之前是一直存在的,所以如果守护进程的当前工作目录在一个装配文件系统中,那么该文件系统就不能被拆卸。 另外,某些守护进程可能会把当前工作目录更改到某个指定位置,在此位置做它们的工作。例如,行式打印机假脱机守护进程常常将其工作目录更改到它们的spool目录上。 6、重设文件创建掩码 将文件方式创建屏蔽字设置为0:umask(0)。 由继承得来的文件方式创建的屏蔽字可能会拒绝设置某些许可权。例如,若守护进程要创建一个组可读、写的文件,而继承的文件方式创建屏蔽字,屏蔽了这两种许可权,则所要求的组可读、写就不能起作用。 7、处理SIGCHLD信号 处理SIGCHLD信号并不是必须的。但对于某些进程, 特别是服务器进程往往在请求到来时fork子进程出来处理请求。如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)而仍占用系统资源。如 果父进程等待子进程结束,将增加父进程的负担,影响服务器进程的并发性能。在系统V下可以简单地将SIGCHLD信号的操作设为SIG_IGN,即忽略掉。这样,内核在子进程结束时不会产生僵尸进程,这一点与BSD4不同,在BSD4下必须显示等待子进程结束才能释放僵尸进程。 8、记录信息 在Linux/Unix下有个syslogd的守护进程,向用户提供了syslog()系统调用。任何程序都可以通过syslog记录事件。  源码实现及分析:
chain
2018/08/02
2.6K0
守护进程
守护进程(daemon)是一类在后台运行的特殊进程,用于执行特定的系统任务。很多守护进程在系统引导的时候启动,并且一直运行直到系统关闭。另一些只在需要的时候才启动,完成任务后就自动结束。
zy010101
2020/09/02
1.9K0
相关推荐
进程组、会话、控制终端概念,如何创建守护进程?
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文