Categories

windows下安装基于Apache的Subversion服务器

my:

找到文件

libapr-1.dll
libaprutil-1.dll
libsvn_delta-1.dll
libsvn_fs-1.dll
libsvn_repos-1.dll
libsvn_subr-1.dll

从Subversion的安装目录VisualSVN Server\bin拷贝文件到Apache的modules目录,这几个文件覆盖为最新版本后Apache顺利启动。

apache的配置文件二个so模块启用,再加另二个模块,下文有介绍,最下面加:

<Location /svn>
DAV svn
SVNListParentPath on
SVNParentPath E:\usr\svn
AuthType Basic
AuthName “whw test svn server”
AuthUserFile E:\usr\svn\svn_auth_passwd
#AuthzSVNAccessFile E:\usr\testfile\svn_auth_passwd
Require valid-user
</Location>

<Location /svn2>
DAV svn
SVNPath E:\usr\svn2
AuthType Basic
AuthName “whw test svn server”
AuthUserFile E:\usr\svn2\svn_auth_passwd
Require valid-user
</Location>

<Location /svn>

意味着可以通过像这样的URL(http://MyServer/svn)来访问Subversion版本库

DAV svn

告诉Apache哪个模块负责服务像那样的URL--在这里就是Subversion模块

SVNListParentPath on

在Subversion 1.3及更高版本中,这个指示器使得Subversion列出由SVNParentPath指定的目录下所有的版本库

SVNParentPath D:\SVN

告诉Subversion在目录D:\SVN下寻找版本库

AuthType Basic

启用基本的验证,比如用户名/密码对

AuthName “Subversion repositories”

当一个验证对话框弹出时,告诉用户这个验证是用来做什么的

AuthUserFile D:\passwd

指定D:\passwd用为密码文件用来验证用户的用户名及密码

AuthzSVNAccessFile D:\svnaccessfile

指定D:\svnaccessfile来限定各个用户或组在版本库中目录的访问权限

Require valid-user

限定用户只有输入正确的用户名及密码后才能访问这个路径

参考资料:

http://subversion.tigris.org/ser … entList?folderID=91 (英文)

http://www.iusesvn.com/bbs/forumdisplay.php?fid=7(中文)下载最新版本的Subversion。

运行Subversion安装程序,如果安装程序能够识别你已经安装了Apache,那么你的安装基本上就OK了。如果它不能,那么你要做一些额外的步骤。

使用资源管理器,进入Sibversion的安装目录(通常为c:\program files\Subversion),找到文件httpd/mod_dav_svn.so和mod_authz_svn.so,将它们拷贝到Apache的modules目录(通常为c:\program files\apache group\apache2\modules)。

从Subversion的安装目录拷贝文件libdb43.dll到Apache的modules目录。

使用如记事本之类的文本编辑器编辑Apache的配置文件(通常为c:\program files\apache group\apache2\conf\httd.conf),修改以下内容:

去掉以下行的注释(将开头的#删除):

CODE:
#LoadModule dav_fs_module modules/mod_dav_fs.so
#LoadModule dav_module modules/mod_dav.so

在LoadModule节的最后添加以下两行:

CODE:
LoadModule dav_svn_module modules/mod_dav_svn.so
LoadModule authz_svn_module modules/mod_authz_svn.so

2.3. 配置
现在,你已经安装了Apache和Subversion,但是Apache还不知道如果处理像TortoiseSVN一样的Subversion客户端。要让Apache知道哪个URL将被Subversion使用,你要像下面这样编辑Apache的配置文件(通常在c:\program files\apache group\apache2\conf\httpd.conf):

在配置文件的最后添加下面这些行:

CODE:
<Location /svn>
SVNParentPath D:\SVN
AuthType Basic
AuthName “Subversion repositories”
AuthUserFile D:\passwd
#AuthzSVNAccessFile D:\svnaccessfile
Require valid-user
</Location>

这样配置表示:你所有的版本库将位于D:\SVN目录下,要访问你的版本库可以使用这样的URL:http://MyServer/svn/,访问权限将由passwd文件中的用户名/密码来限制。

要创建passwd文件,可以打开命令行(DOS窗口),将当前目录切换到apache2目录(通常为c:\program files\apache group\apache2),然后打入以下命令以创建文件:

CODE:
bin\htpasswd -c passwd <username>

重启Apache服务。

将你的浏览器指向http://MyServer/svn/MyNewRepository(MyNewRepository是你之前创建的Subversion版本库)。如果一切正常,你将被提示输入用户名密码,输入正确的用户名密码后你就可以看到版本库中的内容了。

对你刚刚输入的apache配置作一些简短的说明:

<Location /svn>

意味着可以通过像这样的URL(http://MyServer/svn)来访问Subversion版本库

DAV svn

告诉Apache哪个模块负责服务像那样的URL--在这里就是Subversion模块

SVNListParentPath on

在Subversion 1.3及更高版本中,这个指示器使得Subversion列出由SVNParentPath指定的目录下所有的版本库

SVNParentPath D:\SVN

告诉Subversion在目录D:\SVN下寻找版本库

AuthType Basic

启用基本的验证,比如用户名/密码对

AuthName “Subversion repositories”

当一个验证对话框弹出时,告诉用户这个验证是用来做什么的

AuthUserFile D:\passwd

指定D:\passwd用为密码文件用来验证用户的用户名及密码

AuthzSVNAccessFile D:\svnaccessfile

指定D:\svnaccessfile来限定各个用户或组在版本库中目录的访问权限

Require valid-user

限定用户只有输入正确的用户名及密码后才能访问这个路径

—————————————-

上面的配置仅仅是一个简单的示例。你还可以对Apache进行许许多多的配置。

1如果你想让所有用户对版本库都有读的权限而只有特定的用户才有写的权限,你可以将这行

CODE:
Require valid-user

改为

CODE:
<LimitExcept GET PROPFIND OPTIONS REPORT>
Require valid-user
</LimitExcept>

2上面的配置使用了passwd文件将你所有的版本库作为一个单元来限定访问权限。如果你想获得更多的控制,如限定某个用户可以访问版本库中的哪个目录,可以把下面这行的#去掉:

CODE:
#AuthzSVNAccessFile D:\svnaccessfile

然后创建一个Subversion授权文件。Apache将确保只有有效的用户可以访问你的/svn位置,然后将用户名传到AuthzSVNAccessFile模块,这样可以依据Subversion授权文件得到更精细的权限控制。注意,路径将被指定为[库:路径]或者简单的[路径]。如果你不明确指定一个库,访问规则将应用到由SVNParentPath指定的目录下所有的版本库中。一个例子可能像这样:

CODE:
[groups]
admin = john, kate
devteam1 = john, rachel, sally
devteam2 = kate, peter, mark
docs = bob, jane, mike
training = zak

# 为所有库指定默认访问规则
# 所有人可以读,管理员可以写,危险分子没有任何权限
[/]
* = r
@admin = rw
dangerman =

# 允许开发人员可以完全访问他们的项目版本库
[proj1:/]
@devteam1 = rw
[proj2:/]
@devteam2 = rw
[bigproj:/]
@devteam1 = rw
@devteam2 = rw
trevor = rw

# 文档编写人员对所有的docs目录有写权限
[/trunk/doc]
@docs = rw

# 培训人员可以完全访问培训版本库
[TrainingRepos:/]
@training = rw

以上介绍的配置为Apache多库方式,即一个location可以同时为多个版本库服务,
假设你的ip为192.168.0.1,在D:\SVN目录下建了两个版本库,proj1和proj2
可以分别用下面两个url来访问你的版本库

CODE:
http://192.168.0.1/svn/proj1
CODE:
http://192.168.0.1/svn/proj1

还有一种配置方式为Apache单库方式,即一个location只能为一个版本库服务,
配置时只要将上面的SVNParentPath改为SVNPath,同时将后面的路径由版本库的父目录改为版本库的目录
如:

CODE:
<Location /svn>
SVNPath D:\SVN\proj1
AuthType Basic
AuthName “Subversion repositories”
AuthUserFile D:\passwd
#AuthzSVNAccessFile D:\svnaccessfile
Require valid-user
</Location>

这时,可以用下面的URL来访问版本库proj1

CODE:
http://192.168.0.1/svn

rsync 使用说明

1. Install

[url]http://www.samba.org/rsync/[/url]

shell> tar zxvf rsync-x.x.x.tar.gz
shell> cd rsync-x.x.x
shell> ./configure && make && make install

目前大部分 Unix/Linux 默认即安装了 rsync。

2. /etc/rsyncd.conf
 
shell> touch /etc/rsyncd.conf
shell> vi /etc/rsyncd.conf

Edit /etc/rsyncd.conf as below:
 
uid = nobody
gid = nobody
use chroot = no
max connections = 5
pid file = /var/run/rsync.pid
lock file = /var/run/rsync.lock
log file = /var/log/rsync.log
 
[<dst-dir>]
path = <dst-dir-fullpath>
ignore errors
read only = no
list = yes
auth users = username
secrets file = /etc/rsyncd.secrets

3. /etc/rsync.secrets
 
shell> echo “jack:password” >> /etc/rsyncd.secrets
shell> chmod 600 /etc/rsyncd.secrets
 
*注:一定要把rsyncd.secrets的权限设为600,否则不能正常进行身份认证。
     包括–password-file指向的密码文件,也必须设成600权限。

    
    
4. Autorun

>>> Idea 1 – only linux <<<

shell> vi /etc/xinetd.d/rsync
 
set DISABLE to yes, the result looks like this:
service rsync
{
        disable         = no  <———- change to yes
        socket_type     = stream
        wait            = no
        user            = root
        server          = /usr/bin/rsync
        server_args     = –daemon
        log_on_failure  += USERID
}

>>> Idea 2 – on openSUSE <<<

shell> echo “/usr/local/bin/rsync –daemon” >> /etc/init.d/boot.local

>>> Idea 3 <<<
 
shell> echo “/usr/local/bin/rsync –daemon” >> /etc/rc.local

5. Run daemon

shell> rsync –daemon
 
*注一:在rsync的man手册的 CONNECTING TO AN RSYNC SERVER 处有提到:
It is also possible to use rsync without a remote shell as the transport. In this case you will connect to a remote  rsync server running on TCP port 873.
 
*注二:在rsync的man手册的 –port=PORT 选项解释中有提到:
This specifies an alternate TCP port number to use rather than the default port 873.

6. rsync 命令实例

6.1 显示目录内容

命令
——
a) rsync <dst-dir>
b) rsync -r <dst-dir>
c) rsync jack@192.168.0.1::<dst-dir>
d) rsync ssh_user@192.168.0.1:<dst-dir>

命令说明
———
a) 显示<dst-dir>目录内容(第一层)
b) 递归显示<dst-dir>目录
内容
c) 显示远程主机<dst-dir>目录内容
   *注1:端口模式, 基于rsync用户的身份验证
   *注2:rsync server上的目录必须具有xx7的权限.
d) 查看
远程主机<dst-dir>目录内容
   *注1:remote shell模式, 通过ssh连接的基于系统本地用户的身份验证
   *注2:这里只使用了一个冒号(:),同时用户名是远
程主机的ssh用户,密码也是ssh用户对应的密码。
   *注3:使用”<dst-dir>”,则列出<dst-dir>文件夹本身的信息。若要列出<dst-dir>文件夹内容,应使用”<dst-dir>/”。
  
参数说明
———
-r          对目录进行递归操作

  
6.2 本地目录之间同步

命令
——
a) rsync -av  –progress <src-dir>/ <dst-dir>             *** 注意(/) ***
b) rsync -av  –progress <src-dir>  <dst-dir>
c) rsync -avu –progress –delete <src-dir>/  <dst-dir>
d) rsync -av  –progress –temp-dir=/tmp <src-dir>/ <dst-dir>

命令说明
———
a) 同步src-dir目录下所有文件到dst-dir目录下
b) 同步src-dir目录下所有文件到dst-dir/src-dir目录下
c) 对src-dir目录内容向dst-dir目录下进行差异更新,有增加/更新则添加替换,有减少则对其删减
d) 比a)多了
–temp-dir=/tmp,即指定/tmp为临时交换区,这样可以避免因目标目录空间不够引起的无法同步文件的错误。

参数说明
———
-a          相当于 -rlptgoD 的集合
-u          等同于 –update,在目标文件比源文件新的情况下不更新
-v          显示同步的文件
–progress  显示文件同步时的百分比进度、传输速率
–delete    删除目标目录中多于源目录的文件

6.3 异地主机之间同步

命令
——
a) rsync -avz  –progress <src-dir> jack@192.168.0.1::<dst-dir>/
b) rsync -avz  –progress <src-dir> jack@192.168.0.1::<dst-dir>/ –password-file=/home/jack/rsync.jack

c) rsync -avuz  –progress –delete <src-dir> jack@192.168.0.1::<dst-dir>/ –password-file=/home/jack/rsync.jack

d) rsync -avz  –progress jack@192.168.0.1::<dst-dir>/<src-dir> <dst-dir>

命令说明
———
a) 同步本地<src-dir>目录的内容到远程主机192.168.0.1的<dst-dir>目录下,jack是rsync数据库用户(参见3. /etc/rsync.secrets)
b) 通过自动读取用户密码而实现非交互登录文件同步
c) 较b)多了-u和–delete

d) 同步远程主机内容到本地目录

参数说明
———
-z                等同于 –compress,对传输的文件压缩,这对节约网络带宽或在网络资源紧张的情况下非常有用
–password-file   引用192.168.0.1上rsync用户jack口令的本地文件,创建方法如下
                  shell> echo “jackpwd” >> /home/jack/rsync.jack
                  shell> chown jack:wheel
/home/jack/rsync.jack
                  shell> chmod 600 /home/jack/rsync.jack

Linux rsync同步由手动到自动

rsync 是一个快速增量文件传输工具,它可以用于在同一主机备份内部的备分,我们还可以把它作为不同主机网络备份工具之用。本文主要讲述的是如何自架rsync服务器,以实现文件传输、备份和镜像。

相对tar和wget来说,rsync 也有其自身的优点,比如速度快、安全、高效;
  手动的:

  首先要建立好两台服务器之间的公钥和密钥,那样就不需要每次都输入密码,也可以实现自动了。

  可以先使用rpm -qa |grep rsync 查看rsync是否已经安装,软件下载 Rysnc的主页地址为:http://rsync.samba.org/

  本试验的测试环境(2台,一个服务端,一个客户端)

  rsync服务端的ip:192.168.48.60

  rsync客户端的ip:192.168.48.148

  -----------------------------------------------

  以下先配置服务器端:

  1. 安装rsync包

  # rpm -q rsync

  rsync-2.6.3-1

  2. rsync只使用一个配置文件就是/etc/rsyncd.conf 这个文件一般是没有的,那就自己建一个好了

  # vi /etc/rsyncd.conf

  uid = nobody

  gid = nobody

  max connections = 200

  timeout = 600

  use chroot = no

  read only = yes

  pid file=/var/run/rsyncd.pid

  host_allow =192.168.48.148 //客户端的IP地址写在这里

  #syslog facility = local7

  #log file=/var/log/rsyncd.log

  #rsync config

  #The ’standard’ things

  [rsync_gmmold] //定义同步的路径(客户端用这个关键字链接)

  path = /home //需要同步的路径

  comment = gmmold //这个暂不知道

  3.启动服务

  #/usr/bin/rsync –daemon 最后在server端将rsync以守护进程形式启动, rsync启动的端口为 873端口

  4.加入开机启动

  echo “/usr/bin/rsync –daemon” >> /etc/rc.local

  5.检查rsync时候启动

  # ps -ef | grep rsync

  6.停止服务

  #kill `cat /var/run/rsyncd.pid`

