服务器端

/**************************************************************************************************************************************
*
* file name:Mp3downloadSer.c
* author :momolyl@126.com
* date :2024/06/04
* brief :2人一组,实现C/S架构,有一台服务器,服务器中存储n首音频,
* 要求客户端可以直接下载服务器的音频,并且可以正常在客户端播放。
* 提示:提示stat 获取音频 xxx.mp3的属性
* note :客户端需向服务器发送“xxx”获取音频,客户端接收到“over”表示传输完毕。
* 该进程端口号为 9999;宏CLIENT_IP 为客户端的IP,需根据实际情况自行修改
*
* CopyRight (c) 2024 momolyl@126.com All Right Reseverd
* ***********************************************************************************************************************************/
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <errno.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/udp.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>
#include <sys/types.h>
#include <time.h>
#include <stdlib.h>
#include <stdio.h>
#include <strings.h>
#include <unistd.h>
#include <time.h>
#include <signal.h>
#include <sys/stat.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define CLIENT_PORT "9999"
#define CLIENT_IP "192.168.64.233"
pthread_t pthreadsend;
pthread_t pthreadrecv;
int flag = 0;
// 接收消息线程:等待接收客户端的消息
void *pthreadrecv_task(void *argv)
{ // 1.创建UDP套接字
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 2.需要先绑定服务器的端口和地址
struct sockaddr_in host_addr; host_addr.sin_family = AF_INET; // 协议族,是固定的
host_addr.sin_port = htons(atoi("9999")); // 目标端口,必须转换为网络字节序
host_addr.sin_addr.s_addr = htonl(INADDR_ANY); // 目标地址 "192.168.64.xxx" 已经转换为网络字节序 INADDR_ANY bind(udp_socket, (struct sockaddr *)&host_addr, sizeof(host_addr));
// 3.调用recvfrom等待接收数据,并且接收客户端的网络信息
char buf[128] = {0};
struct sockaddr_in client; socklen_t client_len = sizeof(client);
char music[8] = "xxx"; while (1)
{ recvfrom(udp_socket, buf, sizeof(buf), 0, (struct sockaddr *)&client, &client_len); // 默认会阻塞
printf("消息来源:[%s] 消息内容:[%s]\n", inet_ntoa(client.sin_addr), buf);
// 当接收到的内容为“xxx”时,向该IP发送音频文件 if (!strcmp(buf, music))
{
flag = 1;
} bzero(buf, sizeof(buf));
}
} int main(int argc, char const *argv[])
{
// 1.创建UDP套接字
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
} // 创建线程等待接受客户端的信息
// pthread_t pthreadrecv;
int result = pthread_create(&pthreadrecv, NULL, pthreadrecv_task, NULL);
if (result)
{
fprintf(stderr, "create thread error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
} // int size = 0;
while (1)
{
if (flag)
{
printf("I am coming!\n");
// 1.创建UDP套接字
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
} // 2.向目标主机发送消息,需要设置目标端口和目标地址
char buf[1024] = {0}; struct sockaddr_in dest_addr;
dest_addr.sin_family = AF_INET; // 协议族,是固定的
dest_addr.sin_port = htons(atoi(CLIENT_PORT)); // 服务器端口,必须转换为网络字节序
dest_addr.sin_addr.s_addr = inet_addr(CLIENT_IP); // 服务器地址 "192.168.64.xxx"
FILE *fd = fopen("./xxx.mp3", "rb"); // 打开文件 fseek(fd, 0, SEEK_SET);
struct stat *statbuf;
stat("./xxx.mp3", statbuf);
int count = statbuf->st_size / 1024 + 1;
printf("loop num is %d\n", count); while (count--)
{ // 从音频文件中读取内容
// read(fd, buf, 1024);
fread(buf, 1, 1024, fd); // 向客户端发送本次读取到的内容
sendto(udp_socket, buf, sizeof(buf), 0, (struct sockaddr *)&dest_addr, sizeof(dest_addr));
// printf("size of buf is %d", size);
bzero(buf, 1024);
// printf("I am working!\n");
usleep(1000 * 1000 / 10000);
}
printf("over!over!\n");
// 通知客户端文件发送完毕
sprintf(buf, "over");
printf("the buf is %s\n", buf); // 向客户端发送本次读取到的内容
sendto(udp_socket, buf, strlen(buf), 0, (struct sockaddr *)&dest_addr, sizeof(dest_addr));
bzero(buf, sizeof(buf));
flag = 0;
}
} return 0;
}

