《Linux/Unix系统编程手册》读书笔记 目录

第8章

本章讲了用户和组,还有记录用户的密码文件/etc/passwd,shadow密码文件/etc/shadow还有组文件/etc/group。

每个用户都有唯一的用户名和相关的用户标识符(UID)。用户可以属于一个或多个组,每个组都有唯一的组名和相关的组标识符(GID)。

用户和组的用途为:1、可以确定各种系统资源的所有权;2、对赋予进程访问上述资源的权限加以控制。

首先来看一下密码文件/etc/passwd

lancelot@debian:~$ cat /etc/passwd
root:x:::root:/root:/bin/bash
daemon:x:::daemon:/usr/sbin:/bin/sh
bin:x:::bin:/bin:/bin/sh
sys:x:::sys:/dev:/bin/sh
sync:x:::sync:/bin:/bin/sync
games:x:::games:/usr/games:/bin/sh
man:x:::man:/var/cache/man:/bin/sh
lp:x:::lp:/var/spool/lpd:/bin/sh
mail:x:::mail:/var/mail:/bin/sh
news:x:::news:/var/spool/news:/bin/sh
uucp:x:::uucp:/var/spool/uucp:/bin/sh
proxy:x:::proxy:/bin:/bin/sh
www-data:x:::www-data:/var/www:/bin/sh
backup:x:::backup:/var/backups:/bin/sh
list:x:::Mailing List Manager:/var/list:/bin/sh
irc:x:::ircd:/var/run/ircd:/bin/sh
gnats:x:::Gnats Bug-Reporting System (admin):/var/lib/gnats:/bin/sh
nobody:x:::nobody:/nonexistent:/bin/sh
libuuid:x::::/var/lib/libuuid:/bin/sh
messagebus:x::::/var/run/dbus:/bin/false
colord:x:::colord colour management daemon,,,:/var/lib/colord:/bin/false
usbmux:x:::usbmux daemon,,,:/home/usbmux:/bin/false
Debian-exim:x::::/var/spool/exim4:/bin/false
avahi:x:::Avahi mDNS daemon,,,:/var/run/avahi-daemon:/bin/false
pulse:x:::PulseAudio daemon,,,:/var/run/pulse:/bin/false
speech-dispatcher:x:::Speech Dispatcher,,,:/var/run/speech-dispatcher:/bin/sh
hplip:x:::HPLIP system user,,,:/var/run/hplip:/bin/false
sshd:x::::/var/run/sshd:/usr/sbin/nologin
rtkit:x:::RealtimeKit,,,:/proc:/bin/false
statd:x::::/var/lib/nfs:/bin/false
saned:x::::/home/saned:/bin/false
Debian-gdm:x:::Gnome Display Manager:/var/lib/gdm3:/bin/false
lancelot:x:::lancelot,,,:/home/lancelot:/bin/bash
mysql:x:::MySQL Server,,,:/nonexistent:/bin/false
ftp:x:::ftp daemon,,,:/srv/ftp:/bin/false
telnetd:x::::/nonexistent:/bin/false

第一个字段是登录名;第二个字段是经过加密后的密码(x),实际上经过加密后的密码是存放在shadow密码文件;

第三个字段是用户的ID(UID);第四个字段是组ID(GID);第五个字段是注释;

第六个字段是主目录,是用户登录后的初始路径;第七个字段是登录shell。

接着来看实际存放密码的shadow 密码文件/etc/shadow

格式如下:

lancelot:$$tnTgvJYU$OhoUNZNIeNU7rlZf/f14oD2g.Uz8SbrnWeZbR4yL4XXRvzbCeijsAZE7Y9HlzU4thKVBVcqucwntJBi/4BoY60:::::::
mysql:!:::::::
ftp:*:::::::
telnetd:*:::::::

很明显可以看到第一个字段为用户登录名,第二个字段为见过加密后的密码,后面的字段为与安全性相关的字段。

看完用户,我们来看组文件/etc/group

