前文我们聊了下http协议里的缓存控制机制以及varnish架构组件介绍,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/12620538.html;今天我们来聊一下怎样配置使用varnish;

  前边我们说到过varnish有两个配置文件,一个是/etc/varnish/varnish.params,这个配置文件主要是定义varnishd主控进程的一些运行时参数以及定义varnishd监听在那个套接字上,以及连接varnish使用的密钥文件;另外一个配置文件是/etc/varnish/default.vcl这个配置文件其实是varnish.params文件中指定的默认缓存策略配置文件,这个里面主要是配置缓存相关策略,用varnish专有配置语言vcl写的配置文件;我们先来了解下varnish.params配置文件吧

[root@test_node1-centos7 ~]# vim /etc/varnish/varnish.params 

# Varnish environment configuration description. This was derived from
# the old style sysconfig/defaults settings # Set this to 1 to make systemd reload try to switch VCL without restart.
RELOAD_VCL=1 # Main configuration file. You probably want to change it.
VARNISH_VCL_CONF=/etc/varnish/default.vcl # Default address and port to bind to. Blank address means all IPv4
# and IPv6 interfaces, otherwise specify a host name, an IPv4 dotted
# quad, or an IPv6 address in brackets.
# VARNISH_LISTEN_ADDRESS=192.168.1.5
VARNISH_LISTEN_PORT=6081 # Admin interface listen address and port
VARNISH_ADMIN_LISTEN_ADDRESS=127.0.0.1
VARNISH_ADMIN_LISTEN_PORT=6082 # Shared secret file for admin interface
VARNISH_SECRET_FILE=/etc/varnish/secret # Backend storage specification, see Storage Types in the varnishd(5)
# man page for details.
VARNISH_STORAGE="malloc,256M" # User and group for the varnishd worker processes
VARNISH_USER=varnish
VARNISH_GROUP=varnish # Other options, see the man page varnishd(1)
#DAEMON_OPTS="-p thread_pool_min=5 -p thread_pool_max=500 -p thread_pool_timeout=300"

  提示:RELOAD_VCL这个参数主要是指定varnish是否支持不重启切换VCL配置文件,1表示支持,0表示不支持;这里一般都是设置成1,在工作中千万不要随意重启varnish,一旦重启好多缓存项都将失效,很有可能因为缓存失效,前端访问压力压到后端真正的服务器上,导致服务器崩溃的情况;VARNISH_VCL_CONF该参数指定缓存策略配置文件,默认是default.vcl,我们可以直接编辑这个文件,然后通过VCL编译编译成不同的配置名称的vcl;VARNISH_LISTEN_PORT该参数指定varnishd对外提供访问的端口,通常情况在前面没有代理的情况,我们需要把这个端口改成80或者443;VARNISH_ADMIN_LISTEN_ADDRESS该参数是指定varnish的管理接口接听地址,为了安全通常使用本机回环地址,防止远程连接;VARNISH_ADMIN_LISTEN_PORT该参数指定varnish的管理接口监听的端口,通常这个端口可以不用更改,因为管理接口一般都只有管理员使用;VARNISH_SECRET_FILE该参数指定varnish的管理接口连接所用到认证文件,通常不需要更改;VARNISH_STORAGE该参数指定varnish的缓存存储方式,varnish的缓存存储方式有三种,第一种是malloc内存存储,其配置语法是malloc[,size],这种存储方式重启后所有缓存项都将失效;第二种是file文件,配置语法file[,path[,size[,granularity]]],通常我们只需要指定文件的路径及文件大小,这种磁盘文件存储的方式是黑盒,重启后所有缓存项都将失效;第三种也是磁盘文件的方式存储,和第二种不同的是这种存储方式重启后所有缓存项都有效,但是这种存储方式在varnish4.0还处于试验阶段,所以我们能用的就两种,一种是内存存储,一种是文件黑盒存储,这两种方式都是重启后所有缓存项失效,所以varnish缓存服务器上不能随意重启的;VARNISH_USER和VARNISH_GROUP这两个参数是指定varnishd进程的启动用户和组;DAEMON_OPTS是指定varnish运行时参数,每个参数都需要用-p来加以指定,可重复多次使用来指定不同的参数;-r表示死定指定参数为只读状态;这里提示下varnish重载VCL配置文件是直接使用varnish的专用重载命令varnish_reload_vcl命令;了解了varnishd的vcl配置文件,接下来我们修改下对外提供服务端端口,然后尝试启动varnish看看;

  提示:因为本机上运行的有httpd把80端口给占用了,我这里以8000端口为例对外提供服务;同时我们也给定了varnish的缓存是基于文件黑盒存储方式,并指定其文件大小为500M;

  提示:可以看到varnish对外提供服务的端口已经起来了,但是用浏览器访问8000端口提示503,说后端server没有找到,这是因为默认情况下varnish指定的后端server是127.0.0.1:8080,我们要配置后端主机server可以在defalult.vcl中配置;如下所示

  提示:以上配置表示默认后端提供web服务的主机地址是127.0.0.1,端口是8080;通常情况我们是需要更改这个配置文件来指定后端主机和端口的,然后重新编译该配置文件然后加载使用;

  提示::这样修改配置文件后,需要用varnishadm这个工具连接到varnish提供的命令行接口上去编译配置文件,然后再加载使用;首先我们来说说varnishadm这个工具怎么使用吧

