[root@bogon code]# cat a.c
#include<stdio.h>
#include<pwd.h>
int main()
{
struct passwd *pw;//定义指针pw记录返回值
pw=getpwnam("root");
printf("%s %s %d %d %s %s %s\n",pw->pw_name,pw->pw_passwd,pw->pw_uid,pw->pw_gid,pw->pw_gecos,pw->pw_dir,pw->pw_shell);
return 0;
}
[root@bogon code]# gcc a.c
[root@bogon code]# ./a.out
root x 0 0 root /root /bin/bash
[root@bogon code]# cat /etc/passwd
root:x:0:0:root:/root:/bin/bash

结构体passwd如下

struct psswd
{
char *pw_name;
char *pw_passwd;
uid_t *pw_uid;
gid_t *pw_gid;
char *pw_gecos;//用户信息
char *pw_dir;
char *pw_shell;
};

还可以根据uid获取记录信息,使用方法如上

struct passwd *getpwuid(uid_t uid)

上面两个函数如果没有发现匹配记录,返回NULl且不会改变errno,所以可以根据errno的值来区分出错和未匹配记录。

下面我们来看看如何获取组记录

[root@bogon code]# cat b.c
#include<stdio.h>
#include<grp.h>
int main()
{
struct group *grp;
grp=getgrnam("root");
printf("%s %s %d\n",grp->gr_name,grp->gr_passwd,grp->gr_gid);
return 0;
} [root@bogon code]# gcc b.c
[root@bogon code]# ./a.out
root x 0
[root@bogon code]# cat /etc/group
root:x:0:

扫描密码文件所有记录

endpwent()读取完毕关闭,如果不关闭,那么在下面再getpwent打印所有记录,不会有任何显示,因为第一次打印完后,指针已经到达密码文件尾部。或者也可以使用setpwent(),该函数相当与再把指针重返密码文件开始位置

[root@bogon code]# cat c.c
#include<stdio.h>
#include<pwd.h>
int main()
{
struct passwd *pwd;
while((pwd=getpwent())!=NULL)
printf("%s %s\n",pwd->pw_name,pwd->pw_passwd);
endpwent();
return 0;
}
[root@bogon code]# gcc c.c
[root@bogon code]# ./a.out
root x
bin x
daemon x
adm x
lp x
sync x
shutdown x
halt x
mail x
内容过多,该处省略

当然组密码文件也可以使用类似的函数,getgrent,setgrent,endgrent用法类似上面的

接下来看一下怎么从shadow密码文件读取记录,还有getspnam(),setspent()两函数没用,用法跟上面的也差不多

[root@bogon code]# cat c.c
#include<stdio.h>
#include<shadow.h>
int main()
{
struct spwd *spw;
while((spw=getspent())!=NULL)
printf("%s %s\n",spw->sp_namp,spw->sp_pwdp);
endspent();
return 0;
}
[root@bogon code]# gcc c.c
[root@bogon code]# ./a.out
root $1$GpBw2JM2$wdjExHPJ0qkM9lej7mzzq/
bin *
daemon *
adm *
lp *
sync *

最后来谈一谈加密函数

[root@bogon code]# cat d.c
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<error.h>
#include<string.h>
#include<pwd.h>
#include<shadow.h>
int main()
{
struct passwd *pw;
struct spwd *spw;
char *mypw,*myspw,*password,*encrypted,*p;
char username[20];
printf("username:");
scanf("%s",username);
pw=getpwnam(username);//首先查询是否有该用户,如果没有就直接exit(1)
if(pw==NULL)
{
perror("getpwnam");
exit(1);
}
spw=getspnam(username);//读取shadow密码文件该用户的信息
if(spw!=NULL)
pw->pw_passwd=spw->sp_pwdp;//获取用户shadow密码
password=getpass("Password:");//getpass函数用来输入密码
encrypted=crypt(password,pw->pw_passwd);//对刚输入的密码加密,salt是pw->pw_paswd
for(p=password;*p!='\0';)//为了安全,一旦获取加密密码,就立即把输入的密码从内存上抹除,在这里是直接赋值\0
*p++='\0';
if(strcmp(encrypted,pw->pw_passwd))//对比加密后的密码是否与shadow用户密码一致
printf("username or password wrong\n");
else
printf("input succeed\n");
return 0;
}
[root@bogon code]# gcc -lcrypt d.c
d.c:25:11: warning: assignment makes pointer from integer without a cast [enabled by default]
encrypted=crypt(password,pw->pw_passwd);//这个我还不知道怎么去掉这个警告
^
[root@bogon code]# ./a.out //输入正确的用户名和密码
username:root
Password:
input succeed
[root@bogon code]# ./a.out //输入正确的用户名和错误密码
username:root
Password:
username or password wrong
[root@bogon code]# ./a.out //输入错误的用户名
username:errorname
getpwnam: Success //暂时没明白为什么提示信息是Success
[root@bogon code]#