以下是客户端的配置:

  1. 安装rsync包

  [root@rhel403 old]# rpm -q rsync

  rsync-2.6.3-1

  2. 与服务器端同步:

  #rsync -ave ssh root@192.168.48.60:/home/ /home/ganfic/laji/ //把60机子的/home下的文件同步备份到/home/ganfic/laji/

  自动同步备份:

  1. 在tmp下建一个b.sh文件

  #vi /tmp/b.sh

  把以下内容写到里面去

  #!bin/sh

  rsync -ave ssh root@192.168.48.60:/home/ /home/ganfic/laji/

  :wq

  2 第2步就要涉及到crontab了,在命令行输入

  #crontab -e

  然后把 * * * * * sh /tmp/b.sh //* * * * *部分代表每一分钟,然后 sh /tmp/b.sh 表示运行 b.sh脚本(在下面我再补充说明),-e 使用环境变量EDITOR或RVISUAL指定的编辑器编辑crontab文件。新创建的crontab文件会放在/var/spool/cron目录下,文件名就是用户名,用cd /var/spool/cron后 会看到用户文件,然后用 vi root(我是在root用户下的)就会看到* * * * * sh /tmp/b.sh已经被转到这里来了。

  这样就完成了,之后就是看看 有没有运行了,打开日志文件,日志文件在 #vi /var/log/cron 就会看到有没有运行了,也可以这样,在没有用自动运行之前,我们是用手动操作的,那时也把60的机子的/home目录下的文件拷了下来,那样,我们进本机的/home/ganfic/laji文件夹里就会看到60机子下/home的内容,把这些内容都删了,然后再过1分钟看会不会把内容再补上,有的话就是成功了。

  Linux 防火墙是用iptables,所以我们至少在服务器端要让你所定义的rsync 服务器端口通过,客户端上也应该让通过。

  iptables -A INPUT -p tcp -m state –state NEW -m tcp –dport 873 -j ACCEPT

  启动rsync服务:

  /usr/bin/rsync –daemon –config=/etc/rsyncd/rsyncd.conf

  $rsync_HOME/rsync –daemon –config=/etc/rsyncd/rsyncd.conf

补充:

  rsync 的相关知识:

  rsync命令的用法

  在配置完rsync服务器后,就可以从客户端发出rsync命令来实现各种同步的操作。rsync有很多功能选项,下面就对介绍一下常用的选项:

  rsync的命令格式可以为:

  1. rsync [OPTION]… SRC [SRC]… [USER@]HOST:DEST

  2. rsync [OPTION]… [USER@]HOST:SRC DEST

  3. rsync [OPTION]… SRC [SRC]… DEST

  4. rsync [OPTION]… [USER@]HOST::SRC [DEST]

  5. rsync [OPTION]… SRC [SRC]… [USER@]HOST::DEST

  6. rsync [OPTION]… rsync://[USER@]HOST[:PORT]/SRC [DEST]

命令格式:

  #rsync [option] 源路径 目标路径

  其中:

  [option]:

  a:使用archive模式,等于-rlptgoD,即保持原有的文件权限

  z:表示传输时压缩数据

  v:显示到屏幕中

  e:使用远程shell程序(可以使用rsh或ssh)

  –delete:精确保存副本,源主机删除的文件,目标主机也会同步删除

  –include=PATTERN:不排除符合PATTERN的文件或目录

  –exclude=PATTERN:排除所有符合PATTERN的文件或目录

  –password-file:指定用于rsync服务器的用户验证密码

  源路径和目标路径可以使用如下格式:

  rsync://[USER@]Host[:Port]/Path <–rsync服务器路径

  [USER@]Host::Path <–rsync服务器的另一种表示形式

  [USER@]Host:Path <–远程路径

  LocalPath <–本地路径

  crontab的用法:

  crontab [ -u user ] { -l | -r | -e }

  -u user 表示指定用户user的crontab文件,这个前提事你必须要有其权限才能其他用户的crontab文件。如果使用该选项的化,默认设定当前用户的crontab文件。

  -l 列出目前的crontab文件的内容

  -r 删除目前的crontab文件

  -e 使用环境变量EDITOR或RVISUAL指定的编辑器编辑crontab文件。新创建的crontab文件会放在/var/spool/cron目录下,文件名就是用户名。

  crontab内容的格式如下:

  分<>时<>日<>月<>星期<>要运行的命令

  其中<>表示空格

  分:1-59 时:0-23(0表示子夜) 日:1-31 月:1-12 星期:0-6(0表示星期天)

  crontab 文件中的每一行内容必须有上面这5个时间域,并用空格隔开。在时间域中可以用横杆“-”表示一个时间范围,如星期一到星期五:1-5,还可以使用逗号 “,”,如要在星期一和星期三执行任务,可以写成1,3表示。可以用星号“ * ”表示连续的时间段,如果对某个时间域没有特定的限制就可以在该域中写入*。 在crontab文件中注释的行在行首用“ # ”表示。下面举例说明哈。

  #每一分钟执行一次

  * * * * * test.sh

  #每五分钟执行一次

  */5 * * * * test.sh

  #星期一和星期五的12点到18点之间每两小时整执行一次

  0 12-18/2 * * 1,5 test.sh

Linux Shell简介

原文: http://www.mandrakeuser.org/doc
中译本:吴晓光 http://dummy.linux.net.cn/~xgwu/cmuo/
整理: flaboy 北南南北
点评:flaboy兄当时整理时发在了LinuxSir.Org 论坛 SHELL讨论区中,《Linux Shell简介》,是作为基础教材的形式帖出的,他在整理时“本文编译整理时对相关章节做了相应的删改处理,去掉了针对Mandrake Linux的部分内容。”;

我看这个文档极其不错,进行了再次整理,并对文档的结构进行了调整,根据文档的内容,我适当的加了序列号,并做了一个目录,主要是方便大家阅读;感谢原作者及中译者,同时也感谢flaboy兄的整理;

目录

版权信息
前言
第一篇:超级工具/Terminals,xterms 和 Shells

第二篇:自动补齐/命令行的历史记录/编辑命令行/可用的 Shell 快捷方式

第三篇:命令的排列/命令的任务调度/命令的替换

第四篇:文件名匹配/输出重定向

第五篇:bash 配置文件/提示符/改变 $PATH

第六篇:命令的别名、Shell 函数/从这里出发/Shell 常见问题

关于本文
相关文档


+++++++++++++++++++++++++++++++++++++++++++++++++++++
正文
+++++++++++++++++++++++++++++++++++++++++++++++++++++

版权信息

本文的内容来源于 MUO 的 Basics 部分,其原始英文版可以从这里获得http://www.mandrakeuser.org/docs。中文版来自吴晓光的CMUO http://dummy.linux.net.cn/~xgwu/cmuo/。MUO 是 Mandrake Linux( http://www.mandrakelinux.com/ )为用户提供的入门手册,其内容实用并且实时更新,非常适合初学者做入门参考。与常见的各种Linux教程不同,MUO介绍给Linux初学者的是学习Linux的方法而非对某个系统的描述,这对各种有着千差万别的Linux发行版的学习尤为重要。

本文编译整理时对相关章节做了相应的删改处理,去掉了针对Mandrake Linux的部分内容。

前言
使用 Shell

以下将介绍并解释基本的 shell 命令和机制。

第一篇:超级工具/Terminals,xterms 和 Shells


一、超级工具

您或许听过这样的论调:命令行(the mommand line)早就已经过时了,那东西神秘兮兮的,等等。有些人甚至觉得操作系统中应该没有这些命令才好。

事实是上,您可以不懂任何 shell ,就能使用 Linux 。您启动系统后可以直接进入 X Window ,最后在 X Window 下关机。
我坚信,用 Linux 而不懂 shell ,就象开车只会用头档(first gear)一样。当然,最初看起来,直接而简单,在大多数情况下都管用。但速度慢,而且无法真正体验驾驶的乐趣。

对,命令行很有趣。就象用一大堆收集到的积木,竟可以完成许多意想不到的创举,一些极其复杂的工作,只需几行命令就可以解决。这是因为,在 Unix 中,shell 可不是简单的命令解释器(典型的有 Windows 中的 DOS ),而是一个全功能的编程环境。
这并不意味着 shell 非常容易学通,您知道,好事多磨,这还是要花点工夫的。;-) 但请相信我,这绝对值得。您在很短时间内,就能被一大帮门外汉吹捧为 Unix wizard(奇才) 。*grin*

二、为了说明 shell ,这里需要一些背景知识。


1、Terminals, xterms 与 Shells

追溯到 Unix 诞生的那个年代,当时还没有现在流行的”个人计算机”。被称为计算机的机器,还是吞吐磁带与 magnetic memory (用术语’core’来表示系统 memory)的庞然大物。DEC 公司(现在的 Compaq)推出的 PDP-11 ,体积小(被称为 mini)而且价格底,在大学中引起了巨大的反响,很多学校直到那时才买得起一台计算机(PDP-11 物美价廉,只有 10000$)。

这些机器的操作系统由汇编语言、机器语言写成,所以运行起来效率很高,但都无法移植(unportable)。每家计算机公司都给自己的机器配上独有的操作系统,然后再销售。

这种笨拙的作法很快就被人们意识到了,于是就开始兴建一个可以在不同品牌机器上运行的操作系统。1969 年,Ken Thompson 开始写后来成为 Unix 的第一行代码。(Thompson 曾经参加了一个项目:MULTICS,Unix 是与这有关的一个玩笑词) 其实,Dennis Ritchie 为这个新的操作系统设计了一种新的编程语言– C 语言后,事情才真正开始。

虽然 Unix 的效率不及原来的操作系统,但有三个突出的优点:可以任意移植到其他机器,其中的 C 语言大大简化了编程,而且这些都 free 。很快,全美国的大学都忙着开始为机器安装 Unix 。

2、终端(Terminals)

Unix 是可以在许多种机器上运行的操作系统,但人们又如何使用这些机器呢?他们是通过哑终端来连接到这些机器,也就是用键盘、显示器及足够的 electronics (电子元件)组成的机器与中央计算机(central computer)相连。在这些终端上,用户可以敲字符(teletypy),这就是字符串’tty’表示终端设备文件,和’getty’命令的名称来历。

您可能会问,现在这些东西都在哪儿。 这些终端的厂家无法达成一项最终标准,这导致每种牌子的终端都有各自的键盘布局、各自的在屏幕上显示字符的方法、发送或接收什么信号表示什么字符、控制代码等等。

为了避免这些混乱,就创建了一个含有所有不同终端特性的(capability)文件,这就是’termcap’。用一个工具打开’/etc/termcap’瞧瞧,可别吓着了 ;-)

Linux 终端大多数用’vt100′或’linux’作为终端类型。

3、xterms

在八十年代初期,产生了一个 Unix 的图形子系统– the X Window System 。九十年代早期,为了更好地实现基于 Intel 的 Unix 类系统上(如FreeBSD、NetBSD、Linux)的应用,产生了一个系统分支– XFree86 。

X Window 中一个很大的好处是可以运行多个虚拟(virtual)终端。甚至在 X Window 下就有这么个应用程序–’xterm’。您将发现’xterm’和’virtual terminal’在很多情况下都是一样的。有的地方说’打开一个 xterm’,其实您不是非要用’xterm’程序,其他的终端模拟器(terminal emulator),如 rxvt、konsole、aterm、eterm、wterm 等等,一样有效。

终端模拟器(又称为虚拟终端)通过伪(pseudo) tty 设备– pty 与系统相连,并且使用自己的显示标准– xterm 。这导致不同的终端模拟器可能在一些按键或程序上存在细小的差别,这取决于模拟器多大程度上遵守了’xterm’的显示标准。

4、Shells

为了在终端中运行程序,需要 shell 。shell 是操作系统的一部分,用来与用户打交道,并且可以用来协调各个命令。

第一个真正的 Unix shell — ’sh’,亦称为’Bourne shell’,诞生于 1975 年,作者是 Steve Bourne 。很快,出现了其他 shell ,如基于原始’Bourne shell’的’ksh’、’zsh’,后者常用作专属 Unixes 系统中的标准 shell ;也有一些从 C 语言中衍生出来的 shell ,如’csh’或’tcsh’。

在 Linux 中,标注的 shell 是’bash’,即 the GNU Bourne-Again Shell (有点玩笑的味道……)。这个 shell 功能非常强大(甚至有人觉得太庞大了),压缩的 man page 就有 50 KB 。

三、Shell 起步

首先,有一点小说明:在平常应用中,建议您不要用’root’帐号运行 shell ,如果您还是新手,这一点尤其要注意。作为普通用户,不管您有意还是无意,都无法破坏系统;但如果是’root’,那就不同了,只要敲几个字母,就可能导致灾难性后果。

当您登入系统或打开一个 xterm 窗口,首先看到的是提示符(prompt)。Red Hat Linux 的标准提示符包括了您的用户名、登入的主机名(没有设置的话,是’localhost’)、当前所在的目录(working directory)、提示符号:

[tom@belbo tom]$

我以用户名’tom’登入名为’belbo’的主机,当前在我的 home 目录–’/home/tom’中。’root’的提示符:

[root@belbo root]#

除了不同的用户名外,提示符号由’$'变成了’#'。根据 Bourne shell 的传统,普通用户的提示符以’$'结尾,而超级用户用’#'。

提示符的每个部分都可以定制,您在后面将有更深的了解。

要运行命令的话,您只要在提示符后敲进命令,然后在按 键。shell 将在其路径中(详情见后)搜索这个命令,找到以后就运行,并在终端里输出相应的结果(如果有的话),命令结束后,再给出新的提示符:

[tom@belbo tom]$ whoami
tom
[tom@belbo tom]$

 

顺带指出,当您敲 ENTER 时,光标(cursor)在哪里并不要紧,因为 shell 总是会整行地读取。

基本的命令有:’ls’(list directory ,列出目录内容)、’cp’(copy ,复制)、’mv’(move / rename ,移动/重命名),’cd ‘(change directory ,改变目录),这些命令后面都可以跟上一帮可选项,这方面 man page 有详细的介绍(man ls, man mv 等等)。
在您动身前往 shell 领地前,这里有几个术语(terminology)的简短说明。命令可能带一些可选项(options)、参数(arguments):

mv -i file dir

 

其中’-i’是命令’mv’的一个可选项,而’file’和’dir’则是参数。所有可选项在该命令的 man page 都中有详细的介绍(此例中用 man mv),而参数则由您提供。可选项决定命令如何工作,而参数则用于确定命令作用的目标。

到目前为止,介绍得有点象许多人厌恶轻视的 DOS shell ,但伴随着下面的介绍,您将会有新的体验。


第二篇:自动补齐/命令行的历史记录/编辑命令行/可用的 Shell 快捷方式

Unix (及后继者 Linux)在命令行下面诞生,因此,Unix 中的命令行有许多非常实用的功能。在本篇中,我们将来作一些了解。

一、自动补齐;

如何用’cd’(改变目录,change directory)最快地从您当前所在的 home 目录跳到’/usr/src/redhat/’呢?

cd /u<TAB>sr<TAB>r<TAB>

这称为’命令行自动补齐’(automatic command line completion),这在平常应用中是不可缺少的。让我们仔细看看这个例子:

cd /u<TAB>

扩展成了 cd /usr/ ,很简单吧。下面的

cd /u<TAB>sr<TAB>

 

扩展为 cd /usr/src/ 。如果您只敲了cd /us,’/usr’下匹配的(’cd /u*/s*’)三个子目录将列出供您选择:’/usr/sbin’、’/usr/share’和’/usr/src’。

因此, 键可以很方便地用于根据前几个字母,来查找匹配的文件或子目录。比如,ls /usr/bin/zip 将列出所有’/usr/bin’下面,以字符串’zip’开头的文件或子目录。当然,完成这类任务还有更厉害的命令,但这个方法确实很管用。

另外,碰到长文件名时就显得特别方便。假设您要安装一个名为’boomshakalakwhizbang-4.6.4.5-i586.rpm’的 RPM 包,您输入 rpm -i boom ,如果目录下没有其他文件能够匹配,那 shell 就会自动帮忙补齐。

cd /usrl

将扩展成 cd /usr/src/linux ,并等待继续。’/usr/src’中有两个匹配的目录:’/usr/src/linux-[...]‘、’/usr/src/linux’。如何告诉 shell 您想要后面的那个呢?只要跟一个斜线(/ ,slash),就可以选择后面的那个了。