lancelot@debian:~$ cat /etc/group
root:x::
daemon:x::
bin:x::
sys:x::
adm:x::
tty:x::
disk:x::
lp:x::
mail:x::
news:x::
uucp:x::
man:x::
proxy:x::
kmem:x::
dialout:x::
fax:x::
voice:x::
cdrom:x::lancelot
floppy:x::lancelot
tape:x::
sudo:x::
audio:x::pulse,lancelot
dip:x::lancelot
www-data:x::
backup:x::
operator:x::
list:x::
irc:x::
src:x::
gnats:x::
shadow:x::
utmp:x::telnetd
video:x::lancelot
sasl:x::
plugdev:x::lancelot
staff:x::
games:x::
users:x::
nogroup:x::
libuuid:x::
crontab:x::
fuse:x::
scanner:x::saned,lancelot
messagebus:x::
colord:x::
lpadmin:x::
ssl-cert:x::
bluetooth:x::lancelot
utempter:x::
netdev:x::lancelot
Debian-exim:x::
mlocate:x::
ssh:x::
avahi:x::
pulse:x::
pulse-access:x::
rtkit:x::
saned:x::
Debian-gdm:x::
lancelot:x::
mysql:x::
ftp:x::
vboxusers:x::
telnetd:x::

第一个字段是组的名称,第二个字段是经过加密的密码,第三个字段是组ID(GID),第四个字段是用户列表。

对于加密后的密码存放在类似/etc/shadow的文件(/etc/gshadow),格式如下:

cdrom:*::lancelot
floppy:*::lancelot
tape:*::
sudo:*::
audio:*::pulse,lancelot
dip:*::lancelot
www-data:*::
backup:*::
operator:*::
list:*::
irc:*::
src:*::
gnats:*::
shadow:*::
utmp:*::telnetd
video:*::lancelot

最后一个字段很明显是用户列表。

接着,我们来看如何通过库函数来获取上面提到的信息。

一、从/etc/shadow获取记录:

 #include <pwd.h>

 struct passwd *getpwnam(const char *name);

 struct passwd *getpwuid(uid_t uid);

getpwnam()是根据提供的登录名返回一个指向对应的密码记录的指针。getpwuid()是根据提供的uid来返回一个指向对应的密码记录的指针。

如果出现错误返回NULL。

PS:对于getpwnam和getpwuid返回的指针都是指向由静态分配而成的内存,因此都是不可重入。

这个应该是练习8-1的答案。

8-1:执行下列代码时,将会发现,尽管这两个用户在密码文件中对应不同的ID,但该程序的输出还是会将同一个数字显示两次。请问为什么?

printf("%ld %ld\n", (long)(getpwnam("avr")->pw_uid), (long)(getpwnam("tsr")->pw_uid));

书本的答案是这样的:getpwnam()的调用在printf()输出之前,那么getpwnam()返回的结果存放在静态分配的缓冲区中,第二个getpwnam()的返回结果会覆盖第一个的结果。

然后我测试了一下,代码如下:

 /*
* =====================================================================================
*
* Filename: tt1.c
*
* Description:
*
* Version: 1.0
* Created: 2014年04月22日 14时56分10秒
* Revision: none
* Compiler: gcc
*
* Author: alan (), alan19920626@gmail.com
* Organization:
*
* =====================================================================================
*/ #include <stdio.h>
#include <pwd.h> int main(int argc, char *argv[]){
printf("%ld %ld\n", (long)(getpwnam("lancelot")->pw_uid), (long)(getpwnam("root")->pw_uid));
return ;
}

测试结果:

lancelot@debian:~/Code/tlpi$ ./a.out
 

然后整个人就凌乱了。。。。。。。。。。。。。。

后来,我把程序改成这样:

 #include <stdio.h>
#include <pwd.h> int main(int argc, char *argv[]){
struct passwd *p1, *p2;
p1 = getpwnam("lancelot");
p2 = getpwnam("root");
printf("%ld %ld\n", p1->pw_uid, p2->pw_uid);
//printf("%ld %ld\n", (long)(getpwnam("lancelot")->pw_uid), (long)(getpwnam("root")->pw_uid));
return ;
}

才能输出相同的值,因为两个指针变量指向的地址是一样的。但是之前那种是正确的,因为函数的参数是一值传递的方式传递,所以不存在题目说的两个值相同。

后来我查找了作者的网站的磡误:http://www.man7.org/tlpi/errata/index.html

发现这条题的题目修改了!!!!!!

这样传递指针,两个输出的名字就是一样的。。。。。。。。。。。。。。

下面是作者的解释:

好吧,以后还要把书上的错误修改。。。。。。。

PS:再次证明不管例子多简单也要试一试。。。。。