[root@test_node1-centos7 ~]# varnishadm --help
varnishadm: invalid option -- '-'
usage: varnishadm [-n ident] [-t timeout] [-S secretfile] -T [address]:port command [...]
-n is mutually exlusive with -S and -T
[root@test_node1-centos7 ~]#

  提示:以上是varnishadm命令的使用帮助,其实这个命令很好使用,我们只需要用-S(大写)来指定secret文件,然后用-T来指定varnish主机管理接口监听的地址和端口即可,当然这样去连接就是交互式连接,会给我们一个交互式界面输入命令操作varnish,如果不想交互式使用,在后面可以给命令,有点类似mysql这个工具的用法;接下来我们用varinshadm这个工具来连接下varnish的管理接口;

[root@test_node1-centos7 ~]# varnishadm -S /etc/varnish/secret -T 127.0.0.1:6082
200
-----------------------------
Varnish Cache CLI 1.0
-----------------------------
Linux,3.10.0-693.el7.x86_64,x86_64,-sfile,-smalloc,-hcritbit
varnish-4.0.5 revision 07eff4c29 Type 'help' for command list.
Type 'quit' to close CLI session. quit
500
Closing CLI connection
[root@test_node1-centos7 ~]#

  提示:看到以上界面就表示用varinshadm工具成功连接到varinsh的管理接口上了,输入quit表示推出管理界面,输入help表示参看命令列表