假如您不确定是’/usr/src/linux/Documentation’还是’/usr/src/linux/documentation’。而您知道,Linux 是区分大小写的。如果已经仔细读过前面部分的话,您想到可以用:

cd /usrl/d

扩展成了’/usr/src/linux/drivers/’,因此应该是’Documentation’(大写的’D')。

这种补齐对命令也有效:

[tom@belbo tom]$ gre<TAB>

grecord grefer grep

[tom@belbo tom]$ gre

在这里 shell 将列出所有以字符串’gre’开头的已知命令。

二、命令行的历史记录

通过按向上方向键,您可以向后遍历近来在该控制台下输入的命令。用向下方向键可以向前遍历。与 SHIFT 键连用的话,您还可以遍历以往在该控制台中的输出。您也可以编辑旧的命令,然后再运行。

按 后,shell 就进入”reverse-i(ncremental)-search”(向后增量搜索)模式。现在输入您要找的命令的首字母:

(reverse-i-search)`’:. 敲入 ‘i’可能会变成:

(reverse-i-search)`i’: isdnctrl hangup ippp0

如果您再按 键,上面的命令将再次执行。而如果您按了向右、向左方向键或 ,上面的命令将回到普通的命令行,这样您就可以进行适当编辑。

编辑命令行

通过光标和功能键(Home、End 等键),您可以浏览并编辑命令行,如果您需要,还可以用键盘的快捷方式来完成一般的编辑:

l         <CTRL k>:删除从光标到行尾的部分
l         <CTRL u>:删除从光标到行首的部分
l         <ALT d>:删除从光标到当前单词结尾的部分
l         <CTRL w>:删除从光标到当前单词开头的部分
l         <CTRL a>:将光标移到行首
l         <CTRL e>:将光标移到行尾
l         <ALT a>:将光标移到当前单词头部
l         <ALT e>:将光标移到当前单词尾部
l         <CTRL y>:插入最近删除的单词
l         <!$>:重复前一个命令最后的参数。

 

例如:您用命令 mkdir peter/pan/documents/tinkerbell 新建了一个目录,现在您向用命令’cd’进入该目录,您可以用 cd !$,shell 将把前一个命令’mkdir’的参数添加到现在的’cd’后面。

当您更深入了解Linux后,将看到这些快捷方式在其他应用程序下输入时,有时也有效,比如,在浏览器中的输入框中。

三、可用的 Shell 快捷方式

Red Hat Linux 带有不少快捷方式,其中一部分是 bash 原来就有的,而还有一些则是为您预先设置的(在后面您将看到如何设置)。

由于 home 目录是每位用户的活动中心,许多 Unix 对此有特殊的快捷方式。

‘~’就是您的 home 目录的简写形式。我们假设您在其他目录,想把一个名为’sometext’的文件复制到您 home 目录下的 ‘docs’子目录中。除了输入:

cp sometext /home/myusername/docs

您还可以用简写:

cp sometext ~/docs

 

理论上,这也可以应用在命令’cd’上。无论当前路径在哪里,cd ~ 将回到您的 home 目录。其实还可以简化,只要键入 cd ,就可以返回 home 目录了。

Red Hat Linux 为您提供了一些预先设置的快捷方式(称为’别名’,aliases),比如:

l         ll :将执行'ls -l -k'(以长格式列出目录内容,包括一些文件属性,并以 KB 而不是 byte 为单位显示文件大小)

l         ls :将执行'ls -F --color=auto'(列出目录内容,加上文件类型标识,并使用颜色)

现在,您应该对 shell 及一些快捷方式有了进一步的了解,下面我们来看看除了应用一些简单的命令,shell 还能作什么。

第三篇:命令的排列/命令的任务调度/命令的替换


一、命令的排列

现在您将看到一些常用的命令排列。您可能想在一行中给出所有命令,然后就可以把注意力转移到其他地方。没问题,shell 允许您在不同的命令之间,放上特殊的排列字符(queuing characters) 。这儿将介绍最常用的两种。

请注意,为了看起来更清楚,我在这些字符两旁加了空格。而在实际应用中,您不一定要这么做,’ls -a ; du -hs’和’ls -a;du -hs’的效果是一样的。

command1 ; command2

 

先执行 command1 ,不管 command1 是否出错,接下来执行 command2 。

例如:

ls -a ; du -hs

将先在屏幕上列出目录中的所有内容,然后列出所有目录及其子目录所占磁盘大小。

command1 && command2

只有当 command1 正确运行完毕后,才执行 command2 。

例如:

ls -a bogusdir && du -hs

 

将返回 ls: bogusdir: No such file or directory ,而’du’则根本没有运行(这是因为您没有’bogusdir’目录)。如果您将符号换成了’;',’du’将被执行。

为了进一步说明’;'和’&&’的区别,及一般命令排列的用处,下面举一个经典的例子:Linux 内核的编译和安装。

要编译、安装 Linux ,您需要执行一串命令:’make dep’、’make clean’、’make bzImage’、’make modules’、’make modules_install’和’make install’。如果要等一个命令完成后,再输入下一个,再等,再输入,……,那就太麻烦了。另一方面,每个命令只有当前面的命令都正确执行完毕后,才能开始执行。如果您用’;'来排列命令,则即使有命令执行失败,后面的也照常运行,最后,您可能在’/boot’目录下得到一个有问题的内核映像(image)。而用’&&’:

make dep && make clean && make bzImage && make modules && make modules_install && make install

 

不需要中途打断,就可以编译内核及其模块,并完成后面的安装。

二、命令的任务调度

当您在终端里运行一个命令或开启一个程序时,终端要等到命令或程序运行完毕后,才能再被使用。在 Unix 中,我们称这样的命令或程序在前台(foreground)运行。如果您想在终端下运行另一个命令,则需要再打开一个新的终端。

但这里还有一个更优雅的办法,称为任务调度(jobbing)或后台(backgrounding)。当您运用任务的调度或将命令置于后台,终端就立即解放了,这样一来,终端立即就可以接受新的输入。为实现这样的目的,您只需在命令后面添加一个 & :

gqview &

 

告诉 shell 将图片查看器’GQview’放到后台去执行(即当成 job 来运行)。

命令 jobs 将告诉您,在这个终端窗口中,运行着哪些命令与程序:

jobs

[1]+ Running gqview &

 

当您要关闭终端窗口时,这一点就很重要,因为关闭终端将导致所有在其中运行的任务都将被中止,在此例中,如果您关闭了终端,由这个终端开启的 GQview 程序也将被关闭。

但如何将前台运行的一个程序放到后台去?没问题:

gqview

<CTRL z>

[2]+ Stopped gqview

bg

[2]+ gqview &

 

组合键 将挂起终端中正在运行的程序,然后您就可以用 bg 命令将其放到后台去执行。

请注意,在后台运行图形应用程序有时候是有用处的,这样可以在终端下显示这个程序的出错信息,虽然这对您可能没有直接的帮助,当如果碰到了麻烦,向别人询问时,这些出错提示就有用武之地了。

一些图形程序,很可能还处在测试期(Beta),尽管在后台执行,也会在终端中输出一些信息。如果您对此不满,可以用下面命令:

command &>/dev/null &

这不仅将程序送到后台执行,还将其输出发到’/dev/null’文件。’/dev/null’是系统的”碎纸机” (shredder),所有送到那里的信息都将消失殆尽。

三、命令的替换

命令替换(Command substitution)是一项很实用的功能。我们假设,您想看看 XFree86 文档中的 ‘README.mouse’文件,但您不知道这个文件的位置。但您是位机灵的用户,已经听说了’locate’命令,也安装了’slocate’包,您就可以用:

locate README.mouse

发现那个文件在’/usr/X11R6/lib/X11/doc’。现在您就可以在终端里用’less’或在文件管理器中进入那个目录然后读取文件。而命令替换可以给您带来一些便捷:

less $(locate README.mouse)

 

一步到位。命令’locate README.mouse’的输出(= /usr/X11R6/lib/X11/doc/README.mouse)作为’less’的参数,然后就可以显示文件内容了。

这种机制的语法是:

command1 $(command2)

 

除了’$( )’,您还可以用后引号(backquote):

command1 `command2`

 

这样虽然可以减少输入,但可读性差,而且很容易就和没有替换功能的一般单引号混淆。我更欣赏前一种方法,但这最终起决于您。

这里有另外一个例子。我们假设,您打算结束一个名为’rob’的程序。您先得用命令’pidof’找出相应的进程号(Process ID),然后以这个 PID 为参数,运行’kill’命令,这样就可以结束’rob’程序。除了用:

pidof rob

567

kill 567

 

您还可以试试:

kill `pidof rob`

 

怎么样,效率有所提高吧?

在下一篇中,我将接着介绍 shell 的另外两种实用的机制:文件名匹配、输出重定向。

第四篇:文件名匹配/输出重定向


一、文件名匹配

文件名匹配使得您不必一一写出名称,就可以指定多个文件。您将用到一些特殊的字符,称为通配符(wildcards)。

假设您想用’rm’命令删除目录下所有以字符串’.bak’结尾的文件。除了在’rm’后跟上所有文件名作为参数,您还可以用通配符’*':

rm *.bak

 

‘*’可匹配一个或多个字符。在本例中,您告诉 shell 将命令’rm’的参数扩展到”所有以’*.bak’结尾的文件”,shell 就将扩展后的参数告诉’rm’命令。

您将看到,shell 在命令执行前,就将读取并解释命令行。正是因为这个,您才可以将通配符用于 shell 命令的参数中。

让我们更进一步地来认识通配符’*'。假定您有个目录,其中含文件’124.bak’、’346.bak’及’583.bak’。您想只保留文件’583.bak’,可以用:

rm *4*.bak

 

shell 就将’*4*.bak’扩展成”所有含’4′并以’.bak’结尾的字符串”。

注意到 rm 4*.bak 无法工作,因为这匹配的是以’4′开头的文件。由于目录中没有这样的文件,shell 将这个模式扩展为空的字符串,故’rm’将返回出错信息:

rm: cannot remove `4*.bak': No such file or directory

 

如果您想保留文件’345.bak’,而删除’124.bak’和’583.bak’。这看起来有些难度,因为被删文件的名称除了后缀其他都不同。但幸运的是,您可以用不含有来指定文件:

rm *[!6].bak

 

这将被读为:除了以’6.bak’结尾的文件,删除其他所有以’.bak’结尾的文件。您必须将取反号(negation sign)与取反字符(这里是 6)放到括号中,不然的话,shell 会将惊叹号(exclamation mark)解释成历史记录替换的开始(the beginning of a history substitution)。取反号在本篇介绍的所有匹配模式中都有效。

请注意:通配符’*'与取反号连用,很容易产生问题。猜猜

rm *[!6]*.bak

表示什么?这个命令将删除所有文件,甚至包括名称中包含’6′的文件。如果您将通配符’*'放到了取反号前面和后面,实际上取反号将失效,因为 shell 将其解释为”所有名称中任何位置都不含该字符的文件”。在我们的例子里,只有文件’666.bak’不符合该模式。

第二个通配符是问号(question mark):’?'。在匹配时,一个问号只能代表一个字符。为了示范其用途,我们在上例的假设中添加两个新文件:’311.bak~’和’some.text’。现在,列出所有在点号后有四个字符的文件:

ls *.????

 

问号通配符能够有效地避免上面提到的’取反号陷阱’(negation trap):

rm *[!4]?.*

 

将扩展成”所有除了点号前倒数第二个字符为’4′的文件”,也就是只保留文件’346.bak’。

您可能会问,有没有其他匹配方式?到目前为止,您只看到了在指定位置匹配唯一字符的方法。但其实您也可以这样:

ls [13]*

 

将列出所有以字符’1′或’3′开头的文件;在我们的例子中,文件’124.bak’、’311.bak~’和’346.bak’匹配。注意到您必须用中括号将匹配的模式括起来,否则模式只匹配以字符串’13′开头的文件。

接下来,您将高兴地看到还可以定义匹配的范围:

ls *[3-8]?.*

 

将列出所有点号前倒数第二个字符落在’3′到’8′范围的文件。在我们的例子中,匹配的文件是’346.bak’和’583.bak’。

二、引用 shell 的特殊字符

但是,上面的那些机制存在一个缺点:shell 总在命令执行前,试着进行扩展。有时候,会变得很棘手:

l         文件名包含特殊字符。假设您在那个目录中还有一个名为'!56.bak'的文件。下面试图进行模式匹配:
rm !*
rm
rm: too few arguments

 

shell 将’!*’解释成历史记录的替换(加入前一个命令的所有参数),而不是匹配方式。

l 命令本身带特殊字符作参数。一些 Linux 下的命令行工具,比如 (e)grep、sed、awk、find 及 locate ,都使用自己的正则表达式(regular expressions)。这些表达式与模式匹配看起来惊人地相似,但在某些地方又有所不同。

但为了使这些特殊命令生效,shell 就不能先将其当作模式匹配来解释:

find . -name [1-9]* -print
find: paths must precede expression

应该是:

find . -name '[1-9]*' -print
./346.bak
./124.bak
./583.bak

./311.bak~

 

您可以通过反斜线(back slash)来引用特殊字符,比如 ! 、$ 、? 或空格:

ls \!*

!56.bak

 

或者用(单)引号:

ls '!'*

!56.bak

 

请注意,要看清楚引号应该放在什么位置。命令 ls ‘!*’ 将查找名为’!*’的文件,这是由于通配符也在引号间,所以只能依照字面来解释。

三、输出重定向

Unix 的理念是汇集许多小程序,每个东东都有特殊的专长。复杂的任务不是由大型软件完成,而是运用 shell 的机制,组合许多小程序共同完成。重定向就在其中发挥着重要的作用。

1、在多个命令间重定向

这要通过管道(pipe),由管道符号|来标识。语法是:

command1 | command2 | command3 等等

 

这种格式您一定已经见到过了。管道经常将一个程序的输出送到’more’或’less’来阅读。

ls -l | less

 

其中,第一个命令提供目录内容,第二个则将其以翻页的方式显示。更复杂的例子如:

rpm -qa | grep ^x | less

 

第一个命令给出所有已安装的 RPM 包,第二个则将其过滤(filter:’grep’),只剩下以’^x’开头的包,第三个命令则将结果以翻页的方式显示。


2、重定向至文件

有时,您希望将命令的输出结果保存到文件中,或以文件内容作为命令的参数。这可以通过’>’和’<’来实现。

command > file

 

将 command 的输出保存到 file 中,这将覆盖 file 中的内容:

ls > dirlist

 

将当前目录的内容保存到’dirlist’文件。

command < file

 

将 file 内容作为 command 的输入:

sort < dirlist > sdirlist

 

将文件’dirlist’的内容送到命令’sort’,然后再将排序后的结果送到文件’sdirlist’。当然,您也可以一步到位:

ls | sort > sdirlist

 

一种特殊的方式是’command 2> file’。这将 command 执行的出错信息送到 file 中。这个您到时候会需要……

另一种操作符是’>>’,这将输出添加到已存在的文件中:

echo "string" >> file

 

将 string 加到文件 file 中。这是不打开文件而完成编辑的好办法!

但是,’<’和’>’操作符都有一个重要的限制:

command < file1 > file1

 

将删除 file1 的内容,而

command < file1 >> file1

 

却可以很好地工作,将加工过的 file1 内容加回到文件中。

是不是有点多?;-) 不必惊慌,您完全可以按照自己的速度,一步步地来学习。别忘了,实践是最好的学习方法……

熟知了许多 shell 的机制后, 您可能急着想知道如何来定制环境。在后面的两篇中,您将得到这方面的启示。在最后一篇中,还有一段如何处理 shell 出错信息的常见问答(FAQ),及一些配置技巧。

第五篇:bash 配置文件/提示符/改变 $PATH


一、bash 配置文件

在您的 home 目录下,运行

ls .bash*

您将看到这些文件:

l         .bash_history :记录了您以前输入的命令,

l         .bash_logout :当您退出 shell 时,要执行的命令,

l         .bash_profile :当您登入 shell 时,要执行的命令,

l         .bashrc :每次打开新的 shell 时,要执行的命令。

 

请注意后两个的区别:’.bash_profile’只在会话开始时被读取一次,而’.bashrc’则每次打开新的终端(如新的 xterm 窗口)时,都要被读取。按照传统,您得将定义的变量,如 PATH ,放到’.bash_profile’中,而象 aliases(别名)和函数之类,则放在’.bashrc’。但由于’.bash_profile’经常被设置成先读取’.bashrc’的内容,您如果图省事的话,就把所有配置都放进’.bashrc’。

这些文件是每一位用户的设置。系统级的设置存储在’/etc/profile’、’/etc/bashrc’及目录’/etc/profile.d’下的文件中。但您得习惯用各自的配置文件:编辑不需要’root’权限,还可以使您的设置更有个性。当系统级与用户级的设置发生冲突时,将采用用户的设置。

读取’.bashrc’的内容,您如果要省点事的话,就把您所有的配置都放进’.bashrc’。

上面的这些文件是每位用户的设置,系统级的设置存储在’/etc/profile’、’/etc/bashrc’及目录’/etc/profile.d’下的文件中。您最好习惯使用各自的配置文件:编辑不需要’root’权限,还可以使您的设置更具个性。当系统级与用户级的设置发生冲突时,将优先采用用户的设置。

二、提示符

每次当您打开一个控制台(console)或 xterm 时,最先看到的就是提示符(prompt),类似于:

account@hostname ~ $

 

在默认设置下,提示符将显示您的用户名、主机名(默认是’localhost’)、当前所在目录(在 Unix 中,’~'表示您的 home 目录)。

按照传统,最后一个字符可以标识您是普通用户($),还是’root’(#)。

您可以通过 $PS1 变量来设置提示符。命令

echo $PS1

 

将显示当前的设定。其中可用字符的含义在 man bash 的’PROMPTING’部分有说明。

如何才能完成理想的设置呢?对于健忘的初学者来讲,默认设定有些不友好,因为提示符只显示当前目录的最后一部分。如果您看到象这样的提示符

tom@localhost bin $

 

您的当前目录可能是’/bin’、’/usr/bin’、’/usr/local/bin’及’/usr/X11R6/bin’。当然,您可以用

pwd (输出当前目录,print working directory)

能不能叫 shell 自动告诉您当前目录呢?

当然可以。这里我将提到的设定,包括提示符,大都包含在文件’/etc/bashrc’中。您可以通过编辑各自 home 目录下的’.bash_profile’和’.bashrc’来改变设置。

在 man bash 中的’PROMPTING’部分,对这些参数(parameter)有详细说明。您可以加入一些小玩意,如不同格式的当前时间,命令的历史记录号,甚至不同的颜色。

在’~/.bashrc’中,我喜欢的设定是:

PS1="\[\033[1m\][\w]\[\033[0m\] "

‘root’在’~/.bashrc’中的设定 是:

PS1="\[\033[0;31m\][\w]\[\033[0m\] "

 

这样我得到的提示符就是:

[/usr/bin]

 

当用’root’时,变成:

[/usr/bin]

我已经除掉了主机名和用户名,因为我用不着这些。但我首先想一眼就能看出我的身份是普通用户还是’root’。注意到,普通用户的提示符可以是黑底白字,或白底黑字。

要在终端上获得恰当的颜色调配, 您可以编辑下面这个脚本color ,赋予执行权限(chmod +x color),然后再运行。

#!/bin/bash

#

#   This file echoes a bunch of color codes to the

#   terminal to demonstrate what's available.  Each

#   line is the color code of one forground color,

#   out of 17 (default + 16 escapes), followed by a

#   test use of that color on all nine background

#   colors (default + 8 escapes).

#

T='gYw'   # The test text

echo -e "\n                 40m     41m     42m     43m\

     44m     45m     46m     47m";

for FGs in '    m' '   1m' '  30m' '1;30m' '  31m' '1;31m' '  32m' \

           '1;32m' '  33m' '1;33m' '  34m' '1;34m' '  35m' '1;35m' \

           '  36m' '1;36m' '  37m' '1;37m';

  do FG=${FGs// /}

  echo -en " $FGs \033[$FG  $T  "

  for BG in 40m 41m 42m 43m 44m 45m 46m 47m;

    do echo -en "$EINS \033[$FG\033[$BG  $T  \033[0m";

  done

  echo;

done

echo

 

一种更适当的设定:

PS1="\u: \w\\$ "

这样,提示符就变成:

user_name: /usr/bin$

 

您可以通过命令 export 来测试不同的设置(比如,export PS1=”\u: \w\\$ “)。如果找到了适合的提示符,就将设置放到您的’.bashrc”中。这样,每次打开控制台或终端窗口时,都会生效。

您甚至可以给提示符设定主题(theme),也就是搭配不同的颜色,使其看起来象很棒的 ol
的 C64 提示符。如果您对此感兴趣,可以看一下
Bashish( http://hem.passagen.se/arnognulf/index2.html )。

三、改变 $PATH

‘$PATH’与’$PS1′一样,也是环境变量。输入

set

 

将列出所有当前定义的环境变量。

您看到的这些环境变量在 shell 的配置文件中定义,可能是用户自己的配置文件,也可能是由’root’通过’/etc’下面的系统级文件定义的。如果您使用 X ,更多的一些变量将由 X 、您的窗口管理器或桌面环境的启动文件配置。

如果对这些设置不很清楚,您暂时最好不要随便改动。了解如何改变 $PATH 变量很有用,因为这个变量决定了 shell 将到哪些目录中寻找命令或程序。如果要执行的命令的目录在 $PATH 中,您就不必输入这个命令的完整路径,直接输入命令就可以了。一些第三方软件没有将可执行文件放到 Linux 的标准目录中。因此,将这些非标准的安装目录添加到 $PATH 是一种解决的办法。此外,您也将看到如何处理一般的环境变量。

首先,作为惯例,所有环境变量名都是大写。由于 Linux 区分大小写,这点您要留意。当然,您可以自己定义一些变量,如’$path’、’$pAtH’,但 shell 不会理睬这些变量。

第二点是变量名有时候以’$'开头,但有时又不是。当设置一个变量时,您直接用名称,而不需要加’$':

PATH=/usr/bin:/bin:/usr/local/bin:/usr/X11R6/bin

要获取变量值的话,就要在变量名前加’$':

echo $PATH

/usr/bin:/bin:/usr/local/bin:/usr/X11R6/bin

否则的话,变量名就会被当作普通文本了:

echo PATH

PATH

 

处理 $PATH 变量要注意的第三点是:您不能只替换变量,而是要将新的字符串添加到原来的值中。在大多数情况下,您不能用’PATH=/some/directory’,因为这将删除 $PATH 中其他的所有目录,这样您在该终端运行程序时,就不得不给出完整路径。所以,只能作添加:

PATH=$PATH:/some/directory

这样,PATH 被设成当前的值(以 $PATH 来表示)+新添的目录。

到目前为止,您只为当前终端设置了新的 $PATH 变量。如果您打开一个新的终端,运行 echo $PATH ,将返回旧的 $PATH 值,而看不到您刚才添加的新目录。因为您先前定义的是一个局部环境变量(仅限于当前的终端)。

要定义一个全局变量,使在以后打开的终端中生效,您需要将局部变量输出(export),可以用’export’命令:

export PATH=$PATH:/some/directory

 

现在如果您打开一个新的终端,输入 echo $PATH ,也能看到新设置的 $PATH 了。请注意,命令’export’只能改变当前终端及以后运行的终端里的变量。对于已经运行的终端没有作用。

为了将目录永久添加到您的 $PATH ,只要将’export’的那行添加到您的’.bash_profile’文件中。

请不要在’.bashrc’中设置 PATH ,否则会导致 PATH 中目录的意外增长。您每次打开一个新的 shell ,’.bashrc’都会作用。所以如果在该文件中添加目录,您每次打开一个终端,目录又会被添加。这将导致 PATH 变量由于目录复制,不断地增长。

第六篇:命令的别名、Shell 函数/从这里出发/Shell 常见问题


一、命令的别名、Shell 函数

记住所有的命令及各自带的可选项,然后每次一一输入,这确实有点枯燥。但幸运的是,您可以为常用命令定义快捷方式。这些快捷方式可以用较简单的命令别名(alias),或复杂一些的 shell 函数的语法来定义。

1、命令的别名

例如,我用下面的命令来上传 MUO 中的文件:

rsync -e ssh -z -t -r -vv --progress /home/tom/web/muo/rsmuo/docs muo:/www/mandrakeuser/docs

 

显然,如果每次都要逐一输入,那我早晚会变成木头。因此我在’~/.bashrc’中定义了别名:

alias upmuo='rsync -e ssh -z -t -r -vv --progress /home/tom/web/muo/rsmuo/docs muo:/www/mandrakeuser/docs'

 

现在,我只要输入 upmuo 就可以完成上传任务了。

定义别名的语法是:

alias shortcut='command'

 

命令中有空格的话 ,就需要用引号(如在命令与可选项间就有空格)。请注意,您可以用单引号或双引号,但他们是有区别的。

单引号将剥夺其中的所有字符的特殊含义,而双引号中的’$'(参数替换)和’`'(命令替换)是例外。这意味着,如果您想在别名中应用变量或命令的替换,就得用双引号。看一下上面的例子,我在’.bashrc’中定义了一个称为 MUOHOME 的变量:

export MUOHOME=$HOME/web/muo/rsmuo/docs

 

要在上面的别名中用上这个变量,我就必须用双引号:

alias upmuo="rsync -e ssh -z -t -r -vv --progress $MUOHOME muo:/www/mandrakeuser/docs"

 

否则,别名将查找一个名为’$MUOHOME’的目录或文件。

您可以用’alias’在命令行快速地创建别名,或将命令放到各自的’~/.bashrc’,或放到系统级的’/etc/profile.d/alias.sh’中(而在 Mandrake Linux 8 以前的版本里,用的是’/etc/bashrc’)。要删除一个别名,只要输入:unalias alias 。运行 alias 将列出您系统中所有定义的别名。

如果看一下’~/.bashrc’和’/etc/profile.d/alias.sh’,您会发现系统已经定义了一些别名。您可以为同一个命令定义多个别名。当然,您得先确认别名与其他程序名不同,比如象 alias rm=’ls -l’ 这样的就不能工作。您可以在命令行输入这些快捷方式,测试一下。如果 shell 找不到相同名称的命令,那您就可以将其用作别名了。

以下别名可能有用(不要忘了引号!) :

l         alias rpmq='rpm -qa | grep' :现在 rpmq string 就将列出所有名称中含有 string 的已安装 RPM 包,

l         alias ls='ls -ho --color | more' :ls 将以彩色分页方式列出文件,文件大小以 KB为单位,

l         alias use='du --max-depth=1 | sort -n | more' :use 将子目录按大小排好,并以分页方式列出,

 

目录的别名也可以是可移动的介质:alias dlm=’/mnt/cdrom/RedHat/RPMS/’ 。

提示:将有相似功能的别名以相同字母开头,比如将所有目录的别名以’d'作开头,这样有助于记忆。

我相信,您将会用到这些功能。


2、Shell 函数

写 shell 函数涉及到了 shell 脚本,这超出了我们讨论的范围(也不在我的掌握范围之内 ;-) )。事实上,shell 函数属于 shell 脚本,但可以在同一 shell 下被预加载(preload)和执行(而一般的 shell 脚本至少要打开一个 sub-shell)。

