首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >将带有动态数组的结构复制到c++中的缓冲区中。

将带有动态数组的结构复制到c++中的缓冲区中。
EN

Stack Overflow用户
提问于 2013-04-26 23:54:19
回答 2查看 1.4K关注 0票数 1

我想使用套接字API手动构建到dns服务器的dns查询(它是项目的一部分,不能更改)。所以我想复制一个结构,如下所示:

代码语言:javascript
复制
typedef struct {         
    uint16_t dns_id;  /* identification number */       
    uint16_t dns_flags; /* DNS flags */        
    uint16_t dns_qdc; /* number of question entries */      
    uint16_t dns_anc; /* number of answer entries */  
    uint16_t dns_nsc; /* number of authority entries */      
    uint16_t dns_arc; /* number of resource entries */
    unsigned char *host;
    unsigned short qtype;
    unsigned short qclass;     
} DNS_QUERY;


DNS_QUERY* dns_query = (DNS_QUERY *) malloc(sizeof(DNS_QUERY)); 

放入缓冲区中,如下所示:

代码语言:javascript
复制
char* dns_buf = malloc(500);

但是当我这样做的时候,没有任何东西被复制到缓冲区中。当我向*主机元素输入不同的字符串时,sizeof(dns_query)保持不变。当我使用

代码语言:javascript
复制
memcpy(dns_buf, dns_query, sizeof(DNS_QUERY));

它给出的输出如下:

dns_buf:

字符串(Dns_buf)=0

另外,当我使用for循环打印时,如下所示:

代码语言:javascript
复制
for (i = 0 ; i<strlen(dns_buf) ; i++){
        printf("%c", dns_buf[i]);   
    }

它不会输出任何内容

当我使用指针而不是memcpy时,比如:

dns_buf = dns_query;

它给出了相同的输出。有人能告诉我该怎么做吗?

下面是完整的代码:

代码语言:javascript
复制
int udp_connect(char *hostname, char *dns_name, char buf[]){

    int sockfd;
    int num_bytes;
    struct sockaddr_in servaddr;
    char str[INET_ADDRSTRLEN];
    struct hostent *hptr; /* For gethostbyname() */
    char **pptr; /* For inet_ntop() */
    char* dns_buf, recv_buf = NULL; 
    int len = 0;
    int i = 0;
    static unsigned short id = 0; /* For the query ID */

    /* Allocating memory for structs */
    DNS_QUERY* dns_query = malloc(sizeof(DNS_QUERY)); 
    /* QUESTION *dns_question = malloc(sizeof(QUESTION));*/
    recv_buf = malloc(MAXDATASIZE);

    if((hptr = gethostbyname(dns_name)) == NULL){
        perror("Error in gethostbyname()\n");
        exit(1);
    }

    if ((pptr = hptr->h_addr_list) != NULL) { /* (hptr->h_addrtype == AF_INET) && */
        printf("The address is: %s\n", inet_ntop(hptr->h_addrtype, *pptr, str, sizeof(str)));
    } else {
        perror("Error in inet_ntop() \n");
    }

    memset(&servaddr, 0, sizeof (servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(DNS_PORT);
    inet_pton(AF_INET, str, &servaddr.sin_addr);

    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) == -1){
        perror("Error while making a socket");
        return -1;
    }   

    /* Setting hostname into the name segment of the query */
    ChangetoDnsNameFormat(hostname);
    dns_query->host = hostname;
    printf("\ndns_query->host: %s\n", dns_query->host);


    /* memcpy(dns_buf, dns_query, sizeof(DNS_QUERY)); */    

    /* setting up the header */
    dns_query->dns_id = htons(id); /* id */
    dns_query->dns_flags = htons(0x10); /* 0000000100000000 recursion is desired*/  
    dns_query->dns_qdc = htons(0x01); /* We have one question */
    dns_query->dns_anc = htons(0x00);
    dns_query->dns_nsc = htons(0x00);
    dns_query->dns_arc = htons(0x00);

    dns_query->qtype = htons(1); /* For IPv4 */
    dns_query->qclass = htons(1); /* For internet */

    len = sizeof(DNS_QUERY) + strlen(hostname); /* Calculating the length to use it in sendto() */

    dns_buf = malloc(len);

    dns_buf = dns_query;
    /* memcpy(dns_buf, dns_query, sizeof(DNS_QUERY)); */

    for (i = 0 ; i<sizeof(DNS_QUERY) ; i++){
            printf("%02X", dns_buf[i]);   
    }

    /* to check if the same as dns_query, use the following print also */
    for (i = 0 ; i<sizeof(DNS_QUERY) ; i++){
            printf("%02X", dns_query[i]);   
        }
    print("\n");

    /* Sending the datagram to the dns server */
    printf("\n--------------------\nSending datagram: \n%s\n", dns_buf);
    if ((num_bytes = sendto(sockfd, dns_buf, len, 0, (struct sockaddr *) &servaddr, sizeof(servaddr))) == -1){
        perror("Error in sendto");
        exit(1);
    }

    /* Receiving the datagram from the dns server */
    printf("Receiving datagram...\n");
    num_bytes = recvfrom(sockfd, recv_buf, MAXDATASIZE, 0, NULL, NULL);
    printf("Received %i bytes of datagram...\n", num_bytes);

    printf("dns_buf in udp connect: %s\n", recv_buf);

    id = id + 1;

    free(dns_query);
    freeaddrinfo((struct addrinfo *) &servaddr);
    close(sockfd);

    return sockfd;
}
/******************************************************************************/
void ChangetoDnsNameFormat(char *hostname)
{
    int walker=0;
    int i;
    int counter = 0;
    char tmp[40];

    strcat(hostname, "$"); /* For having a $ at the end of the dns name format */   

    for(i=0 ; i< (int) strlen((char*)hostname) ; i++)
    {
        if(hostname[i]=='.')
        {
            tmp[walker] = (char) counter + 48;
            /*printf("%s\n", tmp);*/

            for( ; walker < i ; walker++)
            {
                tmp[walker + 1] = hostname[walker];
            }

            walker++;
            counter = -1;
        }else if(hostname[i]=='$'){

            tmp[walker] = (char) counter + 47;
            /*printf("%s\n", tmp);*/

            for( ; walker < i ; walker++)
            {
                tmp[walker + 1] = hostname[walker];
            }

            walker++;
            counter = -1;
        }
        counter++;
    }
    walker--;
    tmp[walker] = '\0'; /* Terminate the string */

    strcat(tmp, "0"); /* For having a 0 at the end of the dns name format */

    strcpy(hostname, tmp);
}
EN

回答 2

Stack Overflow用户

发布于 2013-04-26 23:57:32

dns_buf不是字符串(以空字符结尾)。

它是一个指向内存空间的指针(作为char *),内存可以在开头包含0作为值,这就是为什么使用strlen(dns_buf)时会得到= to 0的原因,这意味着dns_buf中的0元素是0

要查看可以使用的dns_buf内容,请执行以下操作:

代码语言:javascript
复制
for (i = 0 ; i<sizeof(DNS_QUERY) ; i++){
        printf("%02X", dns_buf[i]);   
    }
print("\n");

// to check if the same as dns_query, use the following print also
for (i = 0 ; i<sizeof(DNS_QUERY) ; i++){
        printf("%02X", dns_query[i]);   
    }
print("\n");
票数 1
EN

Stack Overflow用户

发布于 2013-04-27 00:00:01

该结构实际上并不包含字符host数组,而是一个指向第一个元素的指针。

这意味着,尽管您正在复制整个结构,但您只复制了指向host中包含的数据的指针(然后指向相同的字符串)。你应该做的是:

代码语言:javascript
复制
DNS_QUERY *dns_buf = malloc(sizeof(DNS_QUERY));
memcpy(dns_buf, dns_query, sizeof(DNS_QUERY));
dns_buf.host = "yourmodifiedhost";
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/16240534

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档