[root@test_node1-centos7 ~]# varnishadm -S /etc/varnish/secret -T 127.0.0.1:6082
200
-----------------------------
Varnish Cache CLI 1.0
-----------------------------
Linux,3.10.0-693.el7.x86_64,x86_64,-sfile,-smalloc,-hcritbit
varnish-4.0.5 revision 07eff4c29 Type 'help' for command list.
Type 'quit' to close CLI session. help
200
help [<command>]
ping [<timestamp>]
auth <response>
quit
banner
status
start
stop
vcl.load <configname> <filename>
vcl.inline <configname> <quoted_VCLstring>
vcl.use <configname>
vcl.discard <configname>
vcl.list
param.show [-l] [<param>]
param.set <param> <value>
panic.show
panic.clear
storage.list
vcl.show [-v] <configname>
backend.list [<backend_expression>]
backend.set_health <backend_expression> <state>
ban <field> <operator> <arg> [&& <field> <oper> <arg>]...
ban.list

  提示:用varnishadm工具连接varnish后每执行一个命令就会返回类似http协议中的状态码的数字,其中200就表示命令成功执行并返回相应的内容,这个状态的意思有点类似http里的状态码意思;从上面的help列出命令列表来看,它列出了各个命令的基本使用方法,比如hep命令可以直接使用help表示类出命令列表及命令使用格式,help 某个命令表示查看某个命令的使用方法;如下,我们要查看下ping命令的用法可以在命令行接口上敲help ping

  提示:从上面的帮助信息,我们可以了解到ping命令的主要作用是看看varnish是否存活,我们在varnish shell中敲ping命令返回200 和PONG 1585892735 1.0就表示varnish主机上存活的;

  继续上面的话题,我们修改了default.vcl配置文件,我们需要怎么样去编译呢?接续查看命令帮助

  提示:以上表示vcl.load命令的用法和说明,该命令主要是编译和加载VCL文件,使用方法是vcl.load +配置名称(这个名称是我们自定义的,可以说任何合法名称)+配置文件名称;如下

  提示:以上就表示编译default.vcl配置文件,并起名叫test1,我们可以在varnish shell 中敲 vcl.list来查看当前有几个vcl配置

  提示:可以看到有两个配置,一个是名字为boot的,其状态是active表示当前正在使用的,另一个是我们刚才编译指定的名称test1,状态是available表示有效的,可以用的,意思就是我们可以使用vcl.use来切换使用的;接下来我们来看看vcl.use的用法,并尝试切换我们新编译的配置;

  提示:vcl.use命令主要用来切换至指定配置名称的配置;从上面的返回结果看,test1现在处于active的状态,表示现在varnish应用的是test1的配置;接下来我们就可以在浏览器在尝试访问varnish对外提供访问的端口;

  提示:可以看到我们现在访问8000端口可以正常得到后端httpd服务器的响应;说明我们配置的后端主机ip和端口没有问题;同时上面的结果来看,varnish也是一款反向代理服务软件,通常varnish可以做反向代理,但是它里面的调度算法很简单只有轮询和加权轮询,之所以算法少是因为它的强项不是做反向代理服务器来用,它的强项是做缓存服务器来用,响应客户端的请求,很少通过反向代理到后端取资源;

  配置好varnish的后端web主机后,接下来我们来了解下varnish的配置语言VCL的语法

  VCL(varnish configuration lanuage)是“域”专有类型的配置语言,主要用于编写缓存策略的,VCL有多个状态引擎,状态之间存在相关性,但状态引擎彼此互相隔离;每个状态引擎可使用return(X)指明至那个下一级引擎;每个状态引擎对应于vcl文件中的一个配置端,即为subroutine,大概处理流程是这样的,例如vcl_hash --> return(hit) -->vcl_hit;处理过程要看return是什么,return(hit)就表示下一级处理的subroutine是vcl_hit;

  varnish4.0VCL语法有如下几点:

  1)VCL文件必须是vcl 4.0;开始

  示例:

  提示:以上除了#号开头的表示现有配置生效的指令;“#”表示注释

  2)//和#号和/**/都表示注释,前两者表示单行注释,最后一个表示多行注释;

  3)subroutines必须要有sub关键字指定

  示例:

  提示:这就表示一个subroutine 名字为vcl_recv

  4)没有循环,受限于引擎的内建变量

  5)用return()函数的参数作为下一个操作的关键字来结束语句;用return来实现状态引擎切换;

  VCL有限状态机特定:

  1)每项请求分别处理;

  2)每个请求在任何给定时间都是独立于其他请求的;

  3)状态是有相关性,但又各自隔离;

  4)return(action);退出一种状态并指定varnish进入下一种状态;

  5)内置的VCL代码始终存在,并附加在你自己的VCL下面;也就说我们不写任何VCL代码,它默认都有自己内置的VCL代码,且这个代码始终不变;我们可以用在varnish shell中使用vcl.show -v 指定配置名称来查看当前生效的配置详情(默认VCL代码+自己写的VCL配置代码),如下