客户端

/*************************************************************************************************************************
*
* file name: udp_cs.c
* author : Dazz
* date : 2024/6/4
* function : 实现C/S架构,有一台服务器,服务器中存储n首音频,要求客户端可以直接下载服务器的音频,并且可以正常在客户端播放。
* 用stat 获取音频 xxx.mp3的属性
*
* note : ./a.out port addr
*
* CopyRight (c) 2024 Dazz_24@163.com All Right Reseverd
*
* **************************************************************************************************************************/
#include <stdio.h>
#include <errno.h>
#include <arpa/inet.h>
#include <netinet/udp.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <sys/stat.h>
#include <sys/types.h>
#define DESTPORT 9999 // udp_socket; void *DownloadMusic(void *argv)
{
// 创建一个用来存放音乐的目录
int dir = mkdir("./music", 0664);
if (-1 == dir && (!(errno == EEXIST)))
{
// 返回值为-1且文件不存在则退出
fprintf(stderr, "mkdir fail ,errno:%d,%s\n", errno, strerror(errno));
exit(1);
} // 给服务器发送下载音乐的请求
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(2);
} // 需要先绑定主机的端口和地址
struct sockaddr_in host_addr; host_addr.sin_family = AF_INET; // 协议族,是固定的
host_addr.sin_port = htons(DESTPORT); // 目标端口,必须转换为网络字节序
host_addr.sin_addr.s_addr = htonl(INADDR_ANY); // 目标地址 "192.168.64.xxx" 已经转换为网络字节序 bind(udp_socket, (struct sockaddr *)&host_addr, sizeof(host_addr));
FILE *music; // 创建音乐文件
music = fopen("./music/666.mp3", "wb+");
if (NULL == music)
{
fprintf(stderr, "fopen file fail,errno:%d,%s\n", errno, strerror(errno));
exit(1);
} // 循环调用recvfrom准备接收数据
char buf[1024] = {0};
int size; //
int count; // 记录需要接收多少次
while (1)
{
// 接收服务器传输的数据,一次接收1024字节
size = recvfrom(udp_socket, buf, 1024, 0, NULL, NULL);
// 打印接收了多少次数据
printf(" has received %d times!!! \n", count++); // 如收到服务器传来的“over”,则表示音乐传输完毕
if (!strcmp(buf, "over")) // 这里使用strstr()函数也可以
{
printf("\n\n\n");
printf("***************************************\n");
printf("The music is downloaded!!!!\n");
printf("***************************************\n");
break;
}
printf("recv data size = %d\n", size); // 向创建的音乐文件里写入接收的数据
fwrite(buf, 1, size, music); // 清空缓存区
bzero(buf, sizeof(buf));
} // 关闭文件
fclose(music);
close(udp_socket);
} int main(int argc, char *argv[])
{
// 检查参数有效性
if (argc != 3)
{
fprintf(stderr, "argument is invaild ,errno:%d,%s\n", errno, strerror(errno));
exit(1);
} // 给服务器发送下载音乐的请求
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(2);
} // 创建线程来接收服务器发送的音乐
pthread_t thread;
pthread_create(&thread, NULL, DownloadMusic, NULL); // 向目标主机发送消息,需要设置目标端口和目标地址
char sendbuf[128] = "xxx"; struct sockaddr_in dest_addr;
dest_addr.sin_family = AF_INET; // 协议族,是固定的
dest_addr.sin_port = htons(atoi(argv[1])); // 服务器端口,必须转换为网络字节序
dest_addr.sin_addr.s_addr = inet_addr(argv[2]); // 服务器地址 "192.168.64.xxx" // 向服务器发送请求下载音乐的请求
int xxx = sendto(udp_socket, sendbuf, strlen(sendbuf), 0, (struct sockaddr *)&dest_addr, sizeof(dest_addr)); // 等待子线程结束
pthread_join(thread, NULL); return 0;
}