二、从/etc/group获取记录

 #include <grp.h>

 struct group *getgrnam(const char *name);

 struct group *getgrgid(gid_t gid);

getgrnam根据提供的组名返回指向组的密码记录的指针,getgrigid则根据提供的组号GID返回该指针。失败调用返回NULL

PS:这两个函数也是不可重入的函数。

三、扫描密码文件和组文件

 #include <pwd.h>

 struct passwd *getpwent(void);

 void setpwent(void);

 void endpwent(void);

getpwent()可以逐条返回记录。setpwent()可以重置为。/etc/passwd文件的起始处。endpwent()可以关闭文件。

四、从/etc/shadow密码文件中获取记录

 #include <shadow.h>

 struct spwd *getspnam(const char *name);

 struct spwd *getspent(void);

 void setspent(void);

 void endspent(void);

getspnam和getspent会返回指向shadow密码记录的指针,失败调用返回NULL。

setspent会重置为文件的起始位置,endspent会关闭文件。

五、密码加密

 #define _XOPEN_SOURCE
#include <unistd.h> char *crypt(const char *key, const key *salt);

#define _XOPEN_SOURCE 是为了获取crypt的声明。

key为输入的密码,salt指向一个两字节的字符串,用来改变DES算法。成功调用返回加密后的密码,失败返回NULL。

---------------------吐槽:好好写博客,写博客的时候思考的感觉真好,要多读书,多code,多思考---------------------------

听同学说有很多厉害的人已经找到实习了,真的后悔大一大二的无作为。正因为后悔,所以要做得更好!!!!

继续努力!!!!!!

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

练习:

8-2: 使用sptwent()、getpwent()和endpwent()来实现getpwnam()。

这题个人觉得很简单,因为getpwent()是逐条查找,查找匹配就输出。输出完应该调用sptwent将文件的偏移量重置为文件的起始位置。结束的时候调用endpwent关闭文件。

 /*
* =====================================================================================
*
* Filename: 8-2.c
*
* Description:
*
* Version: 1.0
* Created: 2014年04月22日 16时48分09秒
* Revision: none
* Compiler: gcc
*
* Author: alan (), alan19920626@gmail.com
* Organization:
*
* =====================================================================================
*/ #include <pwd.h>
#include <string.h>
#include <stdio.h>
#include "tlpi_hdr.h" struct passwd * Getpwnam(const char *name){
struct passwd *pwd;
while((pwd = getpwent()) != NULL){
if(strcmp(pwd->pw_name, name) == ){
setpwent();
return pwd;
}
}
setpwent();
return NULL;
} int main(int argc, char *argv[]){
int i;
struct passwd *p;
if(argc < || strcmp(argv[], "--help") == )
usageErr("%s user-names...", argv[]); for(i = ; i < argc; ++i){
p = Getpwnam((const char *)argv[i]);
if(p == NULL)
printf("%s does not exit\n", argv[i]);
else{
printf("%s, UID: %ld, GID: %ld\n", argv[i], (long)p->pw_uid, (long)p->pw_gid);
}
} endpwent();
exit(EXIT_SUCCESS);
}

测试结果:

lancelot@debian:~/Code/tlpi$ ./a.out lancelot root
lancelot, UID: , GID:
root, UID: , GID:

《Linux/Unix系统编程手册》读书笔记5的更多相关文章

  1. 【鸟哥的Linux私房菜】笔记1

    Linux是什么 从操作系统与cpu架构关系到linux  Richard Mathew Stallman GPL 关于GNU计划 Linux的发展 Linux的核心版本 Linux的特色 Linux ...

  2. 【鸟哥的Linux私房菜】笔记3

    正确地开机 最好不要使用root账号登陆!GNOME图形界面 View items as a list X WindowShell 文本交互界面bash是Shell的名称,Linux的默认壳程序就是b ...

  3. 【鸟哥的Linux私房菜】笔记2

    Linux的应用 学习资源整理 安装记录 >< 1.Linux的应用: 网络服务器 数据库 学术机构的高效运算任务 嵌入式系统 ... 2.挂载与磁盘分区 学习资源整理 学习 1.书上的网 ...

  4. 《鸟哥的Linux私房菜》笔记——02. 关于Linux

    Unix 历史 1969年以前:伟大的梦想--Bell, MIT 与 GE 的「Multics」系统 1969年:Ken Thompson 的小型 file server system 1973年:U ...

  5. 《鸟哥的Linux私房菜》笔记——03. 磁盘分区

    Everything is a file. 常见硬件对应于 Linux 下的文件(/dev目录下) 装置 装置在Linux内的档名 SCSI/SATA/U盘硬盘机 /dev/sd[a-p] U盘 /d ...

  6. 鸟哥的linux私房菜学习笔记 __ 命令与文件的搜寻

    连续输入两次[tab]按键就能够知道使用者有多少命令可以下达.那你知不知道这些命令的完整档名放在哪里?举例来说,ls 这个常用的命令放在哪里呢? 就透过 which 或 type 来找寻吧! 范例一: ...

  7. 【鸟哥的Linux私房菜】笔记

    操作系统核心的功能! 驱动程序与操作系统的关系 2. [计算机组成之组件] 3.CPU实际要处理的数据完全来自于主存储器,这是一个很重要的概念! 4.CPU是整个计算机系统最重要的部分,那么目前世界上 ...

  8. 《鸟哥的Linux私房菜》笔记——04. 简单命令行

    键入命令 [dmtsai@study ~]$ command [-options] parameter1 parameter2 ... 指令 選項 參數(1) 參數(2) 注意:有时也可以使用 + 放 ...

  9. 鸟哥的Linux私房菜学习笔记——文件权限与目录配置

    Linux的文件权限和目录配置 在linux中的每个用户必需属于一个组,不能独立于组外.在linux中每个文件有所有者.所在组.其它组的概念. (1)所有者 一般为文件的创建者,谁创建了该文件,就是天 ...

  10. 鸟哥的Linux私房菜学习笔记(1)

    2014/10/29 1.档案的权限管理分为三个部分: 拥有者.群组.其他 2.ls -al 命令可以看到档案的详细信息 3.档案的属性中由十个部分构成 第一个部分是档案类型 -代表档案.d代表文件夹 ...

随机推荐

  1. proxy server 代理服务器

    有时候,我觉得自己需要去搞明白.搞清楚一个概念,帮我打通一下自己的知识体系,或者说,尝试联络起来. 1. 简介 突破自身IP限制,访问国外站点. 访问单位或者团体内部资源. 突破中国电信的IP封锁. ...

  2. printf的一个常用技巧

    acm 的题目经常要求输出最后面一位不能有空格: 用if语句显得代码难看: 实现如下: int a[5]={1,2,3,4,5}; for(int i=0;i<=4;i++) printf(&q ...

  3. IIS OCIEnvCreate failed with return code -1

    现象:windows server2008服务器,MVC使用NHiberate连接Oracle11g,程序部署到IIS后无法访问数据库,抛上述异常:在服务器上安装VS调试可以访问数据库 解决方法:连接 ...

  4. zepto判断手机横竖屏

    var CheckOrientation = (function(){ var win = $( window ), get_orientation, last_orientation, initia ...

  5. 简述负载均衡&CDN技术

    曾经见到知乎上有人问“为什么像facebook这类的网站需要上千个工程师维护?”,下面的回答多种多样,但总结起来就是:一个高性能的web系统需要从无数个角度去考虑他,大到服务器的布局,小到软件中某个文 ...

  6. eclipse中设置中文javadoc+如何查看class的中文javadoc

    一.  eclipse中设置中文javadoc 1.先到http://download.java.net/jdk/jdk-api-localizations/jdk-api-zh-cn/publish ...

  7. 剑指offer--面试题21

    题目:设计包含min函数的栈,pop(),push(),min()的时间复杂度均为O(1) 自己所写代码如下:(写‘栈’的代码还是有些不熟练!) #include <iostream> u ...

  8. 【bzoj1011】[HNOI2008]遥远的行星

    1011: [HNOI2008]遥远的行星 Time Limit: 10 Sec  Memory Limit: 162 MBSec  Special JudgeSubmit: 3711  Solved ...

  9. MySQL 5.7 reference about JSON

    最近需要用到MySQL 5.7中的JSON,总结一下MySQL中关于JSON的内容 参考: 11.6 The JSON Data Type 12.16 JSON Functions JSON Func ...

  10. ASP.NET页面优化性能提升方法记录

    今天与大家分享:一种优化页面执行速度的方法.采用这个方法,可以使用页面的执行速度获得[8倍]的提升效果. 为了让您对优化的效果有个直观的了解,我准备了下面的测试结果截图: 测试环境:1. Window ...