通过 shell 函数,您可以做很多 aliases 无法完成的事情。下面就是一个例子:

function apros() { apropos $1 | egrep -v '(3|\(n\)'; }

定义了一个新命令,称为’apros’。apros name 将先执行’apropos name’(即在 man page 中搜索命令),然后将得到的输出送到管道(|),接着用’egrep’过滤,排除第’3′和第’n'章节的 man page ,这个命令可能没什么大用处,但可以整理’apropos’命令的输出。

函数允许您在函数内部任何位置,使用运行时的参数。而别名,则只允许在命令行尾放一个参数(比如前面的别名’rpmq’)。

‘$1′就是位置参数(positional parameter),表示函数第一个参数的位置标识符。依此类推,还有’$2′等。

function apros() { apropos $1 | egrep -v "\($2"; }

 

如果您这样运行’apros’命令:

apros name man_section_number

 

这个命令将搜索标题中含 name 的 man pages ,但排除 man_section_number 部分:

apros menu 3

 

将搜索标题含’menu’的 man page ,但排除第三章节(关于编程的)。注意到您得引用(quote) 两次,而且还用到了双引号:

l         您必须引用'egrep'的搜索模式,这样可以不至于被 shell 误解。

l         您必须用双引号,这样第二个参数才能被正确解释。

l         您必须引用圆括号,这样使'egrep'按字面意思对待对待参数。

 

是不是有点意思?;-)

shell 函数的处理类似于别名:将其放到您的’.bashrc’文件,这样就能永久生效了。


二、从这里出发

我们谈到的只是 shell 的一个开头。掌握了shell 脚本,您就可以做很多事情,比如将任务自动化,纠正别人脚本中的错误,按照您的习惯定制 Linux 系统。如果您打算学习某种复杂的编程语言,那 shell 脚本也是一个很好的开端,因为基本概念都是类似的。

BASH Programming – Introduction HOW-TO:

http://www.ibiblio.org/mdw/HOWTO/Bash-Prog-Intro-HOWTO.html

将更深入这些主题,并且将把您带到 shell 编程的世界。然后可以继续阅读我强烈推荐的 Advanced Bash-Scripting Guide( http://www.ibiblio.org/mdw/LDP/abs/html/index.html ),作者是:Mendel Cooper 。

如果您偏好纸书,那我推荐 S. Veeraraghavan 的《Teach Yourself Shell Programming》,Sams 出版社。我倒觉得 O’Reilly 公司由 Newham/Rosenblatt 写的《Learning the bash Shell》,不过尔尔,但这可能只有我这么看 ;-)

除了这些,就是练习,练习,再练习。阅读其他人写的 shell 脚本,看看他们在做什么,怎么做,为什么那样做。

请不要用’root’测试您的脚本。Have fun 。

关于本文

flaboy兄当时整理时发在了LinuxSir.Org 论坛 SHELL讨论区中,是作为SHELL 基础教材的形式帖出的,他在整理时“本文编译整理时对相关章节做了相应的删改处理,去掉了针对Mandrake Linux的部分内容。”;

我看这个文档极其不错,进行了再次整理,并对文档的结构进行了调整,根据文档的内容,我适当的加了序列号,并做了一个目录,主要是方便大家阅读;感谢原作者及中译者,同时也感谢flaboy兄的整理;

我想整理文档也是件比较幸福的事情,至少在看文档的时候,我能知道有哪些是比较关健的,也能得到我想要学的东西,所以我会一直整理文档;如果时间允许并有能力的话,我也会写一些。── 北南南北

整理文档是没任何技术含量的,为什么你会做呢?

虽然没有技术含量,但我想如果能为他人带来一点点方便,我想我所做的还是值得的,相对原创者和中译者来说,我做的又算什么呢?

── 北南南北

相关文档 

《详解Bash命令行处理》

rsync 服务器架设方法 v0.1b (正在修订中)

作者: 北南南北
来自:Linuxsir.Org
摘要: rsync 是一个快速增量文件传输工具,它可以用于在同一主机备份内部的备分,我们还可以把它作为不同主机网络备份工具之用。本文主要讲述的是如何自架rsync服务器,以实现文件传输、备份和镜像。相对tar和wget来说,rsync 也有其自身的优点,比如速度快、安全、高效;


+++++++++++++++++++++++++++++++++++
正文
+++++++++++++++++++++++++++++++++++


