Linux 中 RPM 命令参数使用详解Linux(精选9篇)
在Linux操作系统中,有一个系统软件包,它的功能类似于Windows里面的“添加/删除程序”,但是功能又比“添加/删除程序”强很多,它就是Red Hat Package Manager(简称RPM)。此工具包最先是由Red Hat公司推出的,后来被其他Linux开发商所借用。由于它为Linux使用者省去了很多时间,所以被广泛应用于在Linux下安装、删除软件。下面就给大家介绍一下它的具体使用方法。
1.我们得到一个新软件,在安装之前,一般都要先查看一下这个软件包里有什么内容,假设这个文件是:Linux-1.4-6.i368.rpm,我们可以用这条命令查看:
rpm -qpi Linux-1.4-6.i368.rpm
系统将会列出这个软件包的详细资料,包括含有多少个文件、各文件名称、文件大小、创建时间、编译日期等信息。
2.上面列出的所有文件在安装时不一定全部安装,就像Windows下程序的安装方式分为典型、完全、自定义一样,Linux也会让你选择安装方式,此时我们可以用下面这条命令查看软件包将会在系统里安装哪些部分,以方便我们的选择:
rpm -qpl Linux-1.4-6.i368.rpm
3.选择安装方式后,开始安装。我们可以用rpm-ivh Linux-1.4-6.i368.rpm命令安装此软件。在安装过程中,若系统提示此软件已安装过或因其他原因无法继续安装,但若我们确实想执行安装命令,可以在 -ivh后加一参数“-replacepkgs”:
rpm -ivh -replacepkgs Linux-1.4-6.i368.rpm
4.有时我们卸载某个安装过的软件,只需执行rpm-e <文件名>命令即可,
5.对低版本软件进行升级是提高其功能的好办法,这样可以省去我们卸载后再安装新软件的麻烦,要升级某个软件,只须执行如下命令:rpm -uvh <文件名>,注意:此时的文件名必须是要升级软件的升级补丁?
6.另外一个安装软件的方法可谓是Linux的独到之处,同时也是RMP强大功能的一个表现:通过FTP站点直接在线安装软件。当找到含有你所需软件的站点并与此网站连接后,执行下面的命令即可实现在线安装,譬如在线安装Linux-1.4-6.i368.rpm,可以用命令:
rpm -i ftp://ftp.pht.com/pub/linux/redhat/...-1.4-6.i368.rpm
7.在我们使用电脑过程中,难免会有误操作,若我们误删了几个文件而影响了系统的性能时,怎样查找到底少了哪些文件呢?RPM软件包提供了一个查找损坏文件的功能,执行此命令:rpm -Va即可,Linux将为你列出所有损坏的文件。你可以通过Linux的安装光盘进行修复。
8.Linux系统中文件繁多,在使用过程中,难免会碰到我们不认识的文件,在Windows下我们可以用“开始/查找”菜单快速判断某个文件属于哪个文件夹,在Linux中,下面这条命令行可以帮助我们快速判定某个文件属于哪个软件包:
rpm -qf <文件名>
9.当每个软件包安装在Linux系统后,安装文件都会到RPM数据库中“报到”,所以,我们要查询某个已安装软件的属性时,只需到此数据库中查找即可。注意:此时的查询命令不同于1和8介绍的查询,这种方法只适用于已安装过的软件包!命令格式:
rpm -参数 <文件名>
语法:wc [选项] 文件…
说明:该命令统计给定文件中的字节数、字数、行数。如果没有给出文件名,则从标准输入读取。wc同时也给出所有指定文件的总统计数。字是由空格字符区分开的最大字符串。
该命令各选项含义如下:
- c 统计字节数。
- l 统计行数。
- w 统计字数。
这些选项可以组合使用。
输出列的顺序和数目不受选项的顺序和数目的影响。总是按下述顺序显示并且每项最多一列。
行数、字数、字节数、文件名
如果命令行中没有文件名,则输出中不出现文件名。
例如:
$ wc - lcw file1 file2
4 33 file1
7 52 file2
11 11 85 total
省略任选项-lcw,wc命令的执行结果与上面一样
示例:
wc命令用来计算一个文件或者指定的多个文件中的行数,单词数和字符数,
如:
wc filename
第一列显示行数,第二列显示单词数,第三列显示字符数。
wc 有四个参数可选,分别是l,c,m,w
wc -l filename 报告行数
wc -c filename 报告字节数
wc -m filename 报告字符数
wc -w filename 报告单词数
指令模式和输入模式。在指令模式下输入的按键将做为指令来处理:如输入 a,vi即认为是在当前位置插入字符。而在输入模式下,vi则把输入的按键当作插入的字符来处理。指令模式切换到输入模式只需键入相应的输入命令即可(如a,A),而要从输入模式切换到指令模式,则需在输入模式下键入ESC键,如果不晓得现在是处於什麽模式,可以多按几次 [ESC],系统如发出哔哔声就表示已处于指令模式下了。
付:在指令模式进入输入模式的指令:
新增 (append)
a :从光标所在位置後面开始新增资料,光标後的资料随新增资料向後移动。
A:从光标所在列最後面的地方开始新增资料。
插入 (insert)
i:从光标所在位置前面开始插入资料,光标後的资料随新增资料向後移动。
I :从光标所在列的第一个非空白字元前面开始插入资料。
开始 (open) o :在光标所在列下新增一列并进入输入模式。
O: 在光标所在列上方新增一列并进入输入模式。
一、vi打开编辑文件
vi
示例:vi test.conf,vi /tmp/test.conf,...
如果指定的filePathAndName存在则会自动载入该文件内容,如果不存在则会自动创建。
注:每行前显示波浪符号的表示该行为空行。
二、vi进入输入模式
vi工作于两种模式:指令模式和输入模式。指令模式是打开文件后的默认模式模式,此时输入任何字符均被视为控制指令。输入模式就是普通的编辑模式,此时输入字符默认为输入内容。
三、vi编辑指令
1、屏幕翻页
Ctrl+u: 向上翻半屏
Ctrl+f: 向上翻一屏
Ctrl+d: 向下翻半屏
Ctrl+b: 向下翻一屏
2、移动光标指令
移动光标普遍使用的是方向键,考虑兼容问题,vi定义太多的方向指令,下面只是一小小部分(常用的几个):
space: 光标右移一个字符
Backspace: 光标左移一个字符
Enter: 光标下移一行
nG: 光标移至第n行首
n+: 光标下移n行
n-: 光标上移n行
n$: 光标移至第n行尾
0: 光标移至当前行首
$: 光标移至当前行尾
3、插入删除指令
常用插入、删除指令如下:
i:在当前光标前插入,光标后文本向后移
a:从当前光标后插入,光标后文本后移
I:在光标所在行首插入(第一个非空白字符前)
A:从光标所在行末插入
o: 在光标所在行下面新增一行(并进入输入模式)
O: 在光标所在行上方新增一行(并进入输入模式)
x: 删除光标所在字符,等同于[Delete]功能键
X: 删除光标前字符,相当与[Backspace]
dd: 删除光标所在的行
r: 修改光标所在字符
R: 替换当前字符及其后的字符,直到按 [ESC]
s: 从当前光标位置处开始,以输入的文本替代指定数目的字符
S: 删除指定数目的行,并以所输入文本代替之
do: 删至行首
d$: 删至行尾
四、vi退出
退出输入模式,先按一下[ESC]键(有时要多按两下),然后执行:
:w ——保存当前文件
:wq —— 存盘退出(与指令 :x 功能相同)
:q —— 直接退出,如已修改会提示是否保存
:q! ——不保存直接退出
附件:vi详细指令表
进入vi的命令
vi filename: 打开或新建文件,并将光标置于第一行首
vi +n filename: 打开文件,并将光标置于第n行首
vi + filename: 打开文件,并将光标置于最后一行首
vi +/pattern filename: 打开文件,并将光标置于第一个与pattern匹配的串处
vi -r filename: 在上次正用vi编辑时发生系统崩溃,恢复filename
vi filename....filename: 打开多个文件,依次进行编辑
移动光标类命令
h: 光标左移一个字符
l: 光标右移一个字符
space: 光标右移一个字符
Backspace: 光标左移一个字符
k或Ctrl+p: 光标上移一行
j或Ctrl+n: 光标下移一行
Enter: 光标下移一行
w或W : 光标右移一个字至字首
b或B : 光标左移一个字至字首
e或E : 光标右移一个字至字尾
): 光标移至句尾
(: 光标移至句首
}: 光标移至段落开头
{: 光标移至段落结尾
nG: 光标移至第n行首
n+: 光标下移n行
n-: 光标上移n行
n$: 光标移至第n行尾
H: 光标移至屏幕顶行
M: 光标移至屏幕中间行
L: 光标移至屏幕最后行
0: 光标移至当前行首
$: 光标移至当前行尾
屏幕翻滚类命令
Ctrl+u: 向文件首翻半屏
Ctrl+d: 向文件尾翻半屏
Ctrl+f: 向文件尾翻一屏
Ctrl+b: 向文件首翻一屏
nz: 将第n行滚至屏幕顶部,不指定n时将当前行滚至屏幕顶部,
插入文本类命令
i: 在光标前
I: 在当前行首
a: 光标后
A: 在当前行尾
o: 在当前行之下新开一行
O: 在当前行之上新开一行
r: 替换当前字符
R: 替换当前字符及其后的字符,直至按ESC键
s: 从当前光标位置处开始,以输入的文本替代指定数目的字符
S: 删除指定数目的行,并以所输入文本代替之
ncw或nCW: 修改指定数目的字
nCC: 修改指定数目的行
删除命令
ndw或ndW:删除光标处开始及其后的n-1个字
do: 删至行首
d$: 删至行尾
ndd: 删除当前行及其后n-1行
x或X: 删除一个字符,x删除光标后的,而X删除光标前的
Ctrl+u: 删除输入方式下所输入的文本
搜索及替换命令
/pattern: 从光标开始处向文件尾搜索pattern
?pattern: 从光标开始处向文件首搜索pattern
n: 在同一方向重复上一次搜索命令
N: 在反方向上重复上一次搜索命令
:s/p1/p2/g: 将当前行中所有p1均用p2替代
:n1,n2s/p1/p2/g: 将第n1至n2行中所有p1均用p2替代
:g/p1/s//p2/g: 将文件中所有p1均用p2替换
选项设置
all:列出所有选项设置情况
term:设置终端类型
ignorance:在搜索中忽略大小写
list:显示制表位(ctrl+i)和行尾标志($)
number:显示行号
report:显示由面向行的命令修改过的数目
terse:显示简短的警告信息
warn:在转到别的文件时若没保存当前文件则显示no write信息
nomagic:允许在搜索模式中,使用前面不带“/”的特殊字符
nowrapscan:禁止vi在搜索到达文件两端时,又从另一端开始
mesg:允许vi显示其他用户用write写到自己终端上的信息
最后行方式命令
:n1,n2 co n3:将n1行到n2行之间的内容拷贝到第n3行下
:n1,n2 m n3:将n1行到n2行之间的内容移至到第n3行下
:n1,n2 d :将n1行到n2行之间的内容删除
:w :保存当前文件
:e filename:打开文件filename进行编辑
:x:保存当前文件并退出
:q:退出vi
:q!:不保存文件并退出vi
:!command:执行shell命令command
:n1,n2 w!command:将文件中n1行至n2行的内容作为command的输入并执行之,若不指定n1,n2,则表示将整个文件内容作为command的输入
:r!command:将命令command的输出结果放到当前行
寄存器操作
“?nyy:将当前行及其下n行的内容保存到寄存器?中,其中?为一个字母,n为一个数字
”?nyw:将当前行及其下n个字保存到寄存器?中,其中?为一个字母,n为一个数字
“?nyl:将当前行及其下n个字符保存到寄存器?中,其中?为一个字母,n为一个数字
”?p:取出寄存器?中的内容并将其放到光标位置处。这里?可以是一个字母,也可以是一个数字
在终端下输入lsof即可显示系统打开的文件,因为 lsof 需要访问核心内存和各种文件,所以必须以 root 用户的身份运行它才能够充分地发挥其功能
lsof -Pnl +M -i4
lsof -Pnl +M -i6
ls -i :80
补充:实用命令
代码如下复制代码
lsof `which httpd` //那个进程在使用apache的可执行文件
lsof /etc/passwd //那个进程在占用/etc/passwd
lsof /dev/hda6 //那个进程在占用hda6
lsof /dev/cdrom //那个进程在占用光驱
lsof -c sendmail //查看sendmail进程的文件使用情况
lsof -c courier -u ^zahn //显示出那些文件被以courier打头的进程打开,但是并不属于用户zahn
lsof -p 30297 //显示那些文件被pid为30297的进程打开
lsof -D /tmp 显示所有在/tmp文件夹中打开的instance和文件的进程。但是symbol文件并不在列
lsof -u1000 //查看uid是100的用户的进程的文件使用情况
lsof -utony //查看用户tony的进程的文件使用情况
lsof -u^tony //查看不是用户tony的进程的文件使用情况(^是取反的意思)
lsof -i //显示所有打开的端口
lsof -i:80 //显示所有打开80端口的进程
lsof -i -U //显示所有打开的端口和UNIX domain文件
lsof -i UDP@[url]www.akadia.com:123 //显示那些进程打开了到www.akadia.com的UDP的123(ntp)端口的链接
lsof -i tcp@ohaha.ks.edu.tw:ftp -r //不断查看目前ftp连接的情况(-r,lsof会永远不断的执行,直到收到中断信号,+r,lsof会一直执行,直到没有档案被显示,缺省是15s刷新)
lsof -i tcp@ohaha.ks.edu.tw:ftp -n //lsof -n 不将IP转换为hostname,缺省是不加上-n参数
lsof命令参数解释
1) -P :这个选项约束着网络文件的端口号到端口名称的转换,
约束转换可以使lsof运行得更快一些。在端口名称的查找不能奏效时,这是很有用的。
2) -n : 这个选项约束着网络文件的端口号到主机名称的转换。约束转换可以使lsof的运行更快一些。在主机名称的查找不能奏效时,它非常有用。
3) -l :这个选项约束着用户ID号到登录名的转换。在登录名的查找不正确或很慢时,这个选项就很有用。
4) +M :此选项支持本地TCP和UDP端口映射程序的注册报告。
5) -i4 :仅列示IPv4协议下的端口。
步骤:
在/etc/rc.d/init.d/下编程脚本。
判断是否正常关机,和生成文件的脚本:touch-file.sh
#!/bin/sh file=/tmp/check if [ -e $ ];then echo “this is not normal shutdown”
>>/var/log/checkmessage else echo “this is normal shutdown and touch file”
>>/var/log/checkmessage touch $ fi
linux下正常关机删除文件的脚本:rm-file.sh
#!/bin/sh file=/tmp/check echo “this is normal shutdown and rm file”
>>/var/log/checkmessage rm -rf $
让脚本自动执行,
ln -s /etc/rc.d/rc0.d/K01rm-file /etc/rc.d/init.d/rm-file.sh
注意:
#ll /etc/rc.d/rc0.d/
软连接两边的名字不哪呢个相同,后一个可以用绝对路径或是相对路径,千万保证两边的名字不一样。
ln -s /etc/rc.d/rc6.d/K01rm-file /etc/rc.d/init.d/rm-file.sh ln -s
/etc/rc.d/rc3.d/S01touch-file /etc/rc.d/init.d/touch-file.sh
我的系统没用到x-window,所以没有level5,也就是不用在/etc/rc.d/rc5.d/下放脚本。
特别注意:
K开头的代表系统关闭的时候执行,S开头的代表开机的时候执行。注意脚本编写的规范,因为有K开通的软链接并不一定会在关机的时候自动去执行,这是为什么呢?执行K脚本的时候会查询/var/lock/subsys/下是否有与K开头脚本同名的空文件名,如果没有就不去执行,所以要按照脚本编写的规范,启动的时在/var/lock/subsys/先touch一个与K01后面同名的空文件:
#/var/lock/subsys #touch fm-file
1. 安装svn
如果是ubuntu系统,直接 sudo agt-get install subversion
2. 创建svn库
svnadmin create /home/robin/svn
3. 配置用户、权限
a. cd /home/robin/svn/conf
b. vi svnserve.conf , 取消passwd, authz 两行的注释
c. vi passwd , 添加用户
[users]
zhanglb = 123
d. vi authz , 配置权限
i) [groups]下面添加组和用户
admin = zhanglb
ii) 添加目录
[/]
@admin = rw
4. 启动svn服务
svnserve -d -r /home/robin/svn
5. 在其它目录checkout svn
假设svn库所在服务器的ip是192.168.128.128,则svn库地址为
直接checkout地址为: svn://192.168.128.128
用户名 zhanglb
密码 123
其它:
关闭svn服务
1.最常用的方式,就是在make config里修改“General Setup”子菜单中的“Default kernel command string” 选项,通过修改这个选项来修改include/linux/autoconf.h文件中的CONFIG_CMDLINE宏
2.有的时候为了省去make menuconfig的时间,并且内核命令也是固定的,就可以写死内核arch/arm/setup.c文件default_command_line,这个变量本来是初始化为CONFIG_CMDLINE的
3.除了方法1之外,另外一种正规的方法就是将内核命令写入内核参数表中,然后通过start_kernel()->setup_arch()->parse_tags()->parse_tag() ->parse_tag_cmdline()函数来将内核参数表中的内核命令覆盖default_command_line变量(当然你首先得有内核参数表)
4.最后,有的板子会采取直接将内核命令通过bootloader传递给内核,
Linux系统kernel参数传递方式详解
前天正在跟前端的同事调试功能,服务器开好,模拟的玩家登录好,就在倒计时。这时突然运营的同事跑过来说要统计几个服务器玩家的一些情况,也就是需要从几个服的数据库导出部分玩家的数据。好吧,我看了一下时间,11:47。心想,跟前端调试完,去吃个饭再午休一下那就下午再给吧。没想对方来一句“就导个数据库而已,要这么久么?”,而且还是直接跟我上司说的。我嚓,好吧,我导。可问题来了,平时的统计是由php做的,批量部署这些是由运维做的。服务端完全没有对应的工具。而且服务器是在阿里云上的,数据库的用户是限制了ip段登录的,我所在的ip没法登录的。于是,只好终止调试,切ip,写sql,然后用navicat手动一个个服务器导出数据到excel。
事后想想,还是写个脚本吧,不然以后还是会被坑的。
从环境来看,数据库不能直接登录,没法直接导出。不过可以由运维提供key通过ssh登录到远程服务器再将数据导出到本地。
先配置ssh通过key登录服务器。这里略过...
然后就是通过ssh执行命令。先看一下ssh的帮助文档:
usage: ssh [-1246AaCfgKkMNnqsTtVvXxYy] [-b bind_address] [-c cipher_spec][-D [bind_address:]port] [-E log_file] [-e escape_char][-F configfile] [-I pkcs11] [-i identity_file][-L [bind_address:]port:host:hostport] [-l login_name] [-m mac_spec][-O ctl_cmd] [-o option] [-p port][-Q cipher | cipher-auth | mac | kex | key][-R [bind_address:]port:host:hostport] [-S ctl_path] [-W host:port][-w local_tun[:remote_tun]] [user@]hostname [command]
最后一项就是执行指令的。假如远程服务器地址为180.97.33.108,开启的ssh端口为998,ssh用户名为xzc,然后想通过在远程服务器上执行命令ls,那么脚本应该这么写:
代码如下:
ssh xzc@180.97.33.108 -p 998 “ls”
如果ssh的认证key已配置好,那么会把登录后当前目录下的文件列出来,当然第一次登录会提示保存远程服务器的指纹。现在是要导出数据库的数据,那么需要把ls命令换成导出数据库的命令。
代码如下:
echo “select * from user;” | mysql xzc_db -uxzc -pxzcpwd
#或者
mysql xzc_db -uxzc -pxzcpwd -e “select * from user”
上面两命令都可以使用数据库用户xzc,密码xzcpwd从数据库xzc_db打印出user表,使用的是默认的本地数据库地址localhost,默认的端口。如果不是默认,需要指定。
登录OK了,打印也OK了,那么下一步就是导出到文件了。这在bash也就是一个 > 的事。把上面的命令拼起来就是:
代码如下:
ssh xzc@180.97.33.108 -p 998 ‘echo “select * from user;” | mysql xzc_db -uxzc -pxzcpwd‘ > user.txt
这样就把user表导出来本地的user.txt中了。注意“> user.txt”如果放到 ‘‘里则是在远程服务器执行,导出的文件在远程服务器。文件现在也有了,不过是txt,这样交给运营不太好吧。那就导出excel吧。不过遗憾的是我查了N多资料,也找不到mysql不依赖第三方插件或工具导出原生excel的方法。而navicat导出的可是货真价实的excel,如果用notepad++之类的文本工具打开是会乱码的,并且导出的文件不会有编码问题。幸好如果一个txt以tab分割的话,excel也是能认得出来的。于是把user.txt改名user.xls就可以了。但这样做的问题是excel会按自己的方式处理内容的。比如把一个很大的数字转换成科学记数法形式。这些都得手动去处理一下了。
最后,就是写成脚本批量操作了,
附上我使用的脚本一个:
#!/bin/bash# 通过ssh远程执行远程指令# 需要先部署key认证,保证ssh只需要ip、port即可连接# 如果需要和远程服务器交互,请参考ssh的-t、-tt参数# 如果需要反复登录服务器执行多条指令,请使用ssh的通道重用# 参考:en.wikibooks.org/wiki/OpenSSH/Cookbook/Multiplexing# 使用通道需要注意退出通道,如“ssh github.com -O exit”或者“ssh github.com -O stop”# --by coding my life#分别设置ssh用户名、数据库用户名、数据库密码、导出数据SSH_USER=‘xzc_ssh‘DB_USER=‘xzc_db‘DB_PWD=‘xzc_db_pwd123‘EXP_PATH=export_data/# 执行远程命令# $1 服务器ip# $2 ssh端口# $3 指令function exec_remote_command(){ ssh $SSH_USER@$1 -p $2 ‘$3‘}# 执行远程sql,导出数据# $1 服务器ip# $2 ssh端口# $3 指令,多个sql指令如select * from user;select * from bag;也可执行,但结果将会写到同一个文件# s4 服务器# $5 导出文件function export_remote_sql(){ echo export from $4 ... cmd=“echo ”$3“ | mysql $4 -u$DB_USER -p$DB_PWD --default-character-set=utf8” ssh $SSH_USER@$1 -p $2 “$cmd” > $EXP_PATH$4_$5 #如果要导出到远程服务器,将把 > $EXP_PATH$4_$5放到cmd中}# $1 区服名# $2 ip# $3 端口function exec_sqls(){ cat SQLS | while read sql ; dofc=${sql:0:1} if [ “#” == “$fc” ]; then #被注释的不处理continue fi #sql语句中包含空格,不能再以空格来区分。最后一个空格后的是导出的文件名 exp_file=“${sql##* }” #两个#表示正则以最大长度匹配*和一个空格(*后面的空格),截取余下的赋值给exp_file sql_cmd=“${sql%% $exp_file}” #两个%表示从右至左删除%%以后的内容 export_remote_sql $2 $3 “$sql_cmd” $1 “$exp_file” done}# 需要在当前目录下创建服务器列表文件SERVERS,格式为“数据库名 ip ssh端口”,如“xzc_game_s99 127.0.0.1 22”# 需要在当前目录下创建sql命令列表文件SQLS,格式为“sql语句 导出的文件”,如“select * from user; user.xls”# 多个sql请注意用;分开,sql必须以;结束# 文件名中不能包含空格,最终导出的文件为“数据库名_文件名”,如“xzc_game_s99_user.xls”mkdir -p $EXP_PATHcat SERVERS | while read server ; do fc=${server:0:1} if [ “#” == “$fc” ]; then #被注释的不处理 continue fi name=`echo $server|awk ‘{print $1}‘` ip=`echo $server|awk ‘{print $2}‘` port=`echo $server|awk ‘{print $3}‘` exec_sqls $name $ip $portdone
当前目录下的文件如下,其中SERVERS是服务器列表,里面指定数据库名,ip,ss端口,SQLS则指定sql指令及导出的文件名。这两个文件里以#开头的都不会处理:
xzc@xzc-HP-ProBook-4446s:~/桌面/remote_cmd$ lsremote_cmd.sh SERVERS SQLSxzc@xzc-HP-ProBook-4446s:~/桌面/remote_cmd$ cat SERVERS xzc_game_s99 120.0.0.99 6162xzc_game_s91 120.0.0.91 6162xzc_game_s92 120.0.0.92 6162xzc_game_s93 120.0.0.93 6162xzc_game_s94 120.0.0.94 6162#xzc_game_s91 120.0.0.91 6162xzc@xzc-HP-ProBook-4446s:~/桌面/remote_cmd$ cat SQLS #select * money from money; money.xlsselect * from user; user.xlsxzc@xzc-HP-ProBook-4446s:~/桌面/remote_cmd$
ping命令的一般格式为:
ping [-dfnqrRv][-c 发送次数][-i 间隔秒数][-I 网络界面][-l 前置载入][-p 范本样式][-s 数据包大小][-t 存活数值][主机名或IP地址]
参数说明:
-d 使用Socket的SO_DEBUG功能。
-f 极限检测。大量且快速地送网络封包给一台机器,看它的回应。
-n 只输出数值。
-q 不显示任何传送封包的信息,只显示最后的结果。
-r 忽略普通的Routing Table,直接将数据包送到远端主机上。通常是查看本机的网络接口是否有问题。
-R 记录路由过程。
-v 详细显示指令的执行过程。
-c 数目 在发送指定数目的包后停止。
-i 秒数 设定间隔几秒送一个网络封包给一台机器,预设值是一秒送一次。
-I 网络界面使用指定的网络界面送出数据包。
-l 前置载入 设置在送出要求信息之前,先行发出的数据包。
-p 范本样式 设置填满数据包的范本样式。
-s 字节数 指定发送的数据字节数,预设值是56,加上8字节的ICMP头,一共是64ICMP数据字节。
-t 存活数值 设置存活数值TTL的大小。
:linux下的ping和windows下的ping稍有区别,linux下ping不会自动终止,需要按ctrl+c终止或者用参数-c指定要求完成的回应次数
linux下测试本机与目标主机连通性的命令是ping,这里主要讲解两个参数 –c 与 – i
其中 –ccount 次数,也就是ping的次数
-i interval间隔 ,每次ping之间的时间空格
介绍一下ping原理
ping程序实现
@头文件common.h定义,包含程序中用到的头文件及公共变量、宏、常量、静态变量定义
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define IP_HEAD_LEN 20
#define ICMP_LEN 8
#define BUFFER_SIZE 50 * 1024
/*
* 原始套接字的描述符,由于需要在信号处理器
* 和主程序中共享,所以定义为外部变量(全局)
*/
int ip_fd;
/*进程号*/
int p_id;
/*packet_len为IP包头和ICMP包头长度之和*/
extern int packet_len;
/*对端地址*/
struct sockaddr_in send_addr;
/*发送应用缓冲区*/
char send_buf[1024];
/*报文序列号*/
extern int sequence;
/*主机名指针*/
struct hostent *host;
/*标识是否已经接收到回文*/
int flag;
@主函数main.c定义
#include “common.h”
main(int argc, char **argv)
{
/*命令是ping host(主机名)|ip_address(ip地址)*/
if(argc != 2)
{
/*命令不正确*/
fprintf(stderr, “usage: ping .n”);
exit(1);
}
/*创建使用ICMP的原始套接字,这种套接字只有root才能生成*/
ip_fd = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP);
if(ip_fd < 0)
{
fprintf(stderr, “raw socket error.n”);
exit(1);
}
/*改变进程的用户ID, 回收root权限,设置当前用户权限*/
setuid(getpid);
ping(argv[1]);
}
@ping框架的建立ping.c用于初始化ping相关信息及端口信息
#include “common.h”
/*
*handle_alarm用于定时发送IP数据包
*/
void handle_alarm(int signo)
{
send_ip();
alarm(1);
}
ping(char *argv)
{
struct sigaction act;
act.sa_handler = handle_alarm;
act.sa_flags = 0;
sigemptyset(&act.sa_mask);
sigaction(SIGALRM, &act, NULL);
/*获取main的进程id,用于设置ICMP的标志符*/
p_id = getpid();
/*扩大套接字接收缓冲区到50K这样做主要为了减小接收缓冲区溢出的的可能性,若无意中ping一个广播地址或多播地址,将会引来大量应答*/
//setsockopt(ip_fd, SOL_SOCKET, SO_RCVBUF, &BUFFER_SIZE, sizeof(BUFFER_SIZE));
/*只须填写地址信息,不需要指定端口信息,因为原始套接字在传输层之下*/
send_addr.sin_family = AF_INET;
/*判断是主机名还是ip地址*/
if(inet_addr(argv) == INADDR_NONE)
{
/*是主机名*/
if((host = gethostbyname(argv)) == NULL)
{
/*主机名错误*/
perror(“get host by name error: unknow host.”);
exit(1);
}
memcpy(&send_addr.sin_addr, host->h_addr, host->h_length);
}
else
{
/*是ip地址*/
inet_aton(argv, &send_addr.sin_addr);
}
printf(“ping %s(%s) %d(%d) bytes of datan”, argv,
inet_ntoa(send_addr.sin_addr), sizeof(struct timeval),
sizeof(struct timeval) + IP_HEAD_LEN + ICMP_LEN);
flag = 0;
/*触发一个SIGALRM信号*/
raise(SIGALRM);
recv_ip();
}
@发送报文send.c,建立ICMP报文并打包为IP数据包,发送
#include “common.h”
int sequence = 0;
int packet_len = IP_HEAD_LEN + ICMP_LEN;
/*
*send_ip用于发送包含ICMP报文的IP数据包
*/
send_ip(void)
{
if(sequence != 0 && !flag)
{
printf(“Destination Host Unreachablen”);
}
int len;
struct icmphdr *icmp_p;
icmp_p = (struct icmphdr *)send_buf;
/*填写ICMP报文类型*/
icmp_p->type = ICMP_ECHO;
/*填写ICMP报文的代码*/
icmp_p->code = 0;
/*填写ICMP报文的标识符*/
(icmp_p->un).echo.id = p_id;
/*填写ICMP报文的序号,并增加ICMP的序号*/
(icmp_p->un).echo.sequence = sequence ++;
/*记录发送时间*/
gettimeofday((struct timeval*)(icmp_p + 1), NULL);
/*printf(“%dn”, sizeof(struct icmphdr));
printf(“type: %dncode: %dnchecksum: %dnun.echo.id: %dnun.echo.sequence: %dnun.gateway: %dnun.frag.__unused:%dnun.frag.mtu: %dnn”, icmp_p->type, icmp_p->code, icmp_p->checksum, (icmp_p->un).echo.id, (icmp_p->un).echo.sequence, (icmp_p->un).gateway, (icmp_p->un).frag.__unused, (icmp_p->un).frag.mtu);
printf(“type: %dncode: %dnchecksum: %dnun.echo.id: %dnun.echo.sequence: %dnun.gateway: %dnun.frag.__unused: %dnun.frag.mtu: %dnn”, sizeof(icmp_p->type), sizeof(icmp_p->code), sizeof(icmp_p->checksum), sizeof((icmp_p->un).echo.id), sizeof((icmp_p->un).echo.sequence), sizeof((icmp_p->un).gateway), sizeof((icmp_p->un).frag.__unused),sizeof((icmp_p->un).frag.mtu));*/
/*报文的长度等于IP包长度加上ICMP报文长度和数据长度*/
len = sizeof(struct timeval) + packet_len;
/*使用IP计算IP包头校验和的计算方法来计算ICMP的校验和*/
icmp_p->checksum = 0;
icmp_p->checksum = ip_checksum((u_short *)icmp_p, len);
/*发送IP数据包*/
if(sendto(ip_fd, send_buf, len, 0, (struct sockaddr*)&send_addr, sizeof(send_addr)) < 0)
{
fprintf(stderr, “send to error.n”);
}
}
@数据校验check_sum.c实现网络数据的校验工作
#include “common.h”
unsigned short ip_checksum(unsigned short *pcheck, int check_len)
{
int nleft = check_len;
int sum = 0;
unsigned short *p = pcheck;
unsigned short result = 0;
while(nleft > 1)
{
sum = sum + *p ++;
nleft -= 2;
}
if(nleft == 1)
{
*(unsigned char *)(&result) = *(unsigned char *)p;
sum += result;
}
sum = (sum >> 16) + (sum & 0xFFFF);
sum += (sum >> 16);
result = sum;
return result;
}
@数据包接收receive.c,接收IP报文并分析,打印相关信息
#include “common.h”
/*
*recv_ip用于接受包含ICMP报文的IP数据包
*/
recv_ip(void)
{
char recv_buf[1024];
int len;
int n;
struct ip *ip_p;
struct timeval *time_now, *time_send;
struct timeval now;
int iphead_len;
int icmp_len;
struct icmphdr *icmp_p;
float delay;
while(1)
{
n = recvfrom(ip_fd, recv_buf, sizeof(recv_buf), 0, NULL, NULL);
if(n < 0)
{
if(errno = EINTR)
continue;
else
{
printf(“recvfrom error.n”);
continue;
}
}
ip_p = (struct ip*)recv_buf;
/*获取IP包头长度*/
iphead_len = ip_p->ip_hl<<2;
/*获取IP数据包中包含的ICMP报文*/
icmp_p = (struct icmphdr *)(recv_buf + iphead_len);
/*计算ICMP报文长度,它等于接受到的长度减去IP包头的长度*/
icmp_len = n - iphead_len;
if(icmp_len < 8)
{
fprintf(stderr, “error icmp len = %d.n”, icmp_len);
}
/*如果ICMP类型相同则输出显示*/
if(icmp_p->type == ICMP_ECHOREPLY)
{
if((icmp_p->un).echo.id != p_id)
return;
if(icmp_len < 16)
printf(“icmplen = %d.n”, icmp_len);
flag = 1;//表示已经接收到回文;
gettimeofday(&now, NULL);
time_now = &now;
time_send = (struct timeval*)(icmp_p + 1);
if((time_now->tv_usec -= time_send->tv_usec) < 0)
{
time_now->tv_sec --;
time_now->tv_usec += 1000000;
}
time_now->tv_sec -= time_send->tv_sec;
/*计算延时*/
delay = time_now->tv_sec * 1000.0 + time_now->tv_usec / 1000.0;
/*打印接收到的报文相关信息*/
printf(“%d(%d) bytes from %s: icmp_seq=%d ttl=%d time=%.3fmsn”,
icmp_len, n, inet_ntoa(send_addr.sin_addr), (icmp_p->un).echo.sequence,
ip_p->ip_ttl, delay);
}
}
}
@makefile文件
#小型ping程序
#用c语言编写
#03月30日
cc = gcc
bj = main.o ping.o send.o receive.o check_sum.o
ping:$(obj)
$(cc) $(obj) -o ping
main.o:main.c common.h ping.c
$(cc) -c main.c
ping.o:ping.c common.h receive.c send.c
$(cc) -c ping.c
send.o:send.c common.h check_sum.c
$(cc) -c send.c
receive.o:receive.c common.h
$(cc) -c receive.c
check_sum.o:check_sum.c common.h
$(cc) -c check_sum.c
clean:
rm -f $(obj)
参考文献:《linux网络编程》
林宇郭凌云编著
人民邮电出版社出版
进入makefile所在目录
执行make -f makefile
make clean即可
程序运行结果:(平台ubuntu)
lwj@lwj-desktop:~/Desktop/C/myping$ make -f makefile
gcc -c main.c
gcc -c ping.c
gcc -c send.c
gcc -c receive.c
gcc -c check_sum.c
gcc main.o ping.o send.o receive.o check_sum.o -o ping
lwj@lwj-desktop:~/Desktop/C/myping$ make clean
rm -f main.o ping.o send.o receive.o check_sum.o
lwj@lwj-desktop:~/Desktop/C/myping$ sudo ./ping localhost
Password:
ping localhost(127.0.0.1) 8(36) bytes of data
36(56) bytes from 127.0.0.1: icmp_seq=0 ttl=64 time=0.061ms
36(56) bytes from 127.0.0.1: icmp_seq=1 ttl=64 time=0.051ms
36(56) bytes from 127.0.0.1: icmp_seq=2 ttl=64 time=0.039ms
36(56) bytes from 127.0.0.1: icmp_seq=3 ttl=64 time=0.051ms
36(56) bytes from 127.0.0.1: icmp_seq=4 ttl=64 time=0.050ms
36(56) bytes from 127.0.0.1: icmp_seq=5 ttl=64 time=0.052ms
36(56) bytes from 127.0.0.1: icmp_seq=6 ttl=64 time=0.051ms
36(56) bytes from 127.0.0.1: icmp_seq=7 ttl=64 time=0.053ms
[2]+Stoppedsudo ./ping localhost
lwj@lwj-desktop:~/Desktop/C/myping$ sudo ./ping 10.3.2.206
ping 10.3.2.206(10.3.2.206) 8(36) bytes of data
36(56) bytes from 10.3.2.206: icmp_seq=0 ttl=64 time=0.233ms
36(56) bytes from 10.3.2.206: icmp_seq=1 ttl=64 time=0.063ms
36(56) bytes from 10.3.2.206: icmp_seq=2 ttl=64 time=0.044ms
36(56) bytes from 10.3.2.206: icmp_seq=3 ttl=64 time=0.054ms
36(56) bytes from 10.3.2.206: icmp_seq=4 ttl=64 time=0.044ms
36(56) bytes from 10.3.2.206: icmp_seq=5 ttl=64 time=0.052ms
36(56) bytes from 10.3.2.206: icmp_seq=6 ttl=64 time=0.058ms
36(56) bytes from 10.3.2.206: icmp_seq=7 ttl=64 time=0.055ms
36(56) bytes from 10.3.2.206: icmp_seq=8 ttl=64 time=0.066ms
36(56) bytes from 10.3.2.206: icmp_seq=9 ttl=64 time=0.053ms
36(56) bytes from 10.3.2.206: icmp_seq=10 ttl=64 time=0.053ms
36(56) bytes from 10.3.2.206: icmp_seq=11 ttl=64 time=0.052ms
[3]+Stoppedsudo ./ping 10.3.2.206
lwj@lwj-desktop:~/Desktop/C/myping$ sudo ./ping 10.3.2.205