c-使用sun-rpc将结构数组从服务器发送到客户端



如何在ansi-c sun-rpc中正确地将结构从服务器发送到客户端?

test.x IDL文件中,我定义了一个带有字符串和int的结构集群以及作为簇元素的可变长度阵列的类型簇:

struct cluster {
    string name<255>;
    int debuglevel;
};
typedef cluster clusters<32>;

然后,我像一样更改了rpcgen生成的存根

testrongerver.c

clusters *
test_1_svc(void *argp, struct svc_req *rqstp)
{
    static clusters result;
    cluster cl1, cl2;
    cl1.name="cl1";
    cl1.debuglevel="1";
    cl2.name="cl2";
    cl2.debuglevel="2";
    cluster clist[2];
    clist[0]=cl1;
    clist[1]=cl2;
    result.clusters_len = 2;
    result.clusters_val = &clist;
    /*
     * insert server code here
     */
    return(&result);
}

test_client.c

test_prog_1( char* host )
{
    CLIENT *clnt;
    clusters  *result_1;
    char*  test_1_arg;
    clnt = clnt_create(host, test_PROG, test_VERS, "udp");
    if (clnt == NULL) {
        clnt_pcreateerror(host);
        exit(1);
    }
    result_1 = test_1((void*)&test_1_arg, clnt);
    if (result_1 == NULL) {
        clusters* rec_cls = malloc(2*sizeof(struct cluster));
        if(xdr_clusters(&result_1, rec_cls)){
                printf("got xdr_clusters");
        }
        clnt_perror(clnt, "call failed:");
    }
    clnt_destroy( clnt );
}

两者都是编译的,但服务器通常在客户端运行一两个请求后发生segfault,并且在客户端,xdr_clusters函数永远不会返回true。这似乎是某种内存管理不善,我也不确定我是否正确地处理了服务器端的序列化。

我刚刚用合适的值填充了result.clusters_len和result.clusters _val,就像它们在测试中定义的那样。h(由rpcgen):

typedef struct {
    u_int clusters_len;
    cluster *clusters_val;
} clusters;

我是否必须使用服务器端的xdr_clusters来正确序列化结果?

感谢

好吧,我发现了我的错误,让我们总结一下:

  • 知道如何正确初始化int(当然不带引号…)
  • 忘记clist的废话,只需直接对结果结构的内部指针进行malloc即可
  • 阅读该死的编译器警告:当它告诉你,有隐式声明的函数,而你不想要隐式声明时,可能缺少一些东西,在我的情况下,我需要包括stdlib.h和stdio.h,以获得服务器和客户端存根的malloc、printf和exit函数
  • 在客户端:如果结果为NULL,除了抛出错误之外,我们为什么要做任何事情?请参阅下面的新客户端代码以检查打印结果是否正确

testrongerver.c

    test_1_svc(void *argp, struct svc_req *rqstp){
        static clusters result;
        cluster cl1, cl2;
        cl1.name="cl1";
        cl1.debuglevel=1;
        cl2.name="cl2";
        cl2.debuglevel=2;
        result.clusters_len = 2;
        result.clusters_val = malloc(2*sizeof(struct cluster));
        result.clusters_val[0]=cl1;
        result.clusters_val[1]=cl2;
        return(&result);
    }

test_client.c

test_prog_1( char* host )
{
    CLIENT *clnt;
    clusters  *result_1;
    char*  test_1_arg;
    clnt = clnt_create(host, test_PROG, test_VERS, "udp");
    if (clnt == NULL) {
        clnt_pcreateerror(host);
        exit(1);
    }
    result_1 = test_1((void*)&test_1_arg, clnt);
    if (result_1 == NULL) {
        clnt_perror(clnt, "call failed:");
    }else{
        printf("I got %d cluster structs in an arrayn",result_1->clusters_len);
        int j;
        for(j=0;j<result_1->clusters_len;j++){
            printf("cluster #%d: %s@runlevel %dn",j,result_1->clusters_val[j].name,result_1->clusters_val[j].debuglevel);
        }
    }
    clnt_destroy( clnt );
}

因此,我们在客户端打印出了一些不错的值当然,服务器端不再有分段故障:

lars$ ./test_client localhost
I got 2 cluster structs in an array
cluster #0: cl1@runlevel 1
cluster #1: cl2@runlevel 2

相关内容

  • 没有找到相关文章

最新更新