1 什么是rsync;

    rsync is a file transfer program for Unix systems. rsync uses the “rsync algorithm” which provides a very fast method for bringing remote files into sync. It does this by sending just the differences in the files across the link, without requiring that both sets of files are present at one of the ends of the link beforehand. rsync 是一个Unix系统下的文件同步和传输工具。rsync是用 “rsync 算法”提供了一个客户机和远程文件服务器的文件同步的快速方法。

    Some features of rsync include
    rsync 包括如下的一些特性:
    * can update whole directory trees and filesystems
    能更新整个目录和树和文件系统;
    * optionally preserves symbolic links, hard links, file ownership, permissions, devices and times
    有选择性的保持符号链链、硬链接、文件属于、权限、设备以及时间等;
    * requires no special privileges to install
    对于安装来说,无任何特殊权限要求;
    * internal pipelining reduces latency for multiple files
    对于多个文件来说,内部流水线减少文件等待的延时;
    * can use rsh, ssh or direct sockets as the transport
    能用rsh、ssh 或直接端口做为传输入端口;
    * supports anonymous rsync which is ideal for mirroring
    支持匿名rsync 同步文件,是理想的镜像工具;


2 rsync 服务器的理由;

    rsync 服务器架设比较简单,可能我们安装好rsync后,并没有发现配置文件,以及rsync服务器启动程序,因为每个管理员可能对rsync 用途不一样,所以一般的发行版只是安装好软件就完事了,让管理员来根据自己的用途和方向来自己架设rsync服务器;因为这个rsync应用比较广,能在同一台主机进行备份工作,还能在不同主机之间进行工作。在不同主机之间的进行备份,是必须架设rsync 服务器的。以我的观点上看,如果在同一台主机进行文件的备分,用复制工具cp就好了。没必要用rsync 这么相对复杂的工具,cp也简单易用,当然这仅仅是个人观点;

    对于重量级服务器来说,应该有网络备份服务器来说,只有本地备份还是不够的,最好还是有网络备份主机,这样数据的安全才有保证。毕竟数据放在服务器本地上还是不太安全,比如磁盘坏掉、被骇客攻入服务器删除数据。其实服务器本身价值并不大,重要的是数据的价值。

    另外对于大量文件从一台服务器上迁移到另一台服务器上,rsync 的确是一个不可不用传输工具。公司有一台文件服务器,配置是CPU Intel Celeon 333Mhz,内存128M,硬盘IDE 80Gx3=240G,里面仅是第一个硬盘的12G的分区安装系统,用了256M做为交换分区,其它的空间我都用来存数据,通过LVM卷来管理磁盘空间,我分了一个 180G的空间给数据存放,当时数据存储容量已经达到了160多G。当时的情况是服务器空间有限,没做本地备份。更不可能新增硬盘上去,因为这台机器没做RAID,硬盘坏掉一个,数据会全毁掉,安全性没有一点保障。在这种情况下,为了保证数据的安全性,我被迫做了一台带有Raid5支持的文件服务器。在选择如何把数据文件完整的传输到新服务器上,我想到了很多的工具,最后想到了rsync 。我花了十分钟架设并调试了rsync ,然后就开工文件传输,因为文件服务器上的文件太多,老的文件服务器配置又低,大约花了两三天吧才得以把所有文件传输完毕。


3 架设rsync服务器过程;

    架设rsync 服务器比较简单,写一个配置文件rsyncd.conf 。文件的书写也是有规则的,我们可以参照rsync.samba.org 上的文档来做;当然我们首先要安装好rsync 这个软件才行;


3.1 rsync的安装;

    软件安装过于简单,现在Linux各大发行版都提供这个软件包,当然您也可以自己编译安装,在目前的情况下,我看没太大的必要; 

    [root@linuxsir:beinan]$ sudo apt-get  install  rsync  注:在debian、ubuntu 等在线安装方法;
    [root@linuxsir:beinan]# slackpkg  install  rsync   注:Slackware 软件包在线安装;
    [root@linuxsir:beinan]# yum install rsync    注:Fedora、Redhat 等系统安装方法;

     

    其它Linux发行版,请用相应的软件包管理方法来安装;如果是源码包,也就是用下面的办法;

    [root@linuxsir:/home/beinan]# tar xvf  sync-xxxx.tar.gz 或sync-xxx.tar.bz2
    [root@linuxsir:/home/beinan]# cd  sync-xxx
    [root@linuxsir:/home/beinan/sync-xxx]# ./configure --prefix=/usr  ;make ;make install   注:在用源码包编译安装之前,您得安装gcc等编译开具才行;

     


3.2 rsync服务器的配置文件rsyncd.conf ;

    我们可以参照 rsyncd.conf.html。具体步骤如下; 

    [root@linuxsir:~]#mkdir /etc/rsyncd  注:在/etc目录下创建一个rsyncd的目录,我们用来存放rsyncd.conf 和rsyncd.secrets文件;
    [root@linuxsir:~]#touch /etc/rsyncd/rsyncd.conf  注:创建rsyncd.conf ,这是rsync服务器的配置文件;
    [root@linuxsir:~]#touch /etc/rsyncd/rsyncd.secrets  注:创建rsyncd.secrets ,这是用户密码文件;
    [root@linuxsir:~]#chmod 600 /etc/rsyncd/rsyncd.secrets  注:为了密码的安全性,我们把权限设为600;
    [root@linuxsir:~]#ls -lh /etc/rsyncd/rsyncd.secrets
    -rw------- 1 root root 14 2007-07-15 10:21 /etc/rsyncd/rsyncd.secrets
    [root@linuxsir:~]#touch /etc/rsyncd/rsyncd.motd

     

    下一就是我们修改 rsyncd.conf 和rsyncd.secrets 和rsyncd.motd 文件的时候了;

    rsyncd.conf 是rsync服务器主要配置文件,我们来个简单的示例;比如我们要备份服务器上的 /home 和/opt ,在/home中,我想把beinan和samba目录排除在外;

    # Distributed under the terms of the GNU General Public License v2

    # Minimal configuration file for rsync daemon
    # See rsync(1) and rsyncd.conf(5) man pages for help
     

    注: 关于 auth users 是必须在服务器上存在的真实的系统用户,如果你想用多个用户,那就以,号隔开;比如 auth users = beinan , linuxsir

    密码文件:/etc/rsyncd/rsyncd.secrets的内容格式;

    用户名:密码

     

    而我们在例子中rsyncd.secrets的内容如下类似的;在文档中说,有些系统不支持长密码,自己尝试着设置一下吧。另外 rsyncd.secrets文件权限对其它用户组是不可读的。如果你设置错了,可能rsync不工作。

    linuxsir:222222
    beinan:333333

     

    注: 这里的密码值得注意,为了安全,你不能把系统用户的密码写在这里。比如你的系统用户 linuxsir 密码是 abcdefg ,为了安全,你可以让rsync 中的linuxsir 为 222222 。这和samba的用户认证的密码原理是差不多的;

    rsyncd.motd 文件;

    它是定义rysnc 服务器信息的,也就是用户登录信息。比如让用户知道这个服务器是谁提供的等;类似ftp服务器登录时,我们所看到的 linuxsir.org ftp ……。 当然这在全局定义变量时,并不是必须的,你可以用#号注掉,或删除;我在这里写了一个 rsyncd.motd的内容为:

    +++++++++++++++++++++++++++
    + linuxsir.org  rsync  2002-2007 +
    +++++++++++++++++++++++++++

     

    # This line is required by the /etc/init.d/rsyncd script
    pid file = /var/run/rsyncd.pid  
    port = 873
    address = 192.168.1.171 
    #uid = nobody
    #gid = nobody   
    uid = root  
    gid = root  

    use chroot = yes 
    read only = yes 

    #limit access to private LANs
    hosts allow=192.168.1.0/255.255.255.0 10.0.1.0/255.255.255.0 
    hosts deny=*

    max connections = 5
    motd file = /etc/rsyncd/rsyncd.motd

    #This will give you a separate log file
    #log file = /var/log/rsync.log

    #This will log every file transferred – up to 85,000+ per user, per sync
    #transfer logging = yes

    log format = %t %a %m %f %b
    syslog facility = local3
    timeout = 300

    [linuxsirhome]  
    path = /home   
    list=yes
    ignore errors
    auth users = linuxsir
    secrets file = /etc/rsyncd/rsyncd.secrets 
    comment = linuxsir home 
    exclude =   beinan/  samba/     

    [beinan]
    path = /opt
    list=no
    ignore errors
    comment = optdir  
    auth users = beinan
    secrets file = /etc/rsyncd/rsyncd.secrets


4 架设rsync服务器的示例说明;


4.1 全局定义;

    在rsync 服务器中,全局定义有几个比较关健的,根据我们前面所给的配置文件 rsyncd.conf 文件; 

    pid file = /var/run/rsyncd.pid   注:告诉进程写到 /var/run/rsyncd.pid 文件中;
    port = 873  注:指定运行端口,默认是873,您可以自己指定;
    address = 192.168.1.171  注:指定服务器IP地址;
    uid = nobody  
    gid = nobdoy  

     

    注:服务器端传输文件时,要发哪个用户和用户组来执行,默认是nobody。 如果用nobody 用户和用户组,可能遇到权限问题,有些文件从服务器上拉不下来。所以我就偷懒,为了方便,用了root 。不过您可以在定义要同步的目录时定义的模块中指定用户来解决权限的问题。

    use chroot = yes 

     

    注:用chroot,在传输文件之前,服务器守护程序在将chroot 到文件系统中的目录中,这样做的好处是可能保护系统被安装漏洞侵袭的可能。缺点是需要超级用户权限。另外对符号链接文件,将会排除在外。也就是说,你在rsync服务器上,如果有符号链接,你在备份服务器上运行客户端的同步数据时,只会把符号链接名同步下来,并不会同步符号链接的内容;这个需要自己来尝试;

    read only = yes 

     

    注:read only 是只读选择,也就是说,不让客户端上传文件到服务器上。还有一个 write only选项,自己尝试是做什么用的吧;

    #limit access to private LANs
    hosts allow=192.168.1.0/255.255.255.0 10.0.1.0/255.255.255.0 

     

    注:在您可以指定单个IP,也可以指定整个网段,能提高安全性。格式是ip 与ip 之间、ip和网段之间、网段和网段之间要用空格隔开;

    max connections = 5   

     

    注:客户端最多连接数;

    motd file = /etc/rsyncd/rsyncd.motd

     

    注:motd file 是定义服务器信息的,要自己写 rsyncd.motd 文件内容。当用户登录时会看到这个信息。比如我写的是:

    +++++++++++++++++++++++++++
    + linuxsir.org  rsync  2002-2007 +
    +++++++++++++++++++++++++++

     

    log file = /var/log/rsync.log

     

    注:rsync 服务器的日志;

    transfer logging = yes

     

    注:这是传输文件的日志;

    log format = %t %a %m %f %b
    syslog facility = local3
    timeout = 300

     


4.2 模块定义;

    模块定义什么呢?主要是定义服务器哪个目录要被同步。每个模块都要以[name]形式。这个名字就是在rsync 客户端看到的名字,其实有点象Samba服务器提供的共享名。而服务器真正同步的数据是通过 path 来指定的。我们可以根据自己的需要,来指定多个模块。每个模块要指定认证用户,密码文件、但排除并不是必须的; 下面前面配置文件模块的例子:

    [linuxsirhome]  
    注:模块,它为我们提供了一个链接的名字,链接到哪呢,在本模块中,链接到了/home目录;要用[name] 形式;
    path = /home    注:指定文件目录所在位置,这是必须指定的;
    auth users = linuxsir   注:认证用户是linuxsir  ,是必须在 服务器上存在的用户;
    list=yes   注:list 意思是把rsync 服务器上提供同步数据的目录在服务器上模块是否显示列出来。默认是yes 。如果你不想列出来,就no ;如果是no是比较安全的,至少别人不知道你的服务器上提供了哪些目录。你自己知道就行了;
    ignore errors  注:忽略IO错误,详细的请查文档;
    secrets file = /etc/rsyncd/rsyncd.secrets   注:密码存在哪个文件;
    comment = linuxsir home  data  注:注释可以自己定义,写什么都行,写点相关的内容就行;
    exclude =   beinan/   samba/     

    注:exclude 是排除的意思,也就是说,要把/home目录下的beinan和samba 排除在外; beinan/和samba/目录之间有空格分开 ; 

    [beinan] 
    path = /opt  注:指定文件目录所在位置;
    list=no
    comment = optdir  
    auth users = beinan  注:是必段在服务器上存在的用户;
    secrets file = /etc/rsyncd/rsyncd.secrets
    ignore errors

     


5 启动rsync 服务器及防火墙的设置;


5.1 启动rsync服务器;

    启动rsync 服务器相当简单,–daemon 是让rsync 以服务器模式运行; 

    [root@linuxsir:~]#/usr/bin/rsync --daemon  --config=/etc/rsyncd/rsyncd.conf

     

    注: 如果你找不到rsync 命令,你应该知道rsync 是安装在哪了。比如rsync 可执行命令可能安装在了 /usr/local/bin目录;也就是如下的命令;

    [root@linuxsir:~]#/usr/local/bin/rsync --daemon  --config=/etc/rsyncd/rsyncd.conf

     

    当然您也可以写一个脚本来开机自动启动rysnc 服务器,你自己查查文档试试,这个简单。因为我用slackware 也有一个类似的脚本。我感觉不如直接手工运行方面,或者把这个命令写入rc.local文件中,这样也一样能自动运行;


5.2 rsync服务器和防火墙;

    Linux 防火墙是用iptables,所以我们至少在服务器端要让你所定义的rsync 服务器端口通过,客户端上也应该让通过。 

    [root@linuxsir:~]#iptables -A INPUT -p tcp -m state --state NEW  -m tcp --dport 873 -j ACCEPT
    [root@linuxsir:~]#iptables -L  查看一下防火墙是不是打开了 873端口;

     


6 通过rsync客户端来同步数据;


6.1 列出rsync 服务器上的所提供的同步内容;


    首先:我们看看rsync服务器上提供了哪些可用的数据源;
     

    [beinan@beinnaIBM:~]$ rsync  --list-only  linuxsir@linuxsir.org:: 

    注: 前面是rsync 所提供的数据源,也就是我们在rsyncd.conf 中所写的[linuxsirhome]模块。而“linuxsir home data”是由[linuxsirhome]模块中的 comment = linuxsir home data 提供的;为什么没有把beinan 数据源列出来呢?因为我们在[beinan]中已经把list=no了。

    [beinan@beinnaIBM:~]$ rsync  --list-only  linuxsir@linuxsir.org::linuxsirhome 

     

    试试这个?

    +++++++++++++++++++++++++++++++++
    +++ linuxsir.org rsync  server ++
    +++++++++++++++++++++++++++++++++

    linuxsirhome    linuxsir  home data


6.2 rsync 客户端同步数据;

    [beinan@beinnaIBM:~]$ rsync -avzP linuxsir@linuxsir.org::linuxsirhome   linuxsirhome
    Password: 这里要输入linuxsir的密码,是服务器端提供的,在前面的例子中,我们用的是 222222,输入的密码并不显示出来;输好后就回车;

     

    注: 这个命令的意思就是说,用linuxsir 用户登录到服务器上,把linuxsirhome数据,同步到本地目录linuxsirhome上。当然本地的目录是可以你自己定义的,比如 linuxsir也是可以的;当你在客户端上,当前操作的目录下没有linuxsirhome这个目录时,系统会自动为你创建一个;当存在linuxsirhome这个目录中,你要注意它的写权限。

    说明:

    -a 参数,相当于-rlptgoD,-r 是递归 -l 是链接文件,意思是拷贝链接文件;-p 表示保持文件原有权限;-t 保持文件原有时间;-g 保持文件原有用户组;-o 保持文件原有属主;-D 相当于块设备文件;

    -z 传输时压缩;
    -P 传输进度;
    -v 传输时的进度等信息,和-P有点关系,自己试试。可以看文档;

    [beinan@beinnaIBM:~]$ rsync -avzP  --delete linuxsir@linuxsir.org::linuxsirhome   linuxsirhome

     

    这回我们引入一个 –delete 选项,表示客户端上的数据要与服务器端完全一致,如果 linuxsirhome目录中有服务器上不存在的文件,则删除。最终目的是让linuxsirhome目录上的数据完全与服务器上保持一致;用的时候要小心点,最好不要把已经有重要数所据的目录,当做本地更新目录,否则会把你的数据全部删除;

    [beinan@beinnaIBM:~]$ rsync -avzP  --delete  --password-file=rsync.password  linuxsir@linuxsir.org::linuxsirhome   linuxsirhome

     

    这次我们加了一个选项 –password-file=rsync.password ,这是当我们以linuxsir用户登录rsync服务器同步数据时,密码将读取 rsync.password 这个文件。这个文件内容只是linuxsir用户的密码。我们要如下做;

    [beinan@beinnaIBM:~]$ touch rsync.password
    [beinan@beinnaIBM:~]$ chmod 600 rsync.passwod
    [beinan@beinnaIBM:~]$ echo "222222"> rsync.password
     

    注: 这样就不需要密码了;其实这是比较重要的,因为服务器通过crond 计划任务还是有必要的;

    [beinan@beinnaIBM:~]$ rsync -avzP  –delete  –password-file=rsync.password  linuxsir@linuxsir.org::linuxsirhome   linuxsirhome


