找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 3091|回復: 0
打印 上一主題 下一主題
收起左側(cè)

基于GCC_TCPIP Socket—服務端文件傳輸

[復制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:51090 發(fā)表于 2014-9-18 12:54 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <errno.h>
#include <signal.h>


#define IPLEN 40
#define  NUM 1024
#define  RCVPORT 1990
#define  NAME 45
#define  EOF  (-1)

int my_open(int sd,char *p);
int my_rdwr(int fd,int stat_st_size,int sd);


int main(int argc,char * argv[])
{
        int fd;
           int i=1;
        int sockedfd;
        int newsockedfd;
        int stop = EOF;
        char buf[20];
        size_t  len;
        struct sockaddr_in local_addr;
        struct sockaddr_in peer_addr;
        socklen_t addr_len;
        char ipstr[IPLEN];
        pid_t fpid;
       
        sockedfd = socket(PF_INET, SOCK_STREAM, 0);
        if(sockedfd< 0){
                perror("socket()");
                exit(1);
        }
       
        setsockopt(sockedfd,SOL_SOCKET,SO_REUSEADDR,&i,sizeof(i) );
       
        local_addr.sin_family = PF_INET;//ipv4
        local_addr.sin_port = htons(RCVPORT);
        inet_pton(PF_INET, "0.0.0.0", &local_addr.sin_addr);//邦定端口
        if(bind(sockedfd, (struct sockaddr *)&local_addr, sizeof(local_addr)) < 0){
                perror("bind()");
                exit(1);
        }
        if(listen(sockedfd, 200) < 0){
                perror("listen()");
                exit(1);
        }
        while(1)
        {
                addr_len = sizeof(peer_addr);
                newsockedfd = accept(sockedfd,(void*)&peer_addr, &addr_len);
                if(newsockedfd < 0)
                {
                        perror("accept()");
                        exit(1);
                }
                i=1;       
                char n='0';
#if 1
                fpid=fork();
                if(0 ==fpid )
                {
                        close(sockedfd);//關閉父進程監(jiān)聽描述符
#endif
                        while(1)
                        {
                                len = read(newsockedfd,buf,1);
                                if(0 > len )
                                {
                                        perror("read()");
                                        return -1;
                                }
                                if( 0 ==len )
                                        break;               

                                if ( '0'==n )
                                {
                                        n=buf[0];
                                        printf("n=%c, i=%d \n",n,i);
                                        if(n<'4'&&n>='1')
                                        {
                                                snprintf(buf,sizeof(buf),"music/%c.wma",n);
                                                printf("%s \n",buf);
                                                my_fopen_rdwr(newsockedfd,buf);
                                        }
                                        else
                                        {
                                                printf("ERROR()\n");
                                                len=write(newsockedfd,&stop,4);
                                                if(len<0)
                                                {
                                                        perror("write()");
                                                        printf("writr end fill");
                                                }
                                                n='0';
                                                break;
                                        }
                                }
                                else
                                {
                                        if( NAME == i )
                                        {//圖片張數(shù)
                                                len=write(newsockedfd,&stop,4);//結(jié)束寫19
                                                if(len<0)
                                                {
                                                        perror("write()");
                                                        printf("writr end fill");
                                                }
                                                break;
                                        }
                                        snprintf(buf,sizeof(buf),"jpg%c/%d.jpg",n,i++);
                                        printf("%s \n",buf);
                                        my_fopen_rdwr(newsockedfd,buf);
                                }
                        }
                        close(newsockedfd);
                        exit(0);
                       
                }
                else if (0 < fpid)
                {
                        close(newsockedfd);//父進程關閉新套接字
                        signal(SIGCLD, SIG_IGN);//通知內(nèi)核回收進程,該進程不是父進程回收
                }
                else
                {
                        printf("this is fork error\n");
                }
                //printf("*******%s %s %d****pid=%d****\n",__FILE__,__FUNCTION__,__LINE__,fpid);
        }
        close(sockedfd);
        return 0;
}

int my_fopen_rdwr(int sd,char *p)
{
        struct stat stat_init;
        size_t ret;
        int pos,pos1;
        char buf[NUM];
        int j=0;
        FILE * fd;
        fd=fopen(p,"r");
        if(fd<0)
        {
                perror("fopen()");
                return -1;
        }
        if(stat(p,&stat_init)<0)
        {
                perror("stat()");
                return -1;
        }
#if 1
        ret=write(sd,&stat_init.st_size,4);
        pos1+=ret;
        if(ret<0)
        {
                perror("write()");
                return -1;
        }
#endif
        while(stat_init.st_size>0)
        {
                pos=0;
                pos1=0;
                ret=fread(buf+pos,sizeof(char),NUM,fd);
                pos+=ret;
                if(ret<0){
                        perror("fread()");
                        return -1;
                }
                ret=write(sd,buf+pos1,ret);
                pos1+=ret;
                if(ret<0){
                        perror("write()");
                        return -1;
                }
                stat_init.st_size-=ret;
        }
        printf("is over*******\n");
        fclose(fd);
        return 0;
}

本服務端代碼測試通過 ,cpu占用率低,該服務端是采用多進程方式通訊,有客戶端連接到服務端時,父進程會fork個子進程,子進程來連接客戶,父進程繼續(xù)監(jiān)聽端口,子進程下載完畢后自動退出,這樣做的目的是在沒有客戶連接時,只有父進程一個處于監(jiān)聽狀態(tài),節(jié)約內(nèi)存,提高性能,要是有多個客戶連接,父進程只管fork子進程,節(jié)約了時間,同時可以多個客戶端下載,本代碼出錯處理不夠完善,后期繼續(xù)跟新,如有建議和本人寫的不足之處敬請給位多多指教。

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網(wǎng)

快速回復 返回頂部 返回列表