两种RPC编程

第一种:
RPC编程首先需要一个.x文件,然后用rpcgen生成客户端与服务器端程序的c代码,分别编译,运行即可。

rpcgen 简介

rpcgen可以自动生成RPC服务器程序的大多数代码,它的输入为一个规格说明文件,它的输出为一个C语言的源程序。规格文件(*.x)包含常量、全局数据类型以及远程过程的声明。Rpcgen产生的代码包含了实现客户机和服务器程序所需要的大部分源代码。他包括参数整理、发送RPC报文、参数和结果的外部数据表示以及本地数据表示的转换等。不过在由rpcgen生成的源文件中,没有过程的具体实现,所以程序员必须要手工编辑这些文件,实现这些过程。

 

Rpc自动生成的文件

文件名

作用

Makefile.file

该文件用于编译所有客户机,服务器代码

File_clnt.c

该文件包含client_stub,程序员一般不用修改

File_svc.c

该文件包含server_stub,程序员一般不用修改

File.h

该文件包含了从说明中产生的所有XDR类型

File_xdr.c

该文件包含了客户机和服务器stub所需的XDR过滤器,程序员一般不用修改

File_server.c

如果生成此文件,则该文件包含远程服务的stub

File_client.c

如果生成此文件,则该文件包含了骨架客户机程序。

 

Rpcgen的部分选项

-a 生成所有源程序,包括客户机和服务器源程序。

-C 使用ANSI C标准生成编码。

-c 生成xdr转码C程序。(file_xdr.c)。