6.3 让rsync 客户端自动与服务器同步数据;

    服务器是重量级应用,所以数据的网络备份还是极为重要的。我们可以在生产型服务器上配置好rsync 服务器。我们可以把一台装有rysnc机器当做是备份服务器。让这台备份服务器,每天在早上4点开始同步服务器上的数据;并且每个备份都是完整备份。有时硬盘坏掉,或者服务器数据被删除,完整备份还是相当重要的。这种备份相当于每天为服务器的数据做一个镜像,当生产型服务器发生事故时,我们可以轻松恢复数据,能把数据损失降到最低;是不是这么回事??第一步:创建同步脚本和密码文件


    [beinan@beinnaIBM:~] mkdir   /etc/cron.daily.rsync
    [beinan@beinnaIBM:~] cd  /etc/cron.daily.rsync
    [beinan@beinnaIBM:~] touch linuxsir.sh  beinan.sh
    [beinan@beinnaIBM:~] chmod 755 /etc/cron.daily.rsync/*.sh 
    [beinan@beinnaIBM:~] mkdir /etc/rsyncd/
    [beinan@beinnaIBM:~] touch  /etc/rsyncd/rsynclinuxsir.password
    [beinan@beinnaIBM:~] touch  /etc/rsyncd/rsyncbeinan.password
    [beinan@beinnaIBM:~] chmod 600  /etc/rsyncd/rsyncbeinan.*

     

    注: 我们在 /etc/cron.daily/ 中创建了两个文件beinan.sh和linuxsir.sh ,并且是权限是 755的。创建了两个密码文件,linuxsir用户用的是 rsynclinuxsir.password ,而beinan用户用的是 rsyncbeinan.password ,权限是600;

    我们编辑linuxsir.sh,内容是如下的:

    #!/bin/sh
    #linuxsir.org home backup
    /usr/bin/rsync   -avzP  --password-file=/etc/rsyncd/rsynclinuxsir.password   linuxsir@192.168.1.171::linuxsirhome   /home/linuxsirhome/$(date +'%m-%d-%y')

     

    我们编辑 beinan.sh ,内容是:

    #!/bin/sh
    #linuxsir.org beinan home backup
    /usr/bin/rsync   -avzP  --password-file=/etc/rsyncd/rsyncbeinan.password   linuxsir@192.168.1.171::beinan   /home/beinanhome/$(date +'%m-%d-%y')

     

    注:你可以把linuxsir.sh 和beinan.sh 的内容合并到一个文件中,比如都写到 linuxsir.sh 中;

    接着我们修改 /etc/rsyncd/rsynclinuxsir.password 和 rsyncbeinan.password的内容;

    [beinan@beinnaIBM:~] echo "222222" > /etc/rsyncd/rsynclinuxsir.password
    [beinan@beinnaIBM:~] echo "333333"> /etc/rsyncd/rsyncbeinan.password

     

    然后我们再/home目录下创建linuxsirhome 和beinanhome两个目录,意思是服务器端的linuxsirhome数据同步到备份服务器上的/home/linuxsirhome下,beinan数据同步到 /home/beinanhome/目录下。并按年月日归档创建目录;每天备份都存档;

    [beinan@beinnaIBM:~] mkdir /home/linuxsirhome
    [beinan@beinnaIBM:~] mkdir /home/beinanhome

     

    第二步:修改crond服务器的配置文件

    [beinan@beinnaIBM:~] crontab  -e

     

    加入下面的内容:

    # Run daily cron jobs at 4:10 every day  backup linuxsir data: 
    10 4 * * * /usr/bin/run-parts   /etc/cron.daily.rsync    1> /dev/null

     

    注:
    第一行是注释,是说明内容,这样能自己记住。
    第二行表示在每天早上4点10分的时候,运行 /etc/cron.daily.rsync 下的可执行脚本任务;

    第三步:重启crond服务器;

    配置好后,要重启crond 服务器;

    [beinan@beinnaIBM:~]# killall crond    注:杀死crond 服务器的进程;
    [beinan@beinnaIBM:~]# ps aux |grep crond  注:查看一下是否被杀死;
    [beinan@beinnaIBM:~]# /usr/sbin/crond    注:启动 crond 服务器;
    [beinan@beinnaIBM:~]# ps aux  |grep crond  注:查看一下是否启动了?
    root      3815  0.0  0.0   1860   664 ?        S    14:44   0:00 /usr/sbin/crond
    root      3819  0.0  0.0   2188   808 pts/1    S+   14:45   0:00 grep crond

     


7 问题处理;

    当同步出现错误时,可能是你的密码文件权限的问题,或者格式不对,也可能是你复制、粘贴造成的。
    另外权限的问题也应该关注一下,这是最容易出问题的地方;如果您对权限不太了解,应该在LinuxSir.Org 上查找用户和用户组,以及权限方面的知识;


8 未尽事宜;

    本文并不是大而全的rsync 说明文档,仅仅是把网络同步文件的内容说一说,也不一定能完全能说的清楚;对于任何一个工具来说,都足可以写一本书,所以本文也不可能完全全面的介绍rsync 。如果您想了解更多,请参考官方文档或者man ,我也一样查看这些内容来应用rsync 。 如果我们用普通用户用rsync 进行与服务器同步数据时,同步下来的数据,可能属主会改变。为了保持文件的属主和用户组与服务器端完全一致,用root来运行rsync 就可以了……


9 关于本文;

    本文是为了解决公司内部文件上服务器数据迁移到新的服务器应用时而写的,并不是特地为写rysnc 而写rsync 。我用不到的东西,或者我不会的东西,我想写也写不出来;其实我什么不会,只是想应用时,才临时抱佛脚,解决好问题后就忘掉。等用到的时候再查我写过的东西…… :(  本文适合对象,面向所有初学者, 最好是一点Linux也不懂的。如果您是高级应用者,http://rsync.samba.org 完全能满足我们的需要;

配置 Apache MIME 类型,为 XHTML 发送正确的

本文将向您展示:如何配置 Apache 以便为支持可扩展超文本标记语言(Extensible Hypertext Markup Language,XHTML)的浏览器标记文档的媒体类型为 application/xhtml+xml,同时仍然向不支持该语言的浏览器(如 Microsoft® Internet Explorer®)发送 text/html。

当 Web 服务器向浏览器发送文档时,它会给文档加上一个响应报头作为前缀,如 清单 1 所示。此报头包含了用于告诉浏览器如何解释文档的元数据。元数据的一个最重要的部分是最后一行中的 Content-Type。它将告诉浏览器如何呈现内容。例如,浏览器用于显示 JPEG 和 GIF 的代码是不同的。最重要的是,很多浏览器用于显示 XHTML 和超文本标记语言(Hypertext Markup Language,HTML)的代码也是不同的。

清单 1. 一个典型的 HTTP 响应报头

                HTTP/1.1 200 OK
            Date: Thu, 04 Jan 2007 19:39:13 GMT
            Server: Apache/2
            Last-Modified: Wed, 06 Sep 2006 11:19:37 GMT
            ETag: "4dfce0-c4aa-26828440"
            Accept-Ranges: bytes
            Content-Length: 50346
            Content-Style-Type: text/css
            Content-Type: application/xhtml+xml

 

Web 服务器应该给 XHTML 文档加上媒体类型标记 application/xhtml+xml。识别此媒体类型的 Web 浏览器就会相应地将其以 strict 模式而不是 tag soup 模式运行。这使浏览器能更可靠地进行显示,对于级联样式表(Cascading Style Sheets,CSS)布局和基于文档的对象模型的 JavaScript ™ 程序,这一点尤为重要。事实上,在一些情况下同一文档能以两种不同的方式显示,取决于其处理模式是 tag soup 还是 strict。如果想要生成格式良好甚或是有效的 XHTML,则 strict 模式将会是您计划使用并希望使用的模式。

不支持 XHTML 的浏览器也能以 tag soup 模式处理格式良好的文档。结果并不完美,但可以满足一小部分使用很老的浏览器的用户需要。对于大部分使用不符合标准的 Internet Explorer 的用户来说,结果也还可以接受。但是,当前版本的 Internet Explorer(包括 6 和 7)无法识别 application/xhtml+xml 媒体类型。如果向 Internet Explorer 发送 application/xhtml+xml 文档,它将会反过来要求您保存文件,如 图 1 所示。
图 1. Internet Explorer 不知道如何处理 application/xhtml+xml
某些文件可能会损害您的计算机。如果下面的文件信息看起来可疑,或者您不完全信任其来源,请不要打开或保存此文件。文件名:a.xhtml

因此,处理 XHTML 时,要获得最大的兼容性,就需要向 Firefox、Safari、Opera 和其他符合标准的浏览器发送 application/xhtml+xml,而向 Internet Explorer 发送 text/html。在这两种情况下发送的是同一个文件。您只需在超文本传输协议(Hypertext Transfer Protocol,HTTP)的报头中更改文件的媒体类型标记即可。使用 Apache Web 服务器时,可在服务器配置文件或个人目录中的 .htaccess 文件中做此更改。

Apache 配置指令

根据默认,Apache 通过检查文件的扩展名来决定与每个文件一起发送的媒体类型。扩展名类型映射存储于 httpd/conf 目录(通常是类似 /usr/httpd/conf 或 /etc/httpd/conf 的目录)下的 mime.types 文件中。比如,清单 2 显示了 Apache 2.0 的 mime.types 文件的部分内容。

清单 2. Apache 的 mime.types

                # This file controls what Internet media types are sent to the client for
            # given file extension(s).  Sending the correct media type to the client
            # is important so they know how to handle the content of the file.
            # Extra types can either be added here or by using an AddType directive
            # in your config files. For more information about Internet media types,
            # please read RFC 2045, 2046, 2047, 2048, and 2077.  The Internet media type
            # registry is at <http://www.iana.org/assignments/media-types/>.
            # MIME type                     Extensions
            application/atom+xml            atom
            application/mathematica
            application/mathml+xml          mathml
            application/msword              doc
            application/octet-stream        bin dms lha lzh exe class so dll dmg
            application/postscript          ai eps ps
            application/rdf+xml             rdf
            application/reginfo+xml
            application/xhtml+xml           xhtml xht
            application/xslt+xml            xslt
            application/xml                 xml xsl
            application/xml-dtd             dtd
            application/xml-external-parsed-entity
            application/zip                 zip
            audio/mpeg                      mpga mp2 mp3
            image/jpeg                      jpeg jpg jpe
            image/naplps
            image/png                       png
            image/svg+xml                   svg
            image/tiff                      tiff tif
            text/html                       html htm
            text/plain                      asc txt
            text/sgml                       sgml sgm
            text/xml
            text/xml-external-parsed-entity
            video/mpeg                      mpeg mpg mpe

 

一些更老的版本没有根据默认安装所有这些映射,并且可能事实上使用了一些十分有害的映射。尤其应该注意,对于原始 XML 文件使用 text/xml 而不是 application/xml 是一个常见的问题。

具有了这些默认的映射后,您所需要做的全部工作就是为 XHTML 文件加上 .xhtml 或 .xht 后缀,而不是 .html 后缀,之后,所有这类文件都将被作为 application/xhtml+xml 处理。这对于 Firefox、Opera 和 Safari 效果很好,但对于 Internet Explorer 却并非如此。您所需要的是一种方法,通过它可向 Internet Explorer 发送一种媒体类型而向所有其他浏览器发送另一种媒体类型。

浏览器嗅探

2007 年,可以放心假设所有非 Internet Explorer 的浏览器都能识别 application/xhtml+xml。(如果您确实希望支持很老的浏览器,则破解我下面提出的规则也很容易。)因此您就需要识别 Internet Explorer 的所有版本并将媒体类型更改为 text/html。幸运的是,Internet Explorer 在发送 HTTP 请求时可告知服务器浏览器的类型,如 清单 3 所示。

清单 3. Internet Explorer 的 HTTP 请求报头

                GET /test/a.xhtml HTTP/1.1
            Accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/vnd.ms-excel,
            application/msword, application/vnd.ms-powerpoint, */*
            Accept-Language: en-us
            Accept-Encoding: gzip, deflate
            User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.0; .NET CLR 1.1.4322)
            Host: www.xom.nu
            Connection: Keep-Alive

 

关键在于 User-Agent 字段。虽然由于传统原因 Internet Explorer 在开始时感觉像是 Netscape,但 MSIE 字符串可将其识别为 Internet Explorer。所有版本的 Internet Explorer 在 User-Agent 字段中都包含了此字符串,而所有其他现代的浏览器都不具备此特征。

您需要配置服务器以便查看报头中的 User-Agent 字段,并向 Internet Explorer 发送 text/html,向所有其他浏览器发送 application/xhtml+xml。mod-rewrite 模块并不局限于重写 URL。它还能根据 User-Agent 更改 HTTP 响应报头。清单 4 展示了配置文件中需要放入的代码。

清单 4. 向 Internet Explorer 发送 text.html

                RewriteEngine on
            RewriteCond %{HTTP_USER_AGENT} .*MSIE.*
            RewriteCond %{REQUEST_URI} \.xhtml$
            RewriteRule .* - [T=text/html]

 

第一行会打开重写引擎。

第二行的第一个重写条件表明:下面的规则仅适用于 HTTP 请求报头中的 user-agent 字符串包含子字符串 MSIE 的情形。正则表达式 .*MSIE.* 实现了此功能。

第三行的第二个重写条件表明:下面的规则仅适用于浏览器请求的文件具有 .xhtml 扩展名的情形。常规的 .html 文件被作为普通的 text/html 提供给所有浏览器。

最后一行是实际的重写规则。此规则有点不太常见,因为重写并未真正更改 URL 中的任何东西。接下来,匹配整个 URL (.*) 但接着将其替换成它自己 (-)。然而,最后的 [T=text/html] 字段会把 Content-Type 报头更改为 text/html。如果同时匹配上述两个条件,则使用此规则。反之则不然。

安装规则

根据服务器设置的不同,此代码可能位于以下几个位置之一:

  • 主 httpd.conf 文件
  • httpd.conf 文件中的 VirtualHost 部分或单独的虚拟主机配置文件
  • XHTML 文件所在目录中的 .htaccess 文件

这些指导说明在 Apache 1.3 和 2.0 中应该都有效。如果您在 .htaccess 文件中使用这些规则,则似乎没什么作用,这时,确保对目录做出了如下设置以允许在主 httpd.conf 文件中进行覆盖:

<Directory /var/www/foo>
            AllowOverride FileInfo
            </Directory>

 

Lynx

如果可以的话,您应该多支持一个不能识别 application/xhtml+xml 的浏览器:Lynx。Lynx 是一种文本模式浏览器,主要由自动化脚本和处理 shell 的 UNIX® 爱好者使用。它的市场份额很小,但是其独特的功能使它具有了足够的重要性,颇值得我们关注,但前提是不给其他用户造成不便。

幸运的是,所有 Lynx 的 user-agent 字符串都以词 “Lynx” 开头,而其他的 user-agent 字符串都不包含该词。因此,为支持 Lynx,您所要做的全部工作就是将该字符串添加到重写条件正则表达式中,如 清单 5 所示:

清单 5. 向 Lynx 发送 text.html

                RewriteEngine on
            RewriteCond %{HTTP_USER_AGENT} ((.*MSIE.*)|(Lynx.*))
            RewriteCond %{REQUEST_URI} \.xhtml$
            RewriteRule .* - [T=text/html]

 

