这篇文章将为大家详细讲解有关Python中虚拟环境virtualenv,pipreqs如何生成项目依赖第三方包,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。virtualenv简介含义:virtual:虚拟,env:environment环境的简写,所以virtualenv就是虚拟环境,顾名思义,就是虚拟出来的一个新环境,比如我们使用的虚拟机、docker,它们都是把一部分的内容独立出来,这部分独立的内容相当于一个容器,在这个容器只呢个,我们可以“为所欲为”----安装需要的依赖包,软件..,同时这个容器是与外界相互独立的,容器与容器直接也是互相独立不影响。为何要用虚拟环境:【前提概要】Django也是一个非常流行的web框架。由于Django的迭代更新非常快,也比较频繁,所以有一些过时的东西需要丢弃掉,一些新的东西需要加进来,从而导致不同的版本之间不兼容。比如Django1.3、Django1.4、Django1.8之间就有很大的差异性。或者是说,以Python的版本举例,现在工作中使用的Python版本与Python2.x和Python3.x两种。【故事背景】假设要进行Python web开发,使用的是Django。手上还有两个老项目A和B需要维护,而新项目C也正在开发中。这里项目A使用的是django1.3,项目B使用的是django1.4,而新项目C使用的是Django1.8。那么问题来了,如何同时在本地进行ABC这三个项目的开发和维护?正常的模式可能是这样:现在在A项目上有一个BUG需要修复,于是,先执行下面的命令,删除掉原来的版本:pip3 uninstall django 然后再执行下面的命令安装django1.3pip3 install django==1.3数分钟后,bug修复完毕,好,现在进行新项目C的开发了,然后又要重复上面的故事。好了,这还是最理想的情况。最不理想的情况就是基于django的第三方依赖也是跟Django版本相关的,于是除了install和uninstall Django之外,还要uninstall和install其依赖,Orz,这特么的就尴尬了...VirtualEnv能做什么呢?virtualenv可以搭建虚拟且独立的Python运行环境,使得单个项目的运行环境与其它项目独立起来。同时也可以用于在一台机器上创建多个独立的Python运行环境,VirtualEnvWrapper为前者提供了一些便利的命令行上的封装。virtualenv是一个非常不错的Python虚拟环境的创建工具,它最大的好处在于:可以让每个Python项目单独使用一个环境,而不会影响系统环境,也不会影响其它项目的环境。virtualenv可以用于创建独立的Python环境,在这些环境里面可以选择不同的Python版本或者不同的Packages,并且可以在没有root权限的情况下在环境里面重新安装新套件,互相不会产生任何影响,我们可以简单的认为虚拟环境就是一个沙箱系统,我们可以在里面“为所欲为 ”情况下还不会影响外面。virtualenv安装和使用virtualenv本质上是个python包, 使用pip安装:pip3 install virtualenv在工作目录下创建虚拟环境(默认在当前目录):注意需要自定义虚拟环境的名字!创建虚拟环境:cmd命令创建(创建目录为cmd命令当前目录,所以最好切换到所需要的目录下在进行创建)(1) virtualenv env_t1 (创建纯净环境)(2) virtualenv --system-site-packages env_t2 (创建环境,继承原安装的模块)创建虚拟环境有2中模式,一个是纯净模式,一个是基于系统环境的创建(自带了系统装的所有模块)-# 基于系统环境的创建模式,里面的模块其实是对系统的模块的一个引用,在虚拟环境中是卸载不掉的,因为它只是一种引用关系,除非系统卸载的模块,它才会没了。-# 纯净模式,里面的不包含系统的模块,需要自己下载安装,从头开始,所以这是在虚拟环境下安装的,你可以在后期卸载。一般来说,推荐使用纯净模式创建虚拟环境,这样根据自己的需求去安装第三方模块,同时如果想卸载的话也可以在虚拟环境中选择性的卸载。这样自由度更大,而不像另外一种模式,系统里有啥它都引用过来,而且还在虚拟环境中删除不了,影响加载速度等。使用pip3 list 来查看当前环境下安装的模块列表激活该虚拟环境:-windows进到目录里,的Script文件夹输入:activate-linux:soruse env1/Script/activate退出虚拟环境:-deactivate在pycharm中使用虚拟环境:-files--settings--Project--Project Interpreter--add选择虚拟环境路径下的python.exe即可三、pipreqs模块的介绍和使用当运行别人项目的时候,首先需要安装项目的依赖,不然就无法运行,Python的pipreqs模块就给我们提供了自动生成项目依赖内容的功能安装:pip3 install pipreqs使用:# 然后在项目所在的目录下使用命令生成包含依赖内容的文件pipreqs ./如果出现编码格式错误:UnicodeDecodeError: 'gbk' codec can't decode byte 0x80 in position 776: illegal multibyte sequence解决方法在命令后面指定编码格式:pipreqs ./ --encoding=utf8如果已经生成过一次依赖包文件requirements.txt,但是又项目新增了一下模块,此时再在后面添加--force覆盖参数重写即可pipreqs ./ --encoding='utf8' --force拿到依赖包文件我们可以快速自动下载依赖一键安装:pip3 install -r requirements.txttips:导出所有已安装的第三方模块:pip3 freeze > package.txt关于“Python中虚拟环境virtualenv,pipreqs如何生成项目依赖第三方包”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。...
这篇文章主要讲解了“Linux下怎么搭建NFS”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Linux下怎么搭建NFS”吧!一、NFS简介NFS是Network File System的缩写,即网络文件系统。一种使用于分散式文件协定,有SUN公司开发。功能是通过网络让不同的机器、不同的操作系统能够分享个人数据,让应用程序通过网络可以访问位于服务器磁盘中的数据。NFS在文件传送或信息传送的过过程中,依赖于RPC协议。RPC,远程过程调用(Remote Procedure Call),是使客户端能够执行其他系统中程序的一种机制。NFS本身是没有提供信息传输的协议和功能的,但NFS却能让我们通过网络进行资料的分享,就是因为NFS使用了RPC提供的传输协议,可以说NFS就是使用PRC的一个程序。环境说明:系统 :CentOS release 6.7 (Final)NFS IP :10.219.24.22web IP :10.219.24.25iptables:未运行防火墙操作流程:1、软件安装,NFS只需要安装两个软件,在通常情况下是作为系统默认软件安装的[root@mysql01 ~]# rpm -qa|grep rpcbind*rpcbind-0.2.0-11.el6.x86_64[root@mysql01 ~]# rpm -qa|grep nfs-utilsnfs-utils-1.2.3-64.el6.x86_64nfs-utils-lib-1.1.5-11.el6.x86_64说明:[root@mysql01 ~]# yum install rpcbind nfs-utils (如果没有安装就yum安装)rpcbind包 :centos 下面RPC主程序nfs-utils包 :NFS服务主程序,包括NFS的基本命令和监控程序2、开启RCP服务[root@mysql01 ~]# /etc/init.d/rpcbind start查看rpcbind服务端口 [root@mysql01 ~]# netstat -antlp|grep rpcbindtcp 0 0 0.0.0.0:111 0.0.0.0:* LISTEN 1462/rpcbind tcp 0 0 :::111 :::* LISTEN 1462/rpcbind [root@mysql01 ~]# rpcinfo -p localhostprogram vers proto port service100000 4 tcp 111 portmapper100000 3 tcp 111 portmapper100000 2 tcp 111 portmapper100000 4 udp 111 portmapper100000 3 udp 111 portmapper100000 2 udp 111 portmapper100024 1 udp 41757 status100024 1 tcp 42661 status3、开启NFS服务[root@mysql01 ~]# /etc/init.d/nfs start Starting NFS services: [ OK ]Starting NFS quotas: [ OK ]Starting NFS mountd: [ OK ]Starting NFS daemon: [ OK ]Starting RPC idmapd: [ OK ][root@mysql01 ~]# rpcinfo -p localhostprogram vers proto port service......100005 3 tcp 59454 mountd100003 2 tcp 2049 nfs100003 3 tcp 2049 nfs100003 4 tcp 2049 nfs100227 2 tcp 2049 nfs_acl100227 3 tcp 2049 nfs_acl100003 2 udp 2049 nfs100003 3 udp 2049 nfs100003 4 udp 2049 nfs100227 2 udp 2049 nfs_acl100227 3 udp 2049 nfs_acl....100021 4 tcp 42905 nlockmgr说明: rpc上应该能看到好多新被注册的nfs端口两个服务开机自启动[root@mysql01 ~]# chkconfig rpcbind on[root@mysql01 ~]# chkconfig nfs on4、服务端配置共享目录(/oradata)4.1 配置前确认rpcbind、nfs服务进程正常[root@nfs01 ~]# /etc/init.d/nfs status[root@nfs01 ~]# /etc/init.d/rpcbind status[root@nfs01 ~]# ps -ef|egrep "rpc|nfs"4.2 创建共享目录并授权("nfsnobody")#nfsnobody 用户是开启rpc、nfs进程后系统自动创建的[root@mysql01 ~]# df -hFilesystem Size Used Avail Use% Mounted on/dev/sda3 35G 8.9G 25G 27% /tmpfs 491M 76K 491M 1% /dev/shm/dev/sda1 2.9G 4.5M 2.7G 1% /tmp/dev/sr0 3.7G 3.7G 0 100% /media/CentOS_6.7_Final[root@mysql01 ~]# mkdir /oradata[root@mysql01 ~]# chown -R nfsnobody.nfsnobody /oradata [root@mysql01 ~]# chmod 777 /oradata4.3 修改服务端配置文件(/etc/exports)[root@mysql01 ~]# cat /etc/exports #share /data by andy for XXXX at 20170812/oradata 10.219.24.0/8(rw,sync)[root@mysql01 ~]#4.3 查看系统加载的配置[root@mysql01 ~]# cat /var/lib/nfs/etab /oradata 10.219.24.0/8(rw,sync,wdelay,hide,nocrossmnt,secure,root_squash,no_all_squash,no_subtree_check,secure_locks,acl,anonuid=65534,anongid=65534,sec=sys,rw,root_squash,no_all_squash)4.4 重新平滑加载服务[root@mysql01 ~]# showmount -e localhost5、客户端配置[root@mysql02 ~]# /etc/init.d/rpcbind statusrpcbind (pid 1469) is running...[root@mysql02 ~]# /etc/init.d/rpcbind start挂载nfs共享目录[root@mysql02 ~]# mount -t nfs 10.219.24.22:/oradata /mnt[root@mysql02 ~]# df -hFilesystem Size Used Avail Use% Mounted on/dev/sda3 35G 8.7G 25G 26% /tmpfs 491M 76K 491M 1% /dev/shm/dev/sda1 2.9G 4.5M 2.7G 1% /tmp/dev/sr0 3.7G 3.7G 0 100% /media/CentOS_6.7_Final10.219.24.22:/oradata35G 8.9G 25G 27% /mnt6、开机自动挂载方法一:[root@mysql02 ~]# echo "mount -t nfs 10.219.24.22:/oradata /mnt">>/etc/rc.local方法二:开启netfs服务[root@mysql02 ~]# chkconfig netfs on[root@mysql02 ~]# vi /etc/fstab[root@mysql02 ~]# cat /etc/fstab10.219.24.22:/oradata /mnt nfs defaults 0 07、检验-- nfs server[root@mysql01 ~]# touch /oradata/aa-- client[root@mysql02 ~]# cd /mnt[root@mysql02 mnt]# lltotal 0-rw-r--r--. 1 root root 0 Aug 13 04:31 aa感谢各位的阅读,以上就是“Linux下怎么搭建NFS”的内容了,经过本文的学习后,相信大家对Linux下怎么搭建NFS这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是辰迅云,小编将为大家推送更多相关知识点的文章,欢迎关注!...
小编给大家分享一下如何实现判断颜色是否合法的正则表达式,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!"^#([0-9a-fA-F]{6}|[0-9a-fA-F]{3})$";意思是:以#开头,后面是数字和a-f的字符(大写或小写),这个值是6位或3位。要匹配一个3为是为了符合css颜色的简写规则:"#abc"=="#aabbcc"注意:如果需要进行16位和10位的转换,比如将颜色值转成int存在数据库,如果是6位的颜色没问题,如果是3位的颜色就有问题了,因为当你取回来从10进制转为16进制的时候,你不知道他应该是3位还是6位。比如:#0000ff==转10进制==>255。如果是转回来呢:255==转16进制==>ff,如果你知道是6位你可以自己加上“#0000”就变成“#0000ff”,如果是三位就变成“#0ff”因此,如果是需要转10进制保存的情况下,不应该让3位的颜色值存在,或者应该在保存之前给他统一转成6位的颜色值再转10进制。以上是“如何实现判断颜色是否合法的正则表达式”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注辰迅云资讯频道!...
这篇文章给大家分享的是有关php中curl简单采集图片如何生成base64编码的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。示例分析<?php $url="http://c.hiphotos.baidu.com/image/w%3D210/sign=ed30880babec8a13141a50e1c7029157/d52a2834349b033be1a9503e17ce36d3d539bd35.gif";function curl_url($url,$type=0,$timeout=30){ $msg = ['code'=>2100,'status'=>'error','msg'=>'未知错误!']; $imgs= ['image/jpeg'=>'jpeg', 'image/jpg'=>'jpg', 'image/gif'=>'gif', 'image/png'=>'png', 'text/html'=>'html', 'text/plain'=>'txt', 'image/pjpeg'=>'jpg', 'image/x-png'=>'png', 'image/x-icon'=>'ico' ]; if(!stristr($url,'http')){ $msg['code']= 2101; $msg['msg'] = 'url地址不正确!'; return $msg; } $dir= pathinfo($url); //var_dump($dir); $host = $dir['dirname']; $refer= $host.'/'; $ch = curl_init($url); curl_setopt ($ch, CURLOPT_REFERER, $refer); //伪造来源地址 curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);//返回变量内容还是直接输出字符串,0输出,1返回内容 curl_setopt($ch, CURLOPT_BINARYTRANSFER, 1);//在启用CURLOPT_RETURNTRANSFER的时候,返回原生的(Raw)输出 curl_setopt($ch, CURLOPT_HEADER, 0); //是否输出HEADER头信息 0否1是 curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, $timeout); //超时时间 $data = curl_exec($ch); //$httpCode = curl_getinfo($ch,CURLINFO_HTTP_CODE); //$httpContentType = curl_getinfo($ch,CURLINFO_CONTENT_TYPE); $info = curl_getinfo($ch); curl_close($ch); $httpCode = intval($info['http_code']); $httpContentType = $info['content_type']; $httpSizeDownload= intval($info['size_download']); if($httpCode!='200'){ $msg['code']= 2102; $msg['msg'] = 'url返回内容不正确!'; return $msg; } if($type>0 && !isset($imgs[$httpContentType])){ $msg['code']= 2103; $msg['msg'] = 'url资源类型未知!'; return $msg; } if($httpSizeDownload<1){ $msg['code']= 2104; $msg['msg'] = '内容大小不正确!'; return $msg; } $msg['code'] = 200; $msg['status']='success'; $msg['msg'] = '资源获取成功'; if($type==0 or $httpContentType=='text/html') $msg['data'] = $data; $base_64 = base64_encode($data); if($type==1) $msg['data'] = $base_64; elseif($type==2) $msg['data'] = "data:{$httpContentType};base64,{$base_64}"; elseif($type==3) $msg['data'] = "<img src='data:{$httpContentType};base64,{$base_64}' />"; else $msg['msg'] = '未知返回需求!'; unset($info,$data,$base_64); return $msg; } $msg = curl_url($url,3);if($msg['status']=='success') echo $msg['data'];die;?>输出效果如下:下面是部分curl函数参数说明:CURL库方法名称 说明curl_close 关闭一个curl会话curl_copy_handle 拷贝一个curl连接资源的所有内容和参数curl_errno 返回一个包含当前会话错误信息的数字编号curl_error 返回一个包含当前会话错误信息的字符串curl_exec 执行一个curl会话curl_getinfo 获取一个curl连接资源句柄的信息curl_init 初始化一个curl会话curl_multi_add_handle 向curl批处理会话中添加单独的curl句柄资源curl_multi_close 关闭一个批处理句柄资源curl_multi_exec 解析一个curl批处理句柄curl_multi_getcontent 返回获取的输出的文本流curl_multi_info_read 获取当前解析的curl的相关传输信息curl_multi_init 初始化一个curl批处理句柄资源curl_multi_remove_handle 移除curl批处理句柄资源中的某个句柄资源curl_multi_select Get all the sockets associated with the cURL extension, which can then be "selected"curl_setopt_array 以数组的形式为一个curl设置会话参数curl_setopt 为一个curl设置会话参数curl_version 获取curl相关的版本信息curl_setopt()参数名称 说明CURLOPT_INFILESIZE 当你上传一个文件到远程站点,这个选项告诉PHP你上传文件的大小。CURLOPT_VERBOSE 如果你想CURL报告每一件意外的事情,设置这个选项为一个非零值。CURLOPT_HEADER 如果你想把一个头包含在输出中,设置这个选项为一个非零值。CURLOPT_NOPROGRESS 如果你不会PHP为CURL传输显示一个进程条,设置这个选项为一个非零值。注意:PHP自动设置这个选项为非零值,你应该仅仅为了调试的目的来改变这个选项。CURLOPT_NOBODY 如果你不想在输出中包含body部分,设置这个选项为一个非零值。CURLOPT_FAILONERROR 如果你想让PHP在发生错误(HTTP代码返回大于等于300)时,不显示,设置这个选项为一人非零值。默认行为是返回一个正常页,忽略代码。CURLOPT_UPLOAD 如果你想让PHP为上传做准备,设置这个选项为一个非零值。CURLOPT_POST 如果你想PHP去做一个正规的HTTP POST,设置这个选项为一个非零值。这个POST是普通的 application/x-www-from-urlencoded 类型,多数被HTML表单使用。CURLOPT_FTPLISTONLY 设置这个选项为非零值,PHP将列出FTP的目录名列表。CURLOPT_FTPAPPEND 设置这个选项为一个非零值,PHP将应用远程文件代替覆盖它。CURLOPT_NETRC 设置这个选项为一个非零值,PHP将在你的 ~./netrc 文件中查找你要建立连接的远程站点的用户名及密码。CURLOPT_FOLLOWLOCATION 设置这个选项为一个非零值(象 “Location: “)的头,服务器会把它当做HTTP头的一部分发送(注意这是递归的,PHP将发送形如 “Location: “的头)。CURLOPT_PUT 设置这个选项为一个非零值去用HTTP上传一个文件。要上传这个文件必须设置CURLOPT_INFILE和CURLOPT_INFILESIZE选项.CURLOPT_MUTE 设置这个选项为一个非零值,PHP对于CURL函数将完全沉默。CURLOPT_TIMEOUT 设置一个长整形数,作为最大延续多少秒。CURLOPT_LOW_SPEED_LIMIT 设置一个长整形数,控制传送多少字节。CURLOPT_LOW_SPEED_TIME 设置一个长整形数,控制多少秒传送CURLOPT_LOW_SPEED_LIMIT规定的字节数。CURLOPT_RESUME_FROM 传递一个包含字节偏移地址的长整形参数,(你想转移到的开始表单)。CURLOPT_SSLVERSION 传递一个包含SSL版本的长参数。默认PHP将被它自己努力的确定,在更多的安全中你必须手工设置。CURLOPT_TIMECONDITION 传递一个长参数,指定怎么处理CURLOPT_TIMEVALUE参数。你可以设置这个参数为TIMECOND_IFMODSINCE 或 TIMECOND_ISUNMODSINCE。这仅用于HTTP。CURLOPT_TIMEVALUE 传递一个从1970-1-1开始到现在的秒数。这个时间将被CURLOPT_TIMEVALUE选项作为指定值使用,或被默认TIMECOND_IFMODSINCE使用。CURLOPT_URL 这是你想用PHP取回的URL地址。你也可以在用curl_init()函数初始化时设置这个选项。CURLOPT_USERPWD 传递一个形如[username]:[password]风格的字符串,作用PHP去连接。CURLOPT_PROXYUSERPWD 传递一个形如[username]:[password] 格式的字符串去连接HTTP代理。CURLOPT_RANGE 传递一个你想指定的范围。它应该是”X-Y”格式,X或Y是被除外的。HTTP传送同样支持几个间隔,用逗句来分隔(X-Y,N-M)。CURLOPT_POSTFIELDS 传递一个作为HTTP “POST”操作的所有数据的字符串。CURLOPT_REFERER 在HTTP请求中包含一个”referer”头的字符串。CURLOPT_USERAGENT 在HTTP请求中包含一个”user-agent”头的字符串。CURLOPT_FTPPORT 传递一个包含被ftp “POST”指令使用的IP地址。这个POST指令告诉远程服务器去连接我们指定的IP地址。这个字符串可以是一个IP地址,一个主机名,一个网络界面名(在UNIX下),或是‘-'(使用系统默认IP地址)。CURLOPT_COOKIE 传递一个包含HTTP cookie的头连接。CURLOPT_SSLCERT 传递一个包含PEM格式证书的字符串。CURLOPT_SSLCERTPASSWD 传递一个包含使用CURLOPT_SSLCERT证书必需的密码。CURLOPT_COOKIEFILE 传递一个包含cookie数据的文件的名字的字符串。这个cookie文件可以是Netscape格式,或是堆存在文件中的HTTP风格的头。CURLOPT_CUSTOMREQUEST 当进行HTTP请求时,传递一个字符被GET或HEAD使用。为进行DELETE或其它操作是有益的,更Pass a string to be used instead of GET or HEAD when doing an HTTP request. This is useful for doing or another, more obscure, HTTP request. 注意: 在确认你的服务器支持命令先不要去这样做。下列的选项要求一个文件描述(通过使用fopen()函数获得)CURLOPT_FILE 这个文件将是你放置传送的输出文件,默认是STDOUT.CURLOPT_INFILE 这个文件是你传送过来的输入文件。CURLOPT_WRITEHEADER 这个文件写有你输出的头部分。CURLOPT_STDERR 这个文件写有错误而不是stderr。用来获取需要登录的页面的例子,当前做法是每次或许都登录一次,有需要的人再做改进了CURLOPT_AUTOREFERER 自动设置header中的referer信息CURLOPT_BINARYTRANSFER 在启用CURLOPT_RETURNTRANSFER时候将获取数据返回CURLOPT_COOKIESESSION 启用时curl会仅仅传递一个session cookie,忽略其他的cookie,默认状况下curl会将所有的cookie返回给服务端。session cookie是指那些用来判断服务器端的session是否有效而存在的cookie。CURLOPT_CRLF 启用时将Unix的换行符转换成回车换行符。CURLOPT_DNS_USE_GLOBAL_CACHE 启用时会启用一个全局的DNS缓存,此项为线程安全的,并且默认为true。CURLOPT_FAILONERROR 显示HTTP状态码,默认行为是忽略编号小于等于400的HTTP信息CURLOPT_FILETIME 启用时会尝试修改远程文档中的信息。结果信息会通过curl_getinfo()函数的CURLINFO_FILETIME选项返回。CURLOPT_FOLLOWLOCATION 启用时会将服务器服务器返回的“Location:”放在header中递归的返回给服务器,使用CURLOPT_MAXREDIRS可以限定递归返回的数量。CURLOPT_FORBID_REUSE 在完成交互以后强迫断开连接,不能重用。CURLOPT_FRESH_CONNECT 强制获取一个新的连接,替代缓存中的连接。CURLOPT_HTTPGET 启用时会设置HTTP的method为GET,因为GET是默认是,所以只在被修改的情况下使用。CURLOPT_HTTPPROXYTUNNEL 启用时会通过HTTP代理来传输。CURLOPT_MUTE 将curl函数中所有修改过的参数恢复默认值。CURLOPT_RETURNTRANSFER 将curl_exec()获取的信息以文件流的形式返回,而不是直接输出。curl_getinfo()函数的作用是获取一个curl连接资源句柄的信息,curl_getinfo()函数有两个参数,第一个参数是curl的资源句柄,第二个参数是下面一些常量:curl_getinfo()参数名称 说明CURLINFO_EFFECTIVE_URL 最后一个有效的url地址CURLINFO_HTTP_CODE 最后一个收到的HTTP代码CURLINFO_FILETIME 远程获取文档的时间,如果无法获取,则返回值为“-1”CURLINFO_TOTAL_TIME 最后一次传输所消耗的时间CURLINFO_NAMELOOKUP_TIME 名称解析所消耗的时间CURLINFO_CONNECT_TIME 建立连接所消耗的时间CURLINFO_PRETRANSFER_TIME 从建立连接到准备传输所使用的时间CURLINFO_STARTTRANSFER_TIME 从建立连接到传输开始所使用的时间CURLINFO_REDIRECT_TIME 在事务传输开始前重定向所使用的时间CURLINFO_SIZE_UPLOAD 上传数据量的总值CURLINFO_SIZE_DOWNLOAD 下载数据量的总值CURLINFO_SPEED_DOWNLOAD 平均下载速度CURLINFO_SPEED_UPLOAD 平均上传速度CURLINFO_HEADER_SIZE header部分的大小CURLINFO_HEADER_OUT 发送请求的字符串CURLINFO_REQUEST_SIZE 在HTTP请求中有问题的请求的大小CURLINFO_SSL_VERIFYRESULT Result of SSL certification verification requested by setting CURLOPT_SSL_VERIFYPEERCURLINFO_CONTENT_LENGTH_DOWNLOAD 从Content-Length: field中读取的下载内容长度CURLINFO_CONTENT_LENGTH_UPLOAD 上传内容大小的说明CURLINFO_CONTENT_TYPE 下载内容的“Content-type”值,NULL表示服务器没有发送有效的“Content-Type: header”感谢各位的阅读!关于“php中curl简单采集图片如何生成base64编码”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!...
这篇文章主要讲解了“在Ubuntu上怎么安装Thunderbird 3并设置中文显示”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“在Ubuntu上怎么安装Thunderbird 3并设置中文显示”吧!Windows 版本的 Thunderbird 需要升级的话,在thunderbird的帮助菜单里直接点检查更新就可以了Ubuntu 的话,稍微麻烦点:1. sudo add-apt-repository ppa:ubuntu-mozilla-daily/ppa2. sudo apt-get update3. sudo apt-get install thunderbird-3.0安装完成之后,Alt-F2 打开运行窗口,输入 thunderbird-3.0 ,然后回车就可以了不过3.0默认安装完成后,因为没有安装 locale-zh (源里也搜索不到),所以收到中文邮件默认不会用中文编码显示,这样每次还要调整编码,比较麻烦要解决这个问题的话,到thunderbird 的设置界面中,找到:display — fonts — advanced,点击advanced 按钮,按如下图设置即可:图1这里我将编码都选择了GB2312,可根据需要自己选择然后,重启thunderbird,这样,收发邮件默认都会以GB2312来显示了,原来有乱码的中文标题也显示正常了。感谢各位的阅读,以上就是“在Ubuntu上怎么安装Thunderbird 3并设置中文显示”的内容了,经过本文的学习后,相信大家对在Ubuntu上怎么安装Thunderbird 3并设置中文显示这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是辰讯云,小编将为大家推送更多相关知识点的文章,欢迎关注!...
小编给大家分享一下Python如何将一个CSV文件里的数据追加到另一个CSV文件,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!在做数据处理工作时,有时需要将数据合并在一起,本文主要使用Python将两个CSV文件内数据合并在一起,合并方式有很多,本文只追加方式。首先给定两个CSV文件的内容1.CSV2.CSV将2.CSV文件里的数据追加到1.CSV后面直接敲写Python代码with open('1.csv','ab') as f: f.write(open('2.csv','rb').read())#将2.csv内容追加到1.csv的后面查看1.CSV内的数据变化情况非常简单快捷的一次Python操作。以上是“Python如何将一个CSV文件里的数据追加到另一个CSV文件”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注辰讯云资讯频道!...
这篇文章给大家分享的是有关ASP.NET验证码的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。我主要是看到干扰线了,一个验证码里面要是没有干扰线什么的,至少得在噪点和随机码的排版上下工夫: /// <summary> /// 验证码生成类 /// </summary> public class verify_code : IHttpHandler, IRequiresSessionState { public void ProcessRequest(HttpContext context) { int codeW = 80; int codeH = 22; int fontSize = 16; string chkCode = string.Empty; //颜色列表,用于验证码、噪线、噪点 Color[] color = { Color.Black, Color.Red, Color.Blue, Color.Green, Color.Orange, Color.Brown, Color.Brown, Color.DarkBlue }; //字体列表,用于验证码 string[] font = { "Times New Roman", "Verdana", "Arial", "Gungsuh", "Impact" }; //验证码的字符集,去掉了一些容易混淆的字符 char[] character = { '0', '1', '2', '3', '4', '5', '6', '8', '9' }; Random rnd = new Random(); //生成验证码字符串 for (int i = 0; i < 4; i++) { chkCode += character[rnd.Next(character.Length)]; } //写入Session context.Session["sys_verify_code"] = chkCode; //创建画布 Bitmap bmp = new Bitmap(codeW, codeH); Graphics g = Graphics.FromImage(bmp); g.Clear(Color.White); //画噪线 for (int i = 0; i < 4; i++) { int x1 = rnd.Next(codeW); int y1 = rnd.Next(codeH); int x2 = rnd.Next(codeW); int y2 = rnd.Next(codeH); Color clr = color[rnd.Next(color.Length)]; g.DrawLine(new Pen(clr), x1, y1, x2, y2); } //画验证码字符串 for (int i = 0; i < chkCode.Length; i++) { string fnt = font[rnd.Next(font.Length)]; Font ft = new Font(fnt, fontSize); Color clr = color[rnd.Next(color.Length)]; g.DrawString(chkCode[i].ToString(), ft, new SolidBrush(clr), (float)i * 18 + 2, (float)0); } //画噪点 for (int i = 0; i < 100; i++) { int x = rnd.Next(bmp.Width); int y = rnd.Next(bmp.Height); Color clr = color[rnd.Next(color.Length)]; bmp.SetPixel(x, y, clr); } //清除该页输出缓存,设置该页无缓存 context.Response.Buffer = true; context.Response.ExpiresAbsolute = System.DateTime.Now.AddMilliseconds(0); context.Response.Expires = 0; context.Response.CacheControl = "no-cache"; context.Response.AppendHeader("Pragma", "No-Cache"); //将验证码图片写入内存流,并将其以 "image/Png" 格式输出 MemoryStream ms = new MemoryStream(); try { bmp.Save(ms, ImageFormat.Png); context.Response.ClearContent(); context.Response.ContentType = "image/Png"; context.Response.BinaryWrite(ms.ToArray()); } finally { //显式释放资源 bmp.Dispose(); g.Dispose(); } } public bool IsReusable { get { return false; } } }基本验证生成代码demo:using System;using System.Drawing;using System.Drawing.Imaging;using System.IO;using System.Web;public partial class image : System.Web.UI.Page { protected void Page_Load(object sender, EventArgs e) { string tmp = RndNum(4); HttpCookie a = new HttpCookie("ImageV", tmp); Response.Cookies.Add(a); this.ValidateCode(tmp); } private void ValidateCode(string VNum) { Bitmap Img = null; Graphics g = null; MemoryStream ms = null; int gheight = VNum.Length * 12; Img = new Bitmap(gheight, 25); g = Graphics.FromImage(Img); //背景颜色 g.Clear(Color.White); //文字字体 Font f = new Font("Arial Black", 10); //文字颜色 SolidBrush s = new SolidBrush(Color.Black); g.DrawString(VNum, f, s, 3, 3); ms = new MemoryStream(); Img.Save(ms, ImageFormat.Jpeg); Response.ClearContent(); Response.ContentType = "image/Jpeg"; Response.BinaryWrite(ms.ToArray()); g.Dispose(); Img.Dispose(); Response.End(); } private string RndNum(int VcodeNum) { string Vchar = "0,1,2,3,4,5,6,7,8,9,a,b,c,d,e,f,g,h,i,j,k,l,m,n,o,p" + ",q,r,s,t,u,v,w,x,y,z"; string[] VcArray = Vchar.Split(new Char[] { ',' }); string VNum = ""; int temp = -1; Random rand = new Random(); for (int i = 1; i < VcodeNum + 1; i++) { if (temp != -1) { rand = new Random(i * temp * unchecked((int)DateTime.Now.Ticks)); } int t = rand.Next(35); if (temp != -1 && temp == t) { return RndNum(VcodeNum); } temp = t; VNum += VcArray[t]; } return VNum; } }感谢各位的阅读!关于“ASP.NET验证码的示例分析”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!...
这篇文章主要介绍如何使用nodeJS实现一个mock数据服务器,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!对于前端开发者而言,javascript正扮演着越来越重要的地位,它不仅能为浏览器端赋能,在web服务器方面也有很大的价值(我们可以用nodeJS来写服务端代码,启动web服务器),因此本文所要描述的,便是javascript在服务端的应用。我将介绍如何使用nodeJS来搭建一个mock服务器,方便前端自定义mock数据请求,提高前端开发的主观能动性和对项目健壮性的探索。我们将学到koa基本使用koa-router的基本用法koa-logger的使用glob支持文件遍历查寻node几个核心api的使用使用nodemon做自动重启mock服务器基本设计思路通过目录路径和服务端api的映射关系来实现我们的api访问,比如我们访问接口/api/article/122,我们只需要在mock服务器目录的api的article目录下,创建122.json文件即可,json文件的数据可以自定义,方便前端调试。具体实现1.搭建一个node服务const Koa = require('koa');const app = new Koa(); app.listen(3000)2.注册路由 我们使用koa-router来实现后台服务的路由功能,并通过koa提供的上下文ctx将读取到的数据返回给前端:const Koa = require('koa');const Router = require('koa-router'); const app = new Koa();const router = new Router({prefix: '/api'}); router.get('/name', (ctx, next) => { ctx.body = { name: 'xuxiaoxi' } }); app .use(router.routes()) .use(router.allowedMethods()); app.listen(3000)这样我们就能实现一个勉强能用的基本的后台api服务器了,当我们请求/api/name时,会返回相应的数据给前台,这一步是我们实现mock服务的关键一步,接下来我们具体来实现目录的遍历和api的自动注册。3.自动注册api接口并返回数据 我们将在这个阶段实现api服务的自动注册,这里我们使用glob这个第三方模块来遍历目录,并通过node的fs模块读取api文件的数据并返回给前台。glob的使用很简单,感兴趣的朋友可以自行学习,这里就不做过多介绍了。具体实现如下:const Koa = require('koa');const Router = require('koa-router');const glob = require("glob");const { resolve } = require('path');const fs = require('fs'); const app = new Koa();const router = new Router({prefix: '/api'}); // 注册路由glob.sync(resolve('./api', "**/*.json")).forEach((item, i) => { let apiJsonPath = item && item.split('/api')[1]; let apiPath = apiJsonPath.replace('.json', ''); router.get(apiPath, (ctx, next) => { try { let jsonStr = fs.readFileSync(item).toString(); ctx.body = { data: JSON.parse(jsonStr), state: 200, type: 'success' // 自定义响应体 } }catch(err) { ctx.throw('服务器错误', 500); } }); }); app .use(router.routes()) .use(router.allowedMethods()); app.listen(3000);添加控制台日志 我们使用koa-logger实现在终端打印node日志,方便调试,虽然这不是该文章的重点,但是对于想做node开发的前端从业者,还是很有必要了解的。const logger = require('koa-logger') app.use(logger());这样,我们每个请求都会在终端打印出来。路由映射文件的生成 该功能也不是本文的重点,但是会极大的方便前端开发者调试请求,因为如果api路径很长,我们需要一个个查找,但是有了这个map文件,我们只需要拷贝自动生成的路径即可。具体实现如下://...const routerMap = {}; // 存放路由映射 // 注册路由glob.sync(resolve('./api', "**/*.json")).forEach((item, i) => { // ... // 记录路由 routerMap[apiJsonPath] = apiPath; }); fs.writeFile('./routerMap.json', JSON.stringify(routerMap, null , 4), err => { if(!err) { console.log('路由地图生成成功!') } });基本目录结构以上是“如何使用nodeJS实现一个mock数据服务器”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注辰讯云资讯频道!...
这篇文章将为大家详细讲解有关vue中本地静态图片路径怎么写,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。这里写图片描述 需求:如何components里面的index.vue怎样能把assets里面的图片拿出来。1.在img标签里面直接写上路径:<img src="../assets/a1.png" class="" width="100%"/>2.利用数组保存再循环输出:<el-carousel-item v-for="item in carouselData" :key="item.id"> <img :src="item.url" class="carouselImg"/> <span class="carouselSpan">{{ item.title }}</span> </el-carousel-item> data: () => ({ carouselData:[ {url:require('../assets/a1.png'),title:'你看我叼吗1',id:1}, {url:require('../assets/a3.png'),title:'你看我叼吗2',id:2}, {url:require('../assets/a4.png'),title:'你看我叼吗3',id:3} ] }),效果如下:index.vue里面的完整代码是这个:<template> <div> <div class=" block"> <el-carousel class="carouselBlock"> <el-carousel-item v-for="item in carouselData" :key="item.id"> <img :src="item.url" class="carouselImg"/> <span class="carouselSpan">{{ item.title }}</span> </el-carousel-item> </el-carousel> </div> <footer1></footer1> <img src="../assets/a1.png" class="" width="100%"/> </div></template><script> import footer1 from '../components/public/footer' export default { data: () => ({ carouselData:[ {url:require('../assets/a1.png'),title:'你看我叼吗1',id:1}, {url:require('../assets/a3.png'),title:'你看我叼吗2',id:2}, {url:require('../assets/a4.png'),title:'你看我叼吗3',id:3} ] }), components:{ footer1 }, }</script><style lang="scss"> @import '../style/mixin'; .carouselBlock{ width: 100%; height: REM(300); position:relative; .carouselImg{ height: REM(300); width:100%; } .carouselSpan{ position: absolute; bottom: REM(20); left: REM(20); font-size: REM(24); font-weight: bold; } } .el-carousel__container{ width: 100%; height: REM(300); } .el-carousel__item h4 { color: #475669; font-size: 14px; opacity: 0.75; margin: 0; } .el-carousel__item:nth-child(2n) { background-color: #99a9bf; } .el-carousel__item:nth-child(2n+1) { background-color: #d3dce6; }</style>PS:下面看下Vue.js中的图片引用路径当我们在Vue.js项目中引用图片时,关于图片路径有以下几种情形:使用一我们在data里面定义好图片路径imgUrl:'../assets/logo.png'然后,在template模板里面<<span class="hljs-title" >img src="{{imgUrl}}">这样是错误的写法,我们应该用v-bind绑定图片的srcs属性:src="imgUrl">或者<span class="hljs-title" >img src="../assets/logo.png">这种方式是按照正常HTML语法引用路径,放在模板里可以被webpack打包出来。使用二当我们需要在js代码里面写图片路径的时候,如果我们在data里面写imgUrl:'../assets/logo.png'此时webpack只会把它当做字符串处理从而找不到图片地址,此时我们可以使用import引入图片路径::src="avatar" />import avatar from '@/assets/logo.png'在data里面定义avatar: avatar情形三我们也可以把图片放在外层的static文件夹里面,然后在data里面定义:imgUrl : '../../static/logo.png':src="imgUrl" />关于“vue中本地静态图片路径怎么写”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。...
本篇内容介绍了“CentOS怎么添加用户到sudoer列表教程”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!CentOS将用户添加到sudoer列表CentOS默认情况下,linux没有将当前用户列入到sudoer列表中(在redhat系列的linux发行版中最为常见),这时如果你使用sudo来执行某些命令的话,就会提示你该用户不再sudoer列表中。这时,我们就需要手工加入了。1.在命令行下键入:$su ,并输入root账户的密码,切换到root账户,其中的$是命令提示符,不用你敲2.命令行键入:#visudo,就会打开sudo的配置文件。在linux下任何的配置文件都是以文本文件保存,即你可以使用vim来打开sudo的配置文件,但是不建议这么做,因为你visudo是linux提供的修改sudo配置文件的工具,与vim比起来,它会提供更多的提示信息和纠错能力。3.与用vim编辑文本文件一样,我们搜索root关键字,所以在打开的文件中输入 /root4.按键盘上的 n 键,会自动跳到下一个root的关键字,一直跳到这句话:## Allow root to run any commands anywhereroot ALL=(ALL) ALL这时,继续按n键,光标会出现在root ALL=(ALL) ALL中的root上5.输入 yyp 并回车,这是vim中的两个命令。yy表示复制当前行,p表示将复制的内容粘贴到下一行,所以输入该命令后原来的文字变为如下:## Allow root to run any commands anywhereroot ALL=(ALL) ALLroot ALL=(ALL) ALL6.按 j 键,光标会跳到第二个root ALL=(ALL) ALL这一行上,再按数字0键,让光标回到行首,这时按4下 x 键,删除这一行的root这四个字母。然后按下 i 键,进入插入模式,输入你的用户名,比如我现在的用户名为xin,则我输入xin。输入后原来的文字变为如下:## Allow root to run any commands anywhereroot ALL=(ALL) ALLxin ALL=(ALL) ALL7.按下ESC键(键盘的左上角,在F1的左边)推出插入模式,然后按下 :wq 三个字符退出并保存当前配置文件。8.现在已经修改完成了,xin这个用户已经加入到了sudoer的列表,你使用xin这个用户执行sudo 命令时已经可以了。“CentOS怎么添加用户到sudoer列表教程”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注辰讯云网站,小编将为大家输出更多高质量的实用文章!...