getpwnam,getgrnam,getpwent,crypt等函数的更多相关文章

  1. Crypt加密函数简介(C语言)

    定义函数 char * crypt (const char *key,const char * salt); 函数说明 crypt是个密码加密函数,它是基于Data Encryption Standa ...

  2. PHP的几个常用加密函数(转载 https://jellybool.com/post/php-encrypt-functions)

    PHP的几个常用加密函数 在网站的开发过程中,常常需要对部分数据(如用户密码)进行加密,本文主要介绍PHP的几个常见的加密函数 MD5加密: string md5 ( string $str [, b ...

  3. 关于getpw系列函数返回的静态区域

    首先说一下什么是getpw系列函数,它主要是指这些函数: 这些函数根据一个用户名(getpwnam和getpwnam_r两个函数)或者一个用户ID(getpwuid和getpwuid_r)来获取这个用 ...

  4. PHP的几个常用加密函数【转载】

    转自 https://jellybool.com/post/php-encrypt-functions 在网站的开发过程中,常常需要对部分数据(如用户密码)进行加密,本文主要介绍PHP的几个常见的加密 ...

  5. Linux获取当前用户信息函数

    转自:http://net.pku.edu.cn/~yhf/linux_c/function/07.html endgrent(关闭组文件) 相关函数 getgrent,setgrent 表头文件 # ...

  6. UNIX环境高级编程 第6章 系统数据文件和信息

    UNIX系统的正常运作需要用到大量与系统有关的数据文件,例如系统用户账号.用户密码.用户组等文件.出于历史原因,这些数据文件都是ASCII文本文件,并且使用标准I/O库函数来读取. 口令文件 /etc ...

  7. 用户组篇endgrent getpw getuid

    endgrent(关闭组文件) 相关函数 getgrent,setgrent 表头文件 #include<grp.h> #include<sys/types.h> 定义函数 v ...

  8. unix环境高级编程第六章笔记

    口令文件 阴影口令 组文件 附属组ID 登录账户记录 系统标识 口令文件<\h2> /etc/passwd文件是UNIX安全的关键文件之一.该文件用于用户登录时校验用户的口令,文件中每行的 ...

  9. (七) 一起学 Unix 环境高级编程(APUE) 之 进程关系 和 守护进程

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

随机推荐

  1. day32 信号量 事件 管道 进程池

    今日主要内容: 1.管道(Pipe) 数据接收一次就没有了 2.事件(Event) 3.基于事件的进程通信 4.信号量(Semaphore) 5. 进程池(重点) 6.进程池的同步方法和异步方法 7. ...

  2. flask-admin fileadmin 上传文件,中文名的解决方案 重写部分secure_filename

    class upload_view(FileAdmin): def _save_form_files(self, directory, path, form): super() filename = ...

  3. flask上传下载文件(一)下载

    简介: 作为一个可以和用户交互的web应用,必然要有数据导出功能,导出到excel是比较常用的方式. flask有一个扩展叫flask-excel,可能不适合中国人用,因为没有看到修改列名的功能.也许 ...

  4. c# 十进制转二、八、十六进制

    一.十进制转二.八.十.十六进制字符串 Convert.ToString(int decNum,int toBase); decNum为十进制字符串, toBase可以为2.8.10.16 如果要转换 ...

  5. python中的if判断语句

    判断(if)语句 目标 开发中的应用场景 if 语句体验 if 语句进阶 综合应用 01. 开发中的应用场景 生活中的判断几乎是无所不在的,我们每天都在做各种各样的选择,如果这样?如果那样?……  ...

  6. netty源码理解(三) 从channel读取数据

    下面的是ServerBootstrap 的内部类 ServerBootstrapAcceptor extends ChannelInboundHandlerAdapter 的方法 这里其实卡住了我很长 ...

  7. poj2406(kmp算法)

    Given two strings a and b we define a*b to be their concatenation. For example, if a = "abc&quo ...

  8. 解决At least one JAR was scanned for TLDs yet contained no TLDs. Enable debug logging for this logger for a complete list of JARs that were scanned but no TLDs were found in them. Skipping unneeded JARs

    在写spring security小程序时遇到  At least one JAR was scanned for TLDs yet contained no TLDs. Enable debug l ...

  9. 基于区域的OSPF简单认证

    实验要求:掌握OSPF区域简单认证配置 拓扑如下: 配置如下: R1enable configure terminal interface s0/0/0ip address 192.168.1.1 2 ...

  10. 三层交换机实现VLAN间通信

    实验要求:使用三层交换机,让同一vlan的主机能通信,不同vlan的主机也能通信 拓扑如下: 涉及内容: 1.VTP的创建和配置 2.vlan的创建和划分 3.三层交换机的配置 4.端口的trunk模 ...