第二行中的新正则表达式匹配任何包含 MSIE 或以 Lynx 开头的字符串。如果发现另一个浏览器不能很好地处理 application/xhtml+xml,则可向其 user-agent 字符串中做类似的添加。

伪装成 Internet Explorer

一些更老版本的 Opera 和 Safari 通过在其 user-agent 字符串中包含 MSIE 将自己伪装成 Internet Explorer。但是您无需为此忧虑,原因如下:

  • 几乎已经没什么人使用那些版本了。
  • 与一些更新版本的 Safari 和 Opera 不同,那些较老的版本总的说来处理 text/html 比处理 application/xhtml+xml 的效果好。

要了解更精确的目标信息,请参阅 参考资料 中到 user-agent 字符串完整列表的链接和可供您参考的 application/xhtml+xml 浏览器支持。


结束语

XHTML 是 Web 的未来。但是,像很多其他重要的技术一样,它的采用由于得不到 Microsoft 浏览器的有力支持而受到阻碍。如本文所展示的那样,没有理由等待 Microsoft。您可以轻松地向非 Microsoft 的浏览器提供 XHTML,同时仍然告诉 Internet Explorer 将它作为 tag soup 处理。现代浏览器的访客和页面作者将完全受益于 XHTML,而受 Internet Explorer 牵制的访客则仍然可获得大部分内容。适当地设置 Multipurpose Internet Mail Extensions(MIME)媒体类型并不是向旧的浏览器提供 XHTML 所能采用的惟一途径,但它却是往正确方向上迈出的一大步。

rsync 使用说明

rsync是一款非常优秀的文件同步管理软件,它也支持多种操作系统平台,在Unix环境中,rsync有着卓绝的功绩。希望这篇文档能对一些朋友有所帮助。

1. Install

[url]http://www.samba.org/rsync/[/url]

shell> tar zxvf rsync-x.x.x.tar.gz
shell> cd rsync-x.x.x
shell> ./configure && make && make install

目前大部分 Unix/Linux 默认即安装了 rsync。

2. /etc/rsyncd.conf
 
shell> touch /etc/rsyncd.conf
shell> vi /etc/rsyncd.conf

Edit /etc/rsyncd.conf as below:
 
uid = nobody
gid = nobody
use chroot = no
max connections = 5
pid file = /var/run/rsync.pid
lock file = /var/run/rsync.lock
log file = /var/log/rsync.log
 
[<dst-dir>]
path = <dst-dir-fullpath>
ignore errors
read only = no
list = yes
auth users = username
secrets file = /etc/rsyncd.secrets

3. /etc/rsync.secrets
 
shell> echo “jack:password” >> /etc/rsyncd.secrets
shell> chmod 600 /etc/rsyncd.secrets
 
*注:一定要把rsyncd.secrets的权限设为600,否则不能正常进行身份认证。
     包括–password-file指向的密码文件,也必须设成600权限。

    
    
4. Autorun

>>> Idea 1 – only linux <<<

shell> vi /etc/xinetd.d/rsync
 
set DISABLE to yes, the result looks like this:
service rsync
{
        disable         = no  <———- change to yes
        socket_type     = stream
        wait            = no
        user            = root
        server          = /usr/bin/rsync
        server_args     = –daemon
        log_on_failure  += USERID
}

>>> Idea 2 – on openSUSE <<<

shell> echo “/usr/local/bin/rsync –daemon” >> /etc/init.d/boot.local

>>> Idea 3 <<<
 
shell> echo “/usr/local/bin/rsync –daemon” >> /etc/rc.local

5. Run daemon

shell> rsync –daemon
 
*注一:在rsync的man手册的 CONNECTING TO AN RSYNC SERVER 处有提到:
It is also possible to use rsync without a remote shell as the transport. In this case you will connect to a remote  rsync server running on TCP port 873.
 
*注二:在rsync的man手册的 –port=PORT 选项解释中有提到:
This specifies an alternate TCP port number to use rather than the default port 873.

6. rsync 命令实例

6.1 显示目录内容

命令
——
a) rsync <dst-dir>
b) rsync -r <dst-dir>
c) rsync jack@192.168.0.1::<dst-dir>
d) rsync ssh_user@192.168.0.1:<dst-dir>

命令说明
———
a) 显示<dst-dir>目录内容(第一层)
b) 递归显示<dst-dir>目录
内容
c) 显示远程主机<dst-dir>目录内容
   *注1:端口模式, 基于rsync用户的身份验证
   *注2:rsync server上的目录必须具有xx7的权限.
d) 查看
远程主机<dst-dir>目录内容
   *注1:remote shell模式, 通过ssh连接的基于系统本地用户的身份验证
   *注2:这里只使用了一个冒号(:),同时用户名是远
程主机

的ssh用户,密码也是ssh用户对应的密码。
   *注3:使用”<dst-dir>”,则列出<dst-dir>文件夹本身的信息。若要列出<dst-dir>文件夹内容,应使用”<dst-dir>/”。
  
参数说明
———
-r          对目录进行递归操作

  
6.2 本地目录之间同步

命令
——
a) rsync -av  –progress <src-dir>/ <dst-dir>             *** 注意(/) ***
b) rsync -av  –progress <src-dir>  <dst-dir>
c) rsync -avu –progress –delete <src-dir>/  <dst-dir>
d) rsync -av  –progress –temp-dir=/tmp <src-dir>/ <dst-dir>

命令说明
———
a) 同步src-dir目录下所有文件到dst-dir目录下
b) 同步src-dir目录下所有文件到dst-dir/src-dir目录下
c) 对src-dir目录内容向dst-dir目录下进行差异更新,有增加/更新则添加替换,有减少则对其删减
d) 比a)多了
–temp-dir=/tmp,即指定/tmp为临时交换区,这样可以避免因目标目录空间不够引起的无法同步文件的错误。


参数说明
———
-a          相当于 -rlptgoD 的集合
-u          等同于 –update,在目标文件比源文件新的情况下不更新
-v          显示同步的文件
–progress  显示文件同步时的百分比进度、传输速率
–delete    删除目标目录中多于源目录的文件

6.3 异地主机之间同步

命令
——
a) rsync -avz  –progress <src-dir> jack@192.168.0.1::<dst-dir>/
b) rsync -avz  –progress <src-dir> jack@192.168.0.1::<dst-dir>/ –password-file=/home/jack/rsync.jack

c) rsync -avuz  –progress –delete <src-dir> jack@192.168.0.1::<dst-dir>/ –password-file=/home/jack/rsync.jack


d) rsync -avz  –progress jack@192.168.0.1::<dst-dir>/<src-dir> <dst-dir>

命令说明
———
a) 同步本地<src-dir>目录的内容到远程主机192.168.0.1的<dst-dir>目录下,jack是rsync数据库用户(参见3. /etc/rsync.secrets)
b) 通过自动读取用户密码而实现非交互登录文件同步
c) 较b)多了-u和–delete

d) 同步远程主机内容到本地目录

参数说明
———
-z                等同于 –compress,对传输的文件压缩,这对节约网络带宽或在网络资源紧张的情况下非常有用
–password-file   引用192.168.0.1上rsync用户jack口令的本地文件,创建方法如下
                  shell> echo “jackpwd” >> /home/jack/rsync.jack
                  shell> chown jack:wheel
/home/jack/rsync.jack
                  shell> chmod 600 /home/jack/rsync.jack

===============================================
2006/07/18  Created  by wandering
2008/10/12  重新对文档整理,修正了部分错误,增加一些内容
2008/10/22  增加了 –temp-dir 参数

PHP session有效期session.gc_maxlifetime

 PHP中的session有效期默认是1440秒(24分钟)【weiweiok 注:php5里默认的是180分】,也就是说,客户端超过24分钟没有刷新,当前session就会失效。很明显,这是不能满足需要的。
    一个已知管用的方法是,使用session_set_save_handler,接管所有的session管理工作,一般是把session信息存储到数据库,这样可以通过SQL语句来删除所有过期的session,精确地控制session的有效期。这也是基于PHP的大型网站常用的方法。但是,一般的小型网站,似乎没有必要这么劳师动众。
     但是一般的Session的生命期有限,如果用户关闭了浏览器,就不能保存Session的变量了!那么怎么样可以实现Session的永久生命期呢?
    大家知道,Session储存在服务器端,根据客户端提供的SessionID来得到这个用户的文件,然后读取文件,取得变量的值,SessionID可以使用客户端的Cookie或者Http1.1协议的Query_String(就是访问的URL的“?”后面的部分)来传送给服务器,然后服务器读取Session的目录……
    要实现Session的永久生命期,首先需要了解一下php.ini关于Session的相关设置(打开php.ini文件,在“[Session]”部分):
    1、session.use_cookies:默认的值是“1”,代表SessionID使用Cookie来传递,反之就是使用Query_String来传递;
    2、session.name:这个就是SessionID储存的变量名称,可能是Cookie,也可能是Query_String来传递,默认值是“PHPSESSID”;
    3、session.cookie_lifetime:这个代表SessionID在客户端Cookie储存的时间,默认是0,代表浏览器一关闭SessionID就作废……就是因为这个所以Session不能永久使用!
    4、session.gc_maxlifetime:这个是Session数据在服务器端储存的时间,如果超过这个时间,那么Session数据就自动删除!
    还有很多的设置,不过和本文相关的就是这些了,下面开始讲使用永久Session的原理和步骤。
    前面说过,服务器通过SessionID来读取Session的数据,但是一般浏览器传送的SessionID在浏览器关闭后就没有了,那么我们只需要人为的设置SessionID并且保存下来,不就可以……
    如果你拥有服务器的操作权限,那么设置这个非常非常的简单,只是需要进行如下的步骤:
    1、把“session.use_cookies”设置为1,打开Cookie储存SessionID,不过默认就是1,一般不用修改;
    2、把“session.cookie_lifetime”改为正无穷(当然没有正无穷的参数,不过999999999和正无穷也没有什么区别);
    3、把“session.gc_maxlifetime”设置为和“session.cookie_lifetime”一样的时间;
    在PHP的文档中明确指出,设定session有效期的参数是session.gc_maxlifetime。可以在php.ini文件中,或者通过ini_set()函数来修改这一参数。问题在于,经过多次测试,修改这个参数基本不起作用,session有效期仍然保持24分钟的默认值。
    由于PHP的工作机制,它并没有一个daemon线程,来定时地扫描session信息并判断其是否失效。当一个有效请求发生时,PHP会根据全局变量session.gc_probability/session.gc_divisor(同样可以通过php.ini或者ini_set()函数来修改)的值,来决定是否启动一个GC(Garbage Collector)。
   默认情况下,session.gc_probability = 1,session.gc_divisor =100,也就是说有1%的可能性会启动GC。GC的工作,就是扫描所有的session信息,用当前时间减去session的最后修改时间(modified date),同session.gc_maxlifetime参数进行比较,如果生存时间已经超过gc_maxlifetime,就把该session删除。
    到此为止,工作一切正常。那为什么会发生gc_maxlifetime无效的情况呢?
    在默认情况下,session信息会以文本文件的形式,被保存在系统的临时文件目录中。在Linux下,这一路径通常为\tmp,在 Windows下通常为C:\Windows\Temp。当服务器上有多个PHP应用时,它们会把自己的session文件都保存在同一个目录中。同样地,这些PHP应用也会按一定机率启动GC,扫描所有的session文件。
    问题在于,GC在工作时,并不会区分不同站点的session。举例言之,站点A的gc_maxlifetime设置为2小时,站点B的 gc_maxlifetime设置为默认的24分钟。当站点B的GC启动时,它会扫描公用的临时文件目录,把所有超过24分钟的session文件全部删除掉,而不管它们来自于站点A或B。这样,站点A的gc_maxlifetime设置就形同虚设了。
    找到问题所在,解决起来就很简单了。修改session.save_path参数,或者使用session_save_path()函数,把保存session的目录指向一个专用的目录,gc_maxlifetime参数工作正常了。
    严格地来说,这算是PHP的一个bug?
    还有一个问题就是,gc_maxlifetime只能保证session生存的最短时间,并不能够保存在超过这一时间之后session信息立即会得到删除。因为GC是按机率启动的,可能在某一个长时间内都没有被启动,那么大量的session在超过gc_maxlifetime以后仍然会有效。
    解决这个问题的一个方法是,把session.gc_probability/session.gc_divisor的机率提高,如果提到100%,就会彻底解决这个问题,但显然会对性能造成严重的影响。另一个方法是自己在代码中判断当前session的生存时间,如果超出了 gc_maxlifetime,就清空当前session。
    但是如果你没有服务器的操作权限,那就比较麻烦了,你需要通过PHP程序改写SessionID来实现永久的Session数据保存。查查php.net的函数手册,可以见到有“session_id”这个函数:如果没有设置参数,那么将返回当前的SessionID,如果设置了参数,就会将当前的SessionID设置为给出的值……
    只要利用永久性的Cookie加上“session_id”函数,就可以实现永久Session数据保存了!
    但是为了方便,我们需要知道服务器设置的“session.name”,但是一般用户都没有权限查看服务器的php.ini设置,不过PHP提供了一个非常好的函数“phpinfo”,利用这个可以查看几乎所有的PHP信息!