-l 生成客户机stubs。(file_clnt.c

-m 生成服务器stubs,但是不生成main函数。(file_svc.c

-s  rpcgen –C –s tcp file.x,生成服务器stubs,用tcp协议,同时生成了main函数。(file_svc.c

              -h 生成头文件。

              -Sc 生成骨架客户机程序,(file_client.c),生成后还需要手动添加代码。

              -Ss 生成服务器程序,(file_server.c),生成后还需要手动添加代码。

Rpcgen –C  file.x 生成file_xdr.c,file.h,Makefile.file,file_svc.cfile_client.c

Rpcgen –C –a file.x 比上面多生成了2个文件,file_server.cfile_client.c

 

Rpcgen示例程序

规格文件(math.x

/* filename: math.x */

const ADD = 0;

const SUB = 1;

const MUL = 2;

const DIV = 3;

struct MATH

{

    int op; /* 0-ADD, 1-SUB, 2-MUL, 3-DIV */

    float arg1;

    float arg2;

    float result;

};

program MATH_PROG

{

    version MATH_VER

    {

        struct MATH MATH_PROC(struct MATH) = 1;

    } = 2;

} = 0x20000001;

 

rpcgen –C –a math.x 生成7个文件,math.hmath_xdr.cmath_svc.cmath_clnt.cMakefile.mathmath_client.cmath_server.c

math_client.c中添加代码,下面是添加后的代码:

/*

 * This is sample code generated by rpcgen.

 * These are only templates and you can use them

 * as a guideline for developing your own functions.

 */

#include "math.h"

void math_prog_2(char *host)

{

              CLIENT *clnt;

              struct MATH  *result_1;

              struct MATH  math_proc_2_arg;

    /* 2006/07/04 Dongyy Add -> */

    char c;

    printf("choose the operation:/n/t0---ADD/n/t1---SUB/n/t2---MUL/n/t3---DIV/n");

    c = getchar();

    switch(c) {

        case '0':

            math_proc_2_arg.op = ADD;

            break;

        case '1':

            math_proc_2_arg.op = SUB;

            break;

        case '2':

            math_proc_2_arg.op = MUL;

            break;

        case '3':

            math_proc_2_arg.op = DIV;

            break;

        default:

            printf("error:operate/n");

            exit(1);

    }

    printf("input the first number:");

    scanf("%f", &math_proc_2_arg.arg1);

    printf("input the second number:");

    scanf("%f", &math_proc_2_arg.arg2);

    /* <- 2006/07/04 Dongyy Add */

 

#ifndef   DEBUG

              clnt = clnt_create (host, MATH_PROG, MATH_VER, "udp");

              if (clnt == NULL) {

                             clnt_pcreateerror (host);

                             exit (1);

              }

#endif    /* DEBUG */

 

              result_1 = math_proc_2(&math_proc_2_arg, clnt);

              if (result_1 == (struct MATH *) NULL) {

                             clnt_perror (clnt, "call failed");

              }

#ifndef   DEBUG

              clnt_destroy (clnt);

#endif    /* DEBUG */

    /* 2006/07/04 Dongyy Add -> */

    printf("The Result is % .3f /n", result_1->result);

    /* <- 2006/07/04 Dongyy Add */

}

int main (int argc, char *argv[])

{

              char *host;

 

              if (argc < 2) {

                             printf ("usage: %s server_host/n", argv[0]);

                             exit (1);

              }

              host = argv[1];

              math_prog_2 (host);

    exit (0);

}

math_server.c中添加代码,下面是添加后的代码:

/*

 * This is sample code generated by rpcgen.

 * These are only templates and you can use them

 * as a guideline for developing your own functions.

 */

 

#include "math.h"

 

struct MATH *

math_proc_2_svc(struct MATH *argp, struct svc_req *rqstp)

{

              static struct MATH  result;

 

              /*

               * insert server code here

               */

   

    /* 2006/07/04 Dongyy Add -> */

    switch(argp->op){

        case ADD:

            result.result = argp->arg1 + argp->arg2;

            break;

        case SUB:

            result.result = argp->arg1 - argp->arg2;

            break;

        case MUL:

            result.result = argp->arg1 * argp->arg2;

            break;

        case DIV:

            result.result = argp->arg1 / argp->arg2;

            break;

        default:

            break;

    }

    /* <- 2006/07/04 Dongyy Add */

 

              return &result;

}

红色字体部分为用rpcgen生成代码后,手动添加的代码。

添加完后,执行make –f makefile.math 编译生成math_clientmath_server,在命令行运行math_server &,然后运行math_client 127.0.0.1,按照提示输入内容就OK了。

 

参考书籍《Linux C 高级程序员指南》

 

Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=88991


第二种:在服务器端注册,在客户端调用:
#include   <stdio.h> 
#include   <rpc/rpc.h> 
#include   <utmp.h> 
#include   "rpc.h" 
unsigned   int*   nuser(unsigned   int*   ui) 
  { 
                  printf("haha,your   input   value   =   %d/n",*ui); 
                  *ui=*ui+1; 
                  return   ui; 
  };  
main() 
  { 
          registerrpc(DEMOPROG,   DEMOVER,   DEMOFUNC, 
                  nuser,   xdr_u_int,   xdr_u_int); 
          svc_run();                 /*   Never   returns   */ 
          fprintf(stderr,   "Error:   svc_run   returned!/n"); 
          exit(1); 
  }  

#include   <stdio.h> 
#include   <rpc/rpc.h> 
#include   <utmp.h> 
#include   "rpc.h" 
  
main(argc,   argv) 
          int   argc; 
          char   **argv; 
  { 
          unsigned   long   nusers=0; 
          int   stat=0; 
          unsigned   int   val=0; 
          if   (argc   !=   3)   { 
                  fprintf(stderr,   "usage:   nusers   hostname/n"); 
                  exit(-1); 
          } 
          val   =   atoi(argv[2]); 
          printf("val   =   %d/n",val); 
          if   (stat   =   callrpc(argv[1], 
                  DEMOPROG,   DEMOVER,   DEMOFUNC, 
                  xdr_u_int,&val,   xdr_u_int,   &nusers)   !=   0)   { 
                          clnt_perrno(stat); 
                          exit(1); 
          } 
          printf("%d   users   on   %s/n",   nusers,   argv[1]); 
          exit(0); 
  }  

//rpc.h
#define   DEMOPROG                 0x20000001 
#define   DEMOVER                   0x1 
#define   DEMOFUNC                 0x1  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值