[root@test_node1-centos7 ~]# varnishadm -S /etc/varnish/secret -T 127.0.0.1:6082
200
-----------------------------
Varnish Cache CLI 1.0
-----------------------------
Linux,3.10.0-693.el7.x86_64,x86_64,-sfile,-smalloc,-hcritbit
varnish-4.0.5 revision 07eff4c29 Type 'help' for command list.
Type 'quit' to close CLI session. varnish> vcl.list
200
active 0 boot varnish> vcl.show -v boot
200
// VCL.SHOW 0 1221 input
#
# This is an example VCL file for Varnish.
#
# It does not do anything by default, delegating control to the
# builtin VCL. The builtin VCL is called when there is no explicit
# return statement.
#
# See the VCL chapters in the Users Guide at https://www.varnish-cache.org/docs/
# and http://varnish-cache.org/trac/wiki/VCLExamples for more examples. # Marker to tell the VCL compiler that this VCL has been adapted to the
# new 4.0 format.
vcl 4.0; # Default backend definition. Set this to point to your content server.
backend default {
.host = "192.168.0.99";
.port = "80";
} sub vcl_recv {
# Happens before we check if we have this in cache already.
#
# Typically you clean up the request here, removing cookies you don't need,
# rewriting the request, etc.
} sub vcl_backend_response {
# Happens after we have read the response headers from the backend.
#
# Here you clean the response headers, removing silly Set-Cookie headers
# and other mistakes your backend does.
} sub vcl_deliver {
# Happens when we have all the pieces we need, and are about to send the
# response to the client.
#
# You can do accounting or modifying the final object here.
} // VCL.SHOW 1 5479 Builtin
/*-
* Copyright (c) 2006 Verdens Gang AS
* Copyright (c) 2006-2014 Varnish Software AS
* All rights reserved.
*
* Author: Poul-Henning Kamp <phk@phk.freebsd.dk>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
* *
* The built-in (previously called default) VCL code.
*
* NB! You do NOT need to copy & paste all of these functions into your
* own vcl code, if you do not provide a definition of one of these
* functions, the compiler will automatically fall back to the default
* code from this file.
*
* This code will be prefixed with a backend declaration built from the
* -b argument.
*/ vcl 4.0; #######################################################################
# Client side sub vcl_recv {
if (req.method == "PRI") {
/* We do not support SPDY or HTTP/2.0 */
return (synth(405));
}
if (req.method != "GET" &&
req.method != "HEAD" &&
req.method != "PUT" &&
req.method != "POST" &&
req.method != "TRACE" &&
req.method != "OPTIONS" &&
req.method != "DELETE") {
/* Non-RFC2616 or CONNECT which is weird. */
return (pipe);
} if (req.method != "GET" && req.method != "HEAD") {
/* We only deal with GET and HEAD by default */
return (pass);
}
if (req.http.Authorization || req.http.Cookie) {
/* Not cacheable by default */
return (pass);
}
return (hash);
} sub vcl_pipe {
# By default Connection: close is set on all piped requests, to stop
# connection reuse from sending future requests directly to the
# (potentially) wrong backend. If you do want this to happen, you can undo
# it here.
# unset bereq.http.connection;
return (pipe);
} sub vcl_pass {
return (fetch);
} sub vcl_hash {
hash_data(req.url);
if (req.http.host) {
hash_data(req.http.host);
} else {
hash_data(server.ip);
}
return (lookup);
} sub vcl_purge {
return (synth(200, "Purged"));
} sub vcl_hit {
if (obj.ttl >= 0s) {
// A pure unadultered hit, deliver it
return (deliver);
}
if (obj.ttl + obj.grace > 0s) {
// Object is in grace, deliver it
// Automatically triggers a background fetch
return (deliver);
}
// fetch & deliver once we get the result
return (fetch);
} sub vcl_miss {
return (fetch);
} sub vcl_deliver {
return (deliver);
} /*
* We can come here "invisibly" with the following errors: 413, 417 & 503
*/
sub vcl_synth {
set resp.http.Content-Type = "text/html; charset=utf-8";
set resp.http.Retry-After = "5";
synthetic( {"<!DOCTYPE html>
<html>
<head>
<title>"} + resp.status + " " + resp.reason + {"</title>
</head>
<body>
<h1>Error "} + resp.status + " " + resp.reason + {"</h1>
<p>"} + resp.reason + {"</p>
<h3>Guru Meditation:</h3>
<p>XID: "} + req.xid + {"</p>
<hr>
<p>Varnish cache server</p>
</body>
</html>
"} );
return (deliver);
} #######################################################################
# Backend Fetch sub vcl_backend_fetch {
return (fetch);
} sub vcl_backend_response {
if (beresp.ttl <= 0s ||
beresp.http.Set-Cookie ||
beresp.http.Surrogate-control ~ "no-store" ||
(!beresp.http.Surrogate-Control &&
beresp.http.Cache-Control ~ "no-cache|no-store|private") ||
beresp.http.Vary == "*") {
/*
* Mark as "Hit-For-Pass" for the next 2 minutes
*/
set beresp.ttl = 120s;
set beresp.uncacheable = true;
}
return (deliver);
} sub vcl_backend_error {
set beresp.http.Content-Type = "text/html; charset=utf-8";
set beresp.http.Retry-After = "5";
synthetic( {"<!DOCTYPE html>
<html>
<head>
<title>"} + beresp.status + " " + beresp.reason + {"</title>
</head>
<body>
<h1>Error "} + beresp.status + " " + beresp.reason + {"</h1>
<p>"} + beresp.reason + {"</p>
<h3>Guru Meditation:</h3>
<p>XID: "} + bereq.xid + {"</p>
<hr>
<p>Varnish cache server</p>
</body>
</html>
"} );
return (deliver);
} #######################################################################
# Housekeeping sub vcl_init {
return (ok);
} sub vcl_fini {
return (ok);
} varnish>

  提示:以上就是我们没有写任何VCL配置代码,默认就有的VCL配置代码;从上面的配置来看,VCL配置语言主要有三类主要语法

  第一类定义subroutine,主要格式是

sub subroutine {
...
}

  第二类是if else条件判断分支,格式如下

if CONDITION {
...
} else {
...
}

  第三类就是每个subroutine都是需要由return语句结尾,指定下一跳subroutine

  了解了上面的基本语法,我们再来看看VCL的内建函数和关键字

  首先说函数吧,regsub(str,regex,sub)这个函数是VCL内置查找替换字串的一个函数,这个函数只会替换第一次匹配的字串,如果后面有多个字串匹配不予处理;regsuball(str,regex,sub)这个函数和上面的那个函数只有一个区别,这个函数是替换所有匹配的字串;ban(boolean expression)该函数用于清理缓存项的; hash_data(input)对input做hash计算;synthetic(str)该函数用户合成字符串,通常用于嵌入其他代码用;

  关键字:call subroutine,return(action),new,set,unset

  操作符:==, !=, ~, >, >=, <, <=,逻辑操作符&&,||,!,变量赋值=

  内建变量大概有5类,分别是req.*表示由客户端发来的请求报文相关;如req.http.*就表示请求首部的变量,如req.http.User-Agent就表示引用http请求报文中的User-Agent首部的值;req.http.Referer就表示应用http请求首部Referer的值;bereq.*是有varnish发往后端主机的http请求相关;如bereq.http.*就表示引用发往后端主机的http请求首部的值,同req.http.*的逻辑上一样的;beresp.*:由BE主机响应给varnish的响应报文相关;resp.*:由varnish响应给client相关;这四类变量都是同一种逻辑,.http.*就表示引用http对应首部的值;obj.*是存储在缓存空间中的缓存对象的属性;

  常用的变量:

bereq.*, req.*:
bereq.http.HEADERS
bereq.request:请求方法;
bereq.url:请求的url;
bereq.proto:请求的协议版本;
bereq.backend:指明要调用的后端主机; req.http.Cookie:客户端的请求报文中Cookie首部的值;
req.http.User-Agent ~ "chrome" beresp.*, resp.*:
beresp.http.HEADERS
beresp.status:响应的状态码;
reresp.proto:协议版本;
beresp.backend.name:BE主机的主机名;
beresp.ttl:BE主机响应的内容的余下的可缓存时长; obj.*
obj.hits:此对象从缓存中命中的次数;
obj.ttl:对象的ttl值 server.*
server.ip
server.hostname
client.*
client.ip

  用户指定变量用set指令来设置,unset表示删除之意;

  示例:指定响应首部,如果命中缓存就把对应首部的值设置成“HIT via ”+服务端ip地址,没有命中对应首部的值就是“MISS via” +服务端ip地址

  提示:以上这段配置需要写在vcl_deliver中,vcl_deliver主要是varnish响应客户端报文都要经由它处理;有点类似iptables里的postrouting;

  测试,在浏览器中访问,看看响应首部X-Cache的值就可以判断该此请求是否被缓存命中;

  第一次访问,肯定是不会被缓存命中,因为压根就没有缓存,谈不上命中,所以第一次访问X-Cache首部的值应该是"MISS via 192.168.0.99"

  提示:可以看到第一次访问的确是MISS的,那么第二次和后面的访问会不会是miss的呢?

  提示:第二次访问X-Cache响应首部的值就变成了hit via 192.168.0.99 说明第二次访问被缓存命中了;

WEB缓存系统之varnish基础入门(一)的更多相关文章

  1. WEB缓存系统之varnish状态引擎

    前文我们聊了下varnish的VCL配置以及语法特点,怎样去编译加载varnish的vcl配置,以及命令行管理工具varnishadm怎么去连接varnish管理接口进行管理varnish,回顾请参考 ...

  2. WEB缓存系统之varnish缓存项修剪

    前文我们聊了下varnish的状态引擎和不同类型的变量对应该使用在那个状态引擎中,以及每个状态引擎的对应处理事务:回顾请参考https://www.cnblogs.com/qiuhom-1874/p/ ...

  3. WEB缓存系统之varnish代理以及健康状态检测配置

    前文我们聊了下varnish的缓存项修剪配置,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/12666406.html:今天我来说一下varnish作为代理服务 ...

  4. 容器编排系统之Kubernetes基础入门

    一.kubernetes简介 1.什么是kubernetes?它是干什么用的? kubernetes是google公司用go语言开发的一套容器编排系统,简称k8s:它主要用于容器编排:所谓容器编排简单 ...

  5. $Django 等web框架,交互,基础入门

    1 web 应用2 c/s 和bs架构3 python中的web框架    a :socket    b:路由跟视图函数匹配关系    c:模板渲染 django: uwsgi/wsgiref    ...

  6. [转载]WEB缓存技术概述

    [原文地址]http://www.hbjjrb.com/Jishu/ASP/201110/319372.html 引言 WWW是互联网上最受欢迎的应用之一,其快速增长造成网络拥塞和服务器超载,导致客户 ...

  7. varnish缓存系统基础知识

    缓存系统类型 1.页面缓存/pageCache     缓存静态资源(html js css image)  例如:varnish    squid 2.数据缓存/dataCache      缓存应 ...

  8. WEB缓存控制机制与varnish简介

    在说到缓存varnish前,我们首先来了解下对于web服务缓存到底是什么?它有哪些特点,基础原理是什么? http是web应用协议,通常我们说的一次http事务,不外乎就是客户端请求,服务端响应,通常 ...

  9. Web缓存基础:术语、HTTP报头和缓存策略

    简介 对于您的站点的访问者来说,智能化的内容缓存是提高用户体验最有效的方式之一.缓存,或者对之前的请求的临时存储,是HTTP协议实现中最核心的内容分发策略之一.分发路径中的组件均可以缓存内容来加速后续 ...

随机推荐

  1. springDataJPA笔记

    springDataJPA笔记 第一 orm思想 主要目的:操作实体类就相当于操作数据库表 建立两个映射关系: 实体类和表的映射关系 实体类中属性和表中字段的映射关系 不再重点关注:sql语句 实现了 ...

  2. 想要成为一名优秀的Java程序员,你需要这8个锦囊

    私底下,隔三差五就有读者问我:"二哥,怎么样才能像你一样,成为一名优秀的 Java 开发者呢?"假如把"怎么才能像你一样"去掉的话,这个问题就是一个好问题,否则 ...

  3. 7-11 jmu-python-分段函数&数学函数 (15 分)

    本题要求计算下列分段函数f(x)的值(x为从键盘输入的一个任意实数): 输入格式: 直接输入一个实数x 输出格式: 在一行中按“f(x)=result”的格式输出,其中x与result都保留三位小数. ...

  4. Python爬虫抓取微博评论

    第一步:引入库 import time import base64 import rsa import binascii import requests import re from PIL impo ...

  5. JZOJ 5328. 【NOIP2017提高A组模拟8.22】世界线

    5328. [NOIP2017提高A组模拟8.22]世界线 (File IO): input:worldline.in output:worldline.out Time Limits: 1500 m ...

  6. #AcWing系列课程Level-2笔记——5.高精度“+”算法

    高精度"+"算法 编写高精度"+",记住下面的过程,代码也就游刃有余了! 1.首先我们要明白大整数是如何存储的? 2.其次存储完,如何运算? 高精度" ...

  7. 实验二——Linux系统简单文件操作命令

    项目 内容 这个作业属于那个课程 这里是链接 作业要求在哪里 这里是链接 学号-姓名 17041506-张政 作业学习目标 学习在Linux系统终端下进行命令行操作,掌握常用命令行操作并能通过命令行操 ...

  8. CODING 携手优普丰,道器合璧打造敏捷最佳实践

    随着全球进入到信息化时代,越来越多的企业迫切地寻求新的商业模式,要求迭代.探索.不断加速创新以响应快速变化的市场.如今一系列新兴概念如敏捷开发.极限编程.微服务.自动化.DevOps 等大行其道,然而 ...

  9. Community Cloud零基础学习(四)Builder创建自定义的布局

    前几篇讲了Community Cloud权限配置等信息,但是没有太讲过 Community如何进行配置layout,本篇主要描述使用Builder去进行符合需求的Community Layout的构建 ...

  10. ubunto 免输入密码 登录 putty ssh-keygen

    交互式密码不安全,现在改用 ssh 证书方式,不用输入密码使用公钥证书登录. 方法1, 此方法,仅试用于,仅使用win putty 来连接方式使用,如果双方都是 linux 如 rsync 同步等时, ...