————————————————————————————
<title>PHP相关信息显示</title>
<?phpinfo()?>
————————————————————————————
打开编辑器,输入上面的代码,然后在浏览器中运行这个程序,会见到PHP的相关信息(如图1所示)。其中有一项“session.name”的参数,这个就是我们需要的服务器“session.name”,一般是“PHPSESSID”。
记下了SessionID的名称后,我们就可以实现永久的Session数据储存了!
复制代码 代码如下:
session_start();
ini_set(’session.save_path’,'/tmp/’);
//6个钟头
ini_set(’session.gc_maxlifetime’,21600);
//保存一天
$lifeTime = 24 * 3600;
setcookie(session_name(), session_id(), time() + $lifeTime, “/”);

后记:
其实真正的永久储存是不可能的,因为Cookie的保存时间有限,而服务器的空间也有限……但是对于一些需要保存时间比较长的站点,以上方法就已经足够了!
把session放入mysql的Example:
数据库里建表:session ( sesskey varchar32 , expiry int11 , value longtext)
code:
代码执行前已经连接数据库了。
复制代码 代码如下:
define(‘STORE_SESSIONS’,'mysql’);
if (STORE_SESSIONS == ‘mysql’) {
if (!$SESS_LIFE = get_cfg_var(’session.gc_maxlifetime’)) {
$SESS_LIFE = 1440;
}
function _sess_open($save_path, $session_name) {
// 如果没有连接数据库,可以在此执行mysql_pconnect,mysql_select_db
return true;
}
function _sess_close() {
return true;
}
function _sess_read($key) {
$value_query = mysql_query(“select value from sessions where sesskey = ‘” .addslashes($key) . “‘ and expiry > ‘” . time() . “‘”);
$value = mysql_fetch_array($value_query);
if (isset($value['value'])) {
return $value['value'];
}
return false;
}
function _sess_write($key, $val) {
global $SESS_LIFE;
$expiry = time() + $SESS_LIFE;
$value = $val;
$check_query = mysql_query(“select count(*) as total from sessions where sesskey = ‘” . addslashes($key) . “‘”);
$check = mysql_fetch_array($check_query);
if ($check['total'] > 0) {
return mysql_query(“update sessions set expiry = ‘” . addslashes($expiry) . “‘, value = ‘” . addslashes($value) . “‘ where sesskey = ‘” . addslashes($key) . “‘”);
} else {
return mysql_query(“insert into sessions values (‘” . addslashes($key) . “‘, ‘” . addslashes($expiry) . “‘, ‘” . addslashes($value) . “‘)”);
}
}
function _sess_destroy($key) {
return mysql_query(“delete from sessions where sesskey = ‘” . addslashes($key) . “‘”);
}
function _sess_gc($maxlifetime) {
mysql_query(“delete from sessions where expiry < ‘” . time() . “‘”);
return true;
}
session_set_save_handler(‘_sess_open’, ‘_sess_close’, ‘_sess_read’, ‘_sess_write’, ‘_sess_destroy’, ‘_sess_gc’);
}
danoo_session_name( ‘dtvSid’ );
danoo_session_save_path(SESSION_WRITE_DIRECTORY);

还是有点不明白,open,write那些参数哪里来的。
修改php.ini配置的两个常用函数:
get_cfg_var(’session.gc_maxlifetime’) : 取得session.gc_maxlifetime的值
ini_set(’session.cookie_lifetime’,'0′) : 设置session.cookie_lifetime的值为0。

21个实用便利的PHP代码

21个实用便利的PHP代码。

1. PHP可阅读随机字符串

此代码将创建一个可阅读的字符串,使其更接近词典中的单词,实用且具有密码验证功能。

/**************
*@length - length of random string (must be a multiple of 2)
**************/
function readable_random_string($length = 6){
    $conso=array("b","c","d","f","g","h","j","k","l",
    "m","n","p","r","s","t","v","w","x","y","z");
    $vocal=array("a","e","i","o","u");
    $password="";
    srand ((double)microtime()*1000000);
    $max = $length/2;
    for($i=1; $i<=$max; $i++)
    {
    $password.=$conso[rand(0,19)];
    $password.=$vocal[rand(0,4)];
    }
    return $password;
}

2. PHP生成一个随机字符串

如果不需要可阅读的字符串,使用此函数替代,即可创建一个随机字符串,作为用户的随机密码等。

/*************
*@l - length of random string
*/
function generate_rand($l){
  $c= "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
  srand((double)microtime()*1000000);
  for($i=0; $i<$l; $i++) {
      $rand.= $c[rand()%strlen($c)];
  }
  return $rand;
 }

3. PHP编码电子邮件地址

使用此代码,可以将任何电子邮件地址编码为html 字符实体,以防止被垃圾邮件程序收集。

function encode_email($email='info@domain.com', $linkText='Contact Us', $attrs ='class="emailencoder"' )
{
    // remplazar aroba y puntos
    $email = str_replace('@', '@', $email);
    $email = str_replace('.', '.', $email);
    $email = str_split($email, 5);  

    $linkText = str_replace('@', '@', $linkText);
    $linkText = str_replace('.', '.', $linkText);
    $linkText = str_split($linkText, 5);  

    $part1 = '<a href="ma';
    $part2 = 'ilto:';
    $part3 = '" '. $attrs .' >';
    $part4 = '</a>';  

    $encoded = '<script type="text/javascript">';
    $encoded .= "document.write('$part1');";
    $encoded .= "document.write('$part2');";
    foreach($email as $e)
    {
            $encoded .= "document.write('$e');";
    }
    $encoded .= "document.write('$part3');";
    foreach($linkText as $l)
    {
            $encoded .= "document.write('$l');";
    }
    $encoded .= "document.write('$part4');";
    $encoded .= '</script>';  

    return $encoded;
}

4. PHP验证邮件地址

电子邮件验证也许是中最常用的网页表单验证,此代码除了验证电子邮件地址,也可以选择检查邮件域所属 DNS 中的 MX 记录,使邮件验证功能更加强大。

function is_valid_email($email, $test_mx = false)
{
    if(eregi("^([_a-z0-9-]+)(\.[_a-z0-9-]+)*@([a-z0-9-]+)(\.[a-z0-9-]+)*(\.[a-z]{2,4})$", $email))
        if($test_mx)
        {
            list($username, $domain) = split("@", $email);
            return getmxrr($domain, $mxrecords);
        }
        else
            return true;
    else
        return false;
}

5. PHP列出目录内容

function list_files($dir)
{
    if(is_dir($dir))
    {
        if($handle = opendir($dir))
        {
            while(($file = readdir($handle)) !== false)
            {
                if($file != "." && $file != ".." && $file != "Thumbs.db")
                {
                    echo '<a target="_blank" href="'.$dir.$file.'">'.$file.'</a><br>'."\n";
                }
            }
            closedir($handle);
        }
    }
}

6. PHP销毁目录

删除一个目录,包括它的内容。

/*****
*@dir - Directory to destroy
*@virtual[optional]- whether a virtual directory
*/
function destroyDir($dir, $virtual = false)
{
    $ds = DIRECTORY_SEPARATOR;
    $dir = $virtual ? realpath($dir) : $dir;
    $dir = substr($dir, -1) == $ds ? substr($dir, 0, -1) : $dir;
    if (is_dir($dir) && $handle = opendir($dir))
    {
        while ($file = readdir($handle))
        {
            if ($file == '.' || $file == '..')
            {
                continue;
            }
            elseif (is_dir($dir.$ds.$file))
            {
                destroyDir($dir.$ds.$file);
            }
            else
            {
                unlink($dir.$ds.$file);
            }
        }
        closedir($handle);
        rmdir($dir);
        return true;
    }
    else
    {
        return false;
    }
}

7. PHP解析 JSON 数据

与大多数流行的 Web 服务如twitter 通过开放 API 来提供数据一样,它总是能够知道如何解析 API 数据的各种传送格式,包括 JSON,XML 等等。

$json_string='{"id":1,"name":"foo","email":"foo@foobar.com","interest":["wordpress","php"]} ';
$obj=json_decode($json_string);
echo $obj->name; //prints foo
echo $obj->interest[1]; //prints php

8. PHP解析 XML 数据

//xml string
$xml_string="<?xml version='1.0'?>
<users>
   <user id='398'>
      <name>Foo</name>
      <email>foo@bar.com</name>
   </user>
   <user id='867'>
      <name>Foobar</name>
      <email>foobar@foo.com</name>
   </user>
</users>";  

//load the xml string using simplexml
$xml = simplexml_load_string($xml_string);  

//loop through the each node of user
foreach ($xml->user as $user)
{
   //access attribute
   echo $user['id'], '  ';
   //subnodes are accessed by -> operator
   echo $user->name, '  ';
   echo $user->email, '<br />';
}

9. PHP创建日志缩略名

创建用户友好的日志缩略名。

function create_slug($string){
    $slug=preg_replace('/[^A-Za-z0-9-]+/', '-', $string);
    return $slug;
}

10. PHP获取客户端真实 IP 地址

该函数将获取用户的真实 IP 地址,即便他使用代理服务器。

function getRealIpAddr()
{
    if (!emptyempty($_SERVER['HTTP_CLIENT_IP']))
    {
        $ip=$_SERVER['HTTP_CLIENT_IP'];
    }
    elseif (!emptyempty($_SERVER['HTTP_X_FORWARDED_FOR']))
    //to check ip is pass from proxy
    {
        $ip=$_SERVER['HTTP_X_FORWARDED_FOR'];
    }
    else
    {
        $ip=$_SERVER['REMOTE_ADDR'];
    }
    return $ip;
}

11. PHP强制性文件下载

为用户提供强制性的文件下载功能。

/********************
*@file - path to file
*/
function force_download($file)
{
    if ((isset($file))&&(file_exists($file))) {
       header("Content-length: ".filesize($file));
       header('Content-Type: application/octet-stream');
       header('Content-Disposition: attachment; filename="' . $file . '"');
       readfile("$file");
    } else {
       echo "No file selected";
    }
}

12. PHP创建标签云

function getCloud( $data = array(), $minFontSize = 12, $maxFontSize = 30 )
{
    $minimumCount = min( array_values( $data ) );
    $maximumCount = max( array_values( $data ) );
    $spread       = $maximumCount - $minimumCount;
    $cloudHTML    = '';
    $cloudTags    = array();  

    $spread == 0 && $spread = 1;  

    foreach( $data as $tag => $count )
    {
        $size = $minFontSize + ( $count - $minimumCount )
            * ( $maxFontSize - $minFontSize ) / $spread;
        $cloudTags[] = '<a style="font-size: ' . floor( $size ) . 'px'
        . '" href="#" title="\'' . $tag  .
        '\' returned a count of ' . $count . '">'
        . htmlspecialchars( stripslashes( $tag ) ) . '</a>';
    }  

    return join( "\n", $cloudTags ) . "\n";
}
/**************************
****   Sample usage    ***/
$arr = Array('Actionscript' => 35, 'Adobe' => 22, 'Array' => 44, 'Background' => 43,
    'Blur' => 18, 'Canvas' => 33, 'Class' => 15, 'Color Palette' => 11, 'Crop' => 42,
    'Delimiter' => 13, 'Depth' => 34, 'Design' => 8, 'Encode' => 12, 'Encryption' => 30,
    'Extract' => 28, 'Filters' => 42);
echo getCloud($arr, 12, 36);

13. PHP寻找两个字符串的相似性

PHP 提供了一个极少使用的 similar_text 函数,但此函数非常有用,用于比较两个字符串并返回相似程度的百分比。

similar_text($string1, $string2, $percent);
//$percent will have the percentage of similarity

14. PHP在应用程序中使用 Gravatar 通用头像

随着 WordPress 越来越普及,Gravatar 也随之流行。由于 Gravatar 提供了易于使用的 API,将其纳入应用程序也变得十分方便。

/******************
*@email - Email address to show gravatar for
*@size - size of gravatar
*@default - URL of default gravatar to use
*@rating - rating of Gravatar(G, PG, R, X)
*/
function show_gravatar($email, $size, $default, $rating)
{
    echo '<img src="http://www.gravatar.com/avatar.php?gravatar_id='.md5($email).
        '&default='.$default.'&size='.$size.'&rating='.$rating.'" width="'.$size.'px"
        height="'.$size.'px" />';
}

15. PHP在字符断点处截断文字

所谓断字 (word break),即一个单词可在转行时断开的地方。这一函数将在断字处截断字符串。

// Original PHP code by Chirp Internet: www.chirp.com.au
// Please acknowledge use of this code by including this header.
function myTruncate($string, $limit, $break=".", $pad="...") {
    // return with no change if string is shorter than $limit
    if(strlen($string) <= $limit)
        return $string;   

    // is $break present between $limit and the end of the string?
    if(false !== ($breakpoint = strpos($string, $break, $limit))) {
        if($breakpoint < strlen($string) - 1) {
            $string = substr($string, 0, $breakpoint) . $pad;
        }
    }
    return $string;
}
/***** Example ****/
$short_string=myTruncate($long_string, 100, ' ');

16. PHP文件 Zip 压缩

/* creates a compressed zip file */
function create_zip($files = array(),$destination = '',$overwrite = false) {
    //if the zip file already exists and overwrite is false, return false
    if(file_exists($destination) && !$overwrite) { return false; }
    //vars
    $valid_files = array();
    //if files were passed in...
    if(is_array($files)) {
        //cycle through each file
        foreach($files as $file) {
            //make sure the file exists
            if(file_exists($file)) {
                $valid_files[] = $file;
            }
        }
    }
    //if we have good files...
    if(count($valid_files)) {
        //create the archive
        $zip = new ZipArchive();
        if($zip->open($destination,$overwrite ? ZIPARCHIVE::OVERWRITE : ZIPARCHIVE::CREATE) !== true) {
            return false;
        }
        //add the files
        foreach($valid_files as $file) {
            $zip->addFile($file,$file);
        }
        //debug
        //echo 'The zip archive contains ',$zip->numFiles,' files with a status of ',$zip->status;  

        //close the zip -- done!
        $zip->close();  

        //check to make sure the file exists
        return file_exists($destination);
    }
    else
    {
        return false;
    }
}
/***** Example Usage ***/
$files=array('file1.jpg', 'file2.jpg', 'file3.gif');
create_zip($files, 'myzipfile.zip', true);

17. PHP解压缩 Zip 文件

/**********************
*@file - path to zip file
*@destination - destination directory for unzipped files
*/
function unzip_file($file, $destination){
    // create object
    $zip = new ZipArchive() ;
    // open archive
    if ($zip->open($file) !== TRUE) {
        die (’Could not open archive’);
    }
    // extract contents to destination directory
    $zip->extractTo($destination);
    // close archive
    $zip->close();
    echo 'Archive extracted to directory';
}

18. PHP为 URL 地址预设 http 字符串

有时需要接受一些表单中的网址输入,但用户很少添加 http:// 字段,此代码将为网址添加该字段。

if (!preg_match("/^(http|ftp):/", $_POST['url'])) {
   $_POST['url'] = 'http://'.$_POST['url'];
}

19. PHP将网址字符串转换成超级链接

该函数将 URL 和 E-mail 地址字符串转换为可点击的超级链接。

function makeClickableLinks($text) {
 $text = eregi_replace('(((f|ht){1}tp://)[-a-zA-Z0-9@:%_+.~#?&//=]+)',
 '<a href="\1">\1</a>', $text);
 $text = eregi_replace('([[:space:]()[{}])(www.[-a-zA-Z0-9@:%_+.~#?&//=]+)',
 '\1<a href="http://\2">\2</a>', $text);
 $text = eregi_replace('([_.0-9a-z-]+@([0-9a-z][0-9a-z-]+.)+[a-z]{2,3})',
 '<a href="mailto:\1">\1</a>', $text);  

return $text;
}

20. PHP调整图像尺寸

创建图像缩略图需要许多时间,此代码将有助于了解缩略图的逻辑。

/**********************
*@filename - path to the image
*@tmpname - temporary path to thumbnail
*@xmax - max width
*@ymax - max height
*/
function resize_image($filename, $tmpname, $xmax, $ymax)
{
    $ext = explode(".", $filename);
    $ext = $ext[count($ext)-1];  

    if($ext == "jpg" || $ext == "jpeg")
        $im = imagecreatefromjpeg($tmpname);
    elseif($ext == "png")
        $im = imagecreatefrompng($tmpname);
    elseif($ext == "gif")
        $im = imagecreatefromgif($tmpname);  

    $x = imagesx($im);
    $y = imagesy($im);  

    if($x <= $xmax && $y <= $ymax)
        return $im;  

    if($x >= $y) {
        $newx = $xmax;
        $newy = $newx * $y / $x;
    }
    else {
        $newy = $ymax;
        $newx = $x / $y * $newy;
    }  

    $im2 = imagecreatetruecolor($newx, $newy);
    imagecopyresized($im2, $im, 0, 0, 0, 0, floor($newx), floor($newy), $x, $y);
    return $im2;
}

21. PHP检测ajax 请求

大多数的 JavaScript 框架如jquery,Mootools 等,在发出 Ajax 请求时,都会发送额外的 HTTP_X_REQUESTED_WITH 头部信息,头当他们一个ajax请求,因此你可以在服务器端侦测到 Ajax 请求。

if(!emptyempty($_SERVER['HTTP_X_REQUESTED_WITH']) && strtolower($_SERVER['HTTP_X_REQUESTED_WITH']) == 'xmlhttprequest'){
    //If AJAX Request Then
}else{
//something else
}

PHP 判断/移除 BOM(UTF-8)

BOM: 萬國碼檔案簽名 BOM (Byte order Mark, U+FEFF)

BOM 的內容可以表示 UNICODE 是哪種編碼, 但是在接收到的檔案, 要拆解後寫入 DB, 看到 BOM 就覺得有點 ooxx.

在 utf8_encode 看到兩段程式可以來測試 寫入/移除 BOM.

將寫入的檔案內容前加 BOM
<?php
function writeUTF8File($filename,$content)
{
    $f = fopen($filename, ‘w’);
    fwrite($f, pack(“CCC”, 0xef,0xbb,0xbf));
    fwrite($f,$content);
    fclose($f);
}
?>

移除 BOM function
<?php
function removeBOM($str = ”)
{
   if (substr($str, 0,3) == pack(“CCC”,0xef,0xbb,0xbf)) {
       $str = substr($str, 3);
   }
   return $str;
}
?>

由此上述 BOM = pack(“CCC”,0xef,0xbb,0xbf), 所以移除 BOM 的寫法可用上面的 removeBOM function 或 下述其一:
str_replace(“\xef\xbb\xbf”, ”, $bom_content);
preg_replace(“/^\xef\xbb\xbf/”, ”, $bom_content);

另外看到 判斷此字串是不是 UTF-8 的 function:
function isUTF8($string)
{
    return (utf8_encode(utf8_decode($string)) == $string);
}