需求描述
编写程序,完成在主机上实现对备机上目录及文件的操作。例如,主机为A,备机为B,要求编写的程序运行在A机上,该程序实现在B机上创建文件目录及拷贝文件的操作。

需求分析
我们先不考虑用程序实现需求,如果是让我们用手工实现需求中描述的操作,我们会怎么做呢?大家也许都会想到使用telnet的方式。也就是说,在A机上使用telnet远程到B机上去,然后在B机上执行创建目录和拷贝文件的操作。

我们也可以在程序中使用telnet来完成上述需求。

程序编写
我们可以使用两种方式来执行telnet命令,一种是直接使用系统提供的system函数,另一种是使用管道的方式。以下是这两种方式编写出来的程序:

1.使用system函数

/**********************************************************************
* 版权所有 (C)2015, Zhou Zhaoxiong。
*
* 文件名称:System.c
* 文件标识:无
* 内容摘要:测试system的用法
* 其它说明:利用system实现远程操作
* 当前版本:V1.0
* 作 者:Zhou Zhaoxiong
* 完成日期:20151225
*
**********************************************************************/
#include <stdio.h>
#include <string.h>

// 重定义数据类型
typedef unsigned char UINT8;
typedef signed int INT32;

// 函数声明
void DealRemoteFile(void);


/**********************************************************************
* 功能描述:主函数
* 输入参数:无
* 输出参数:无
* 返 回 值:无
* 其它说明:无
* 修改日期 版本号 修改人 修改内容
* ---------------------------------------------------------------
* 20151225 V1.0 Zhou Zhaoxiong 创建
***********************************************************************/
INT32 main()
{
DealRemoteFile();

return 0;
}


/**********************************************************************
* 功能描述:处理远端的文件
* 输入参数:无
* 输出参数:无
* 返 回 值:无
* 其它说明:无
* 修改日期 版本号 修改人 修改内容
* ---------------------------------------------------------------
* 20151225 V1.0 Zhou Zhaoxiong 创建
***********************************************************************/
void DealRemoteFile(void)
{
UINT8 szCmdBuf[1024] = {0};
UINT8 szResultBuf[1024] = {0};

snprintf(szCmdBuf, sizeof(szCmdBuf)-1, "(sleep 0.5; echo zhou; sleep 1; echo zhou; sleep 5; echo \"cd /home/zhou/zhouzx/Test;mkdir -p TestFile;cp TestFile.txt ./TestFile;\"; sleep 2) | telnet 10.10.10.10");

printf("DealRemoteFile: Cmd is %s\n", szCmdBuf);

system(szCmdBuf);

2.使用管道的方式

/**********************************************************************
* 版权所有 (C)2015, Zhou Zhaoxiong。
*
* 文件名称:PipeOpenAndClose.c
* 文件标识:无
* 内容摘要:测试popen和pclose的用法
* 其它说明:利用popen和pclose处理远端的文件
* 当前版本:V1.0
* 作 者:Zhou Zhaoxiong
* 完成日期:20151225
*
**********************************************************************/
#include <stdio.h>
#include <string.h>

// 重定义数据类型
typedef unsigned char UINT8;
typedef signed int INT32;

// 函数声明
void DealRemoteFile(void);


/**********************************************************************
* 功能描述:主函数
* 输入参数:无
* 输出参数:无
* 返 回 值:无
* 其它说明:无
* 修改日期 版本号 修改人 修改内容
* ---------------------------------------------------------------
* 20151225 V1.0 Zhou Zhaoxiong 创建
***********************************************************************/
INT32 main()
{
DealRemoteFile();

return 0;
}


/**********************************************************************
* 功能描述:处理远端的文件
* 输入参数:无
* 输出参数:无
* 返 回 值:无
* 其它说明:无
* 修改日期 版本号 修改人 修改内容
* ---------------------------------------------------------------
* 20151225 V1.0 Zhou Zhaoxiong 创建
***********************************************************************/
void DealRemoteFile(void)
{
UINT8 szCmdBuf[1024] = {0};
UINT8 szResultBuf[1024] = {0};
FILE *fpPipe = NULL;

snprintf(szCmdBuf, sizeof(szCmdBuf)-1, "(sleep 0.5; echo zhou; sleep 1; echo zhou; sleep 5; echo \"cd /home/zhou/zhouzx/Test;mkdir -p TestFile;cp TestFile.txt ./TestFile;\"; sleep 2) | telnet 10.10.10.10");

printf("DealRemoteFile: Cmd is %s\n", szCmdBuf);

fpPipe = popen(szCmdBuf, "r");
if (NULL == fpPipe)
{
printf("DealRemoteFile: fpPipe is NULL!\n");
return;
}

while (fgets(szResultBuf, sizeof(szResultBuf), fpPipe) != NULL)
{
if ('\n' == szResultBuf[strlen(szResultBuf)-1]) // 将换行符去掉
{
szResultBuf[strlen(szResultBuf)-1] = '\0';
}

printf("DealRemoteFile: ResultBuf is %s\n", szResultBuf);
}

pclose(fpPipe); // 关闭管道
fpPipe = NULL;

程序编译及运行
将System.c和PipeOpenAndClose.c文件上传到A机上,分别使用“gcc -g -o System System.c”和“gcc -g -o PipeOpenAndClose PipeOpenAndClose.c”命令对这两个文件进行编译,再分别运行“System”和“PipeOpenAndClose”命令,即可在B机上观察到相关操作已执行成功(注意:要确保TestFile.txt文件存在于/home/zhou/zhouzx/Test目录下)。

程序说明
1.以上两个程序中的szCmdBuf中的命令是程序的核心,其模式是“shell命令 | telnet x.x.x.x”,其中“x.x.x.x”是B机的IP,shell命令中的“sleep”让程序休眠,第一个echo后面的变量是用户名,第二个echo后面的变量是密码,第三个echo便执行具体的创建目录和拷贝文件的操作。

2.popen和pclose配合来实现相关shell命令的执行。popen函数通过创建一个管道,调用fork产生一个子进程,这个进程必须由pclose函数来关闭。

本文中的程序已经提到了GitHub上,欢迎下载阅读:​​https://github.com/zhouzxi/PipeOpenAndClose​