UDP协议实现音频传输的更多相关文章

  1. Python中的端口协议之基于UDP协议的通信传输

    UDP协议: 1.python中基于udp协议的客户端与服务端通信简单过程实现 2.udp协议的一些特点(与tcp协议的比较)        3.利用socketserver模块实现udp传输协议的并 ...

  2. tsunami:一种基于UDP协议的高速传输

    一. 需求 近期在做数据库迁移.常常须要打包实例传输.传统scp感觉非常慢. 二. 软件信息 1. 软件主页:http://tsunami-udp.sf.net/ 2. 软件安装:直接源代码make ...

  3. tsunami:一种基于UDP协议的快速传输

    一. 需求 最近在做数据库迁移,经常需要打包实例传输,传统scp感觉很慢. 二. 软件信息 1. 软件主页:http://tsunami-udp.sf.net/ 2. 软件安装:直接源码make &a ...

  4. 理解UDP协议的首部校验和校验和

    reference: https://blog.csdn.net/qiuchangyong/article/details/79945630 https://seanwangjs.github.io/ ...

  5. UDP协议详解

    1.UDP协议的作用 IP协议无法区别同一个主机系统上的多个应用程序.UDP采用端口标识同一主机上的不同应用程序. 无法采取进程ID来标识不同应用程序的原因: 1)系统中应用程序的进程ID分配和销毁是 ...

  6. TCP和UDP协议的区别

    TCP和UDP都是传输层的协议 UDP协议的特点: UDP协议是一种无连接的.不可靠的传输层协议(尽力而为的协议) 为什么说UDP是一种无连接.不可靠的协议呢?UDP协议在传输报文之前不需要在双方之间 ...

  7. TCP/IP/UDP 协议

    互连网早期的时候,主机间的互连使用的是NCP协议.这种协议本身有很多缺陷,如:不能互连不同的主机,不能互连不同的操作系统,没有纠错功能.为了改善这种缺点,大牛弄出了TCP/IP协议.现在几乎所有的操作 ...

  8. TCP/IP协议网络编程以及UDP和TCP之传输协议

    1.什么是TCP/IP协议? 网络编程协议有很多,目前应用最广泛的是TCP/IP协议(Transmission Control Protocal/Internet Protoal 传输控制协议/英特网 ...

  9. 数据通讯与网络 第五版第24章 传输层协议-UDP协议部分要点

    24.1 介绍 本章节主要集中于传输层协议的解读,图24.1展示TCP.UDP.SCTP在TCP\IP协议栈的位置 24.1.1 服务(Service) 每个协议都提供不同的服务,所以应该合理正确的使 ...

  10. Linux内核--网络栈实现分析(九)--传输层之UDP协议(下)

    本文分析基于Linux Kernel 1.2.13 原创作品,转载请标明http://blog.csdn.net/yming0221/article/details/7549340 更多请查看专栏,地 ...

随机推荐

  1. es6.6.1 java客户端 client基础操作

    1.引入jar包 <dependency> <groupId>org.elasticsearch.client</groupId> <artifactId&g ...

  2. 高德的API来查询行政区域查询

    高德的API来查询行政区域查询 1.api接口文档地址 https://lbs.amap.com/api/webservice/guide/api/district GET https://resta ...

  3. 5分钟理透LangChain的Chain

    LangChain几乎是LLM应用开发的第一选择,它的野心也比较大,它致力于将自己打造成LLM应用开发的最大社区.而LangChain最核心的部分非 Chain 莫属. 那Chain到底是个啥,概念比 ...

  4. 使用POST方法向网站发送数据

    POST方法向网站发送数据 server.py import flask app = flask.Flask(__name__) @app.route('/', methods=['GET','POS ...

  5. JavaScript:JS对象_Array

    <!DOCTYPE html><html>    <head>        <meta charset="utf-8">      ...

  6. 《Objective-C Direct Methods》学习笔记

    原文通过对Objective-C发展史.Objective-C中Runtime的动态派发,C语言的直接派发进行铺垫介绍,引出了direct methods这个"新特性"(文章写于2 ...

  7. 用基础Array数组实现动态数组、链表、栈和队列

    代码地址: https://gitee.com/Tom-shushu/Algorithm-and-Data-Structure.git 一.ArrayList自定义封装 package com.zho ...

  8. P9212 题解

    显然,我们维护的答案具有 可差分 性,所以转换为 \([1,r]\) 上的查询. 首先,对于 \(x,y,a_i\) 先对 \(m\) 取模不影响结果. 下面为了方便令 \(v = a_i\). 如果 ...

  9. Restful和WebService区别

    简介 Restful是一种架构风格,其核心是面向资源,更简单: 而webService底层SOAP协议,主要核心是面向活动: 两个都是通过web请求调用接口 RESTful是什么 REST就是(REp ...

  10. SpringBoot配置Jackson处理字段

    常用框架 阿里fastjson,谷歌gson等 JavaBean序列化为json 性能:Jackson>FastJson>Gson>lib 同个结构 Jackson.Fastjson ...