高级UIKit-09(TCPSocket发送文件、上传和下载)
【day1101_SocketSendFile】:发送文件到服务端
发送文件需要在该文件上拼接消息头,比如类型,文件名,文件大小
// 服务端
- (void)viewDidLoad
{
[super viewDidLoad];
// 创建服务器
self.socketServer = [[AsyncSocket alloc] initWithDelegate:self];
// 设置端口
[self.socketServer acceptOnPort: error:Nil];
}
// 接收连接
- (void)onSocket:(AsyncSocket *)sock didAcceptNewSocket:(AsyncSocket *)newSocket{
self.socketNew = newSocket;
}
// 连接成功
- (void)onSocket:(AsyncSocket *)sock didConnectToHost:(NSString *)host port:(UInt16)port{
[self.socketNew readDataWithTimeout:- tag:];// 读取数据
}
// 读取数据
- (void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag{
// 判断是否是头
NSData *headerData = [data subdataWithRange:NSMakeRange(, )];
NSString *headerString = [[NSString alloc] initWithData:headerData encoding:NSUTF8StringEncoding];
if (headerString && [headerString componentsSeparatedByString:@"&&"].count == ) {
NSArray *fileArray = [headerString componentsSeparatedByString:@"&&"];
NSString *type = fileArray[];
if ([type isEqualToString:@"file"]) { // 如果是文件
self.fileData = [NSMutableData data];
self.fileName = fileArray[];
self.fileLength = [fileArray[] intValue];
NSData *subData = [data subdataWithRange:NSMakeRange(, data.length - )];
[self.fileData appendData:subData];
}else{
}
}else{
[self.fileData appendData:data];
}
NSLog(@"%d,%d",data.length,self.fileData.length);
// 判断是否接收完成
if (self.fileData.length == self.fileLength) {
NSString *filePath = [@"/Users/tarena/yz/第三阶段(高级UI)/day11" stringByAppendingPathComponent:self.fileName];
[self.fileData writeToFile:filePath atomically:YES];
}
[self.socketNew readDataWithTimeout:- tag:];// 读取数据
}
//客户端
- (IBAction)clicked:(id)sender {
// 创建客户端
self.socketClient = [[AsyncSocket alloc] initWithDelegate:self];
// 连接服务器
if ([self.socketClient connectToHost:@"localhost" onPort: error:Nil]) {
NSLog(@"连接成功!");
}
if ([self.sendTextField.text hasPrefix:@"/"]) { // 如果是文件
// 得到文件路径
NSString *filePath = self.sendTextField.text;
// 转成data
NSData *fileData = [NSData dataWithContentsOfFile:filePath];
// 拼接信息头
NSString *headerString = [NSString stringWithFormat:@"file&&%@&&%d",[filePath lastPathComponent],fileData.length];
// 转成data
NSData *headerData = [headerString dataUsingEncoding:NSUTF8StringEncoding];
// 发送数据
NSMutableData *allData = [NSMutableData dataWithLength:];
[allData replaceBytesInRange:NSMakeRange(, headerData.length) withBytes:headerData.bytes];
[allData appendData:fileData];
[self.socketClient writeData:allData withTimeout:- tag:];
}else{ // 如果是文本
NSString *headerString = [NSString stringWithFormat:@"text&& &&"];
NSData *fileData = [headerString dataUsingEncoding:NSUTF8StringEncoding];
// 转为data
NSData *headerData = [headerString dataUsingEncoding:NSUTF8StringEncoding];
// 发送
NSMutableData *allData = [NSMutableData dataWithCapacity:];
[allData replaceBytesInRange:NSMakeRange(, headerData.length) withBytes:headerData.bytes]; // 替换为消息头
[allData appendData:fileData];
[self.socketClient writeData:allData withTimeout:- tag:];
}
}
- (void)onSocket:(AsyncSocket *)sock didWriteDataWithTag:(long)tag{
NSLog(@"发送完成!");
}
【day1102_uploadAndDown】:上传和下载客户端
需求:实现上传和下载功能
分三步完成,先上传,在获取下载列表,最后是下载
上传文件是通过在文件中拼接消息头完成
- (void)viewDidLoad
{
[super viewDidLoad];
self.socketClient = [[AsyncSocket alloc] initWithDelegate:self];
[self.socketClient connectToHost:@"192.168.1.188" onPort: withTimeout:- error:Nil];
// 发送数据
NSLog(@"发送数据");
NSString *headerString = [NSString stringWithFormat:@"upload&&%@&&%d",self.file.fileName,self.file.fileLength];
NSMutableData *allData = [MXUtils getAllDataByHeaderString:headerString];
NSData *fileData = [NSData dataWithContentsOfFile:self.file.filePath];
NSLog(@"%d",self.file.fileLength);
[allData appendData:fileData];
[self.socketClient writeData:allData withTimeout:- tag:];
self.labelUploadInfo.text = [NSString stringWithFormat:@"上传%@",self.file.fileName];
}
-(void)onSocket:(AsyncSocket *)sock didWriteDataWithTag:(long)tag{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"提示" message:@"上传成功" delegate:self cancelButtonTitle:@"确定" otherButtonTitles:Nil, nil];
[alert show];
}
获取下载列表是通过互相发送消息,从服务端把文件对象(也就是文件在服务端的绝对路径)归档发送到客户端,然后在客户端反归档获取文件列表
- (void)viewDidLoad
{
[super viewDidLoad];
self.socketClient = [[AsyncSocket alloc] initWithDelegate:self];
[self.socketClient connectToHost:@"192.168.1.188" onPort: withTimeout:- error:Nil];
NSString *headerString = @"downList&& &&";
NSMutableData *allData = [MXUtils getAllDataByHeaderString:headerString];
[self.socketClient writeData:allData withTimeout:- tag:];
[self.socketClient readDataWithTimeout:- tag:];
}
-(void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag{
// 反归档
NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
self.filePathArray = [unarchiver decodeObjectForKey:@"downlist"];
NSLog(@"%@",self.filePathArray);
[self.tableView reloadData];
}
下载是通过列表中的文件路径发送给服务端,然后服务端根据其路径找到文件返回去
- (void)viewDidLoad
{
[super viewDidLoad];
self.fileData = [NSMutableData data];
self.socketClient = [[AsyncSocket alloc] initWithDelegate:self];
[self.socketClient connectToHost:@"192.168.1.188" onPort: withTimeout:- error:Nil];
// 发送数据
NSLog(@"发送数据");
NSString *headerString = [NSString stringWithFormat:@"download&&%@&&",self.file.filePath];
NSMutableData *allData = [MXUtils getAllDataByHeaderString:headerString];
NSLog(@"%d",self.file.fileLength);
[self.socketClient writeData:allData withTimeout:- tag:];
self.labelDownload.text = [NSString stringWithFormat:@"下载%@",self.file.fileName];
[self.socketClient readDataWithTimeout:- tag:];
}
-(void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag{
[self.fileData appendData:data];
if (self.fileData.length == self.file.fileLength) {
[self.fileData writeToFile:[@"/Users/tarena/yz/第三阶段(高级UI)/day11/download" stringByAppendingPathComponent:self.file.fileName] atomically:YES];
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"提示" message:@"下载成功" delegate:self cancelButtonTitle:@"确定" otherButtonTitles:Nil, nil];
[alert show];
}
[self.socketClient readDataWithTimeout:- tag:];
}
【day1103_UploadAndDownSocketServer】:上传和下载服务端
- (void)viewDidLoad
{
[super viewDidLoad];
self.socketServer = [[AsyncSocket alloc] initWithDelegate:self];
[self.socketServer acceptOnPort: error:Nil];
}
-(void)onSocket:(AsyncSocket *)sock didAcceptNewSocket:(AsyncSocket *)newSocket{
NSLog(@"通道");
self.socketNew = newSocket;
}
-(void)onSocket:(AsyncSocket *)sock didConnectToHost:(NSString *)host port:(UInt16)port{
NSLog(@"连接成功%@",host);
self.host = host;
[self.socketNew readDataWithTimeout:- tag:];
}
-(void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag{
NSLog(@"读取数据成功");
// 判断是否有消息头
NSData *headerData = [data subdataWithRange:NSMakeRange(, )];
NSString *headerString = [[NSString alloc] initWithData:headerData encoding:NSUTF8StringEncoding]; // 消息头
if (headerString && [headerString componentsSeparatedByString:@"&&"].count == ) {
NSLog(@"有消息头");
NSArray *headerArray = [headerString componentsSeparatedByString:@"&&"];
NSString *type = headerArray[];
if ([type isEqualToString:@"upload"]) {
self.allData = [NSMutableData data];
self.fileName = headerArray[];
self.fileLength = [headerArray[] intValue];
NSData *subData = [data subdataWithRange:NSMakeRange(, data.length - )];
[self.allData appendData:subData];
self.labelUploadInfo.text = [NSString stringWithFormat:@"%@在上传%@文件",self.host,self.fileName];
self.progressUpload.progress = self.allData.length * 1.0 / self.fileLength;
}else if([type isEqualToString:@"downList"]){
NSData *data = [MXUtils getFilePathArrayDataByDirectoryPath:@""];
[self.socketNew writeData:data withTimeout:- tag:];
}else if([type isEqualToString:@"download"]){
NSString *filePath = headerArray[];
NSData *data = [NSData dataWithContentsOfFile:filePath];
[self.socketNew writeData:data withTimeout:- tag:];
}
}else{
[self.allData appendData:data];
self.progressUpload.progress = self.allData.length * 1.0 / self.fileLength;
}
if (self.allData.length == self.fileLength) {
NSString *path = [@"" stringByAppendingPathComponent:self.fileName];
NSLog(@"写到%@",path);
[self.allData writeToFile:path atomically:YES];
}
[self.socketNew readDataWithTimeout:- tag:];
}
把消息头存进要发送的数据中 并且固定占用多少字节
使用网络需要导入CFNetwork.framework框架
高级UIKit-09(TCPSocket发送文件、上传和下载)的更多相关文章
- java web学习总结(二十四) -------------------Servlet文件上传和下载的实现
在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的,如果直接使用 ...
- (转载)JavaWeb学习总结(五十)——文件上传和下载
源地址:http://www.cnblogs.com/xdp-gacl/p/4200090.html 在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传 ...
- JavaWeb学习总结,文件上传和下载
在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的,如果直接使用 ...
- java文件上传和下载
简介 文件上传和下载是java web中常见的操作,文件上传主要是将文件通过IO流传放到服务器的某一个特定的文件夹下,而文件下载则是与文件上传相反,将文件从服务器的特定的文件夹下的文件通过IO流下载到 ...
- JavaWeb学习总结(五十)——文件上传和下载
在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的,如果直接使用 ...
- 文件上传和下载(可批量上传)——Spring(二)
针对SpringMVC的文件上传和下载.下载用之前“文件上传和下载——基础(一)”的依然可以,但是上传功能要修改,这是因为springMVC 都为我们封装好成自己的文件对象了,转换的过程就在我们所配置 ...
- Struts2 之 实现文件上传和下载
Struts2 之 实现文件上传和下载 必须要引入的jar commons-fileupload-1.3.1.jar commons-io-2.2.jar 01.文件上传需要分别在struts.xm ...
- 19、文件上传与下载/JavaMail邮件开发
回顾: 一. 监听器 生命周期监听器 ServletRequestListener HttpSessionListener ServletContextListener 属性监听器 ServletRe ...
- JavaWeb——文件上传和下载
在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传,浏览器在上传的过程中是将文件以流的形式提交到服务器端的,如果直接使用 ...
- 文件上传与下载/Mail
文件上传与下载 提交方式:post 表单中要有文件上传的表单项 input type=”file”而且必须有name属性 表单类型要加入 encytype=”mulitpart/form-data” ...
随机推荐
- .net 常用方法
1.String数组转换成Int数组 string[] strArr = "a,b,c".Split(','); int[] intArr = Array.ConvertAll& ...
- c 有关N!阶乘的相关问题----陆续补充上来
第一个:求N!结果中末尾0的个数问题.思路是末尾0的产生 5*偶数,阶乘中偶数的个数肯定比5多,所以求出阶乘中5的个数就可以求出末尾0的个数. #include<stdio.h> in ...
- WPF自学笔记
WPF使用哪几种元素作为顶级元素: 1. Window元素 2. Page元素(与Window元素类似,用于可导航的应用程序) 3. Application元素(定义应用程序资源和启动设置) PS:在 ...
- js 字符串为空
content.replace(/(^\s)|(\s$)/g, "")
- 写一个背景渐变的TextView输入框
1:在res文件夹下新建一个drawble文件夹,并新建一个背景文件如writebg.xml <?xml version="1.0" encoding="utf-8 ...
- HTML5 总结-地理定位-6
HTML5 地理定位 定位用户的位置 HTML5 Geolocation API 用于获得用户的地理位置. 鉴于该特性可能侵犯用户的隐私,除非用户同意,否则用户位置信息是不可用的. 浏览器支持 Int ...
- STL assign 和swap
首先看下在整个container上面的复制. c1=c2 可以等同于 c1.erase(c1.begin(),c1.end()) //delete all elems in c1 c1.insert( ...
- Windows Azure 社区新闻综述(#77 版)
欢迎查看最新版本的每周综述,其中包含有关云计算和 Windows Azure 的社区推动新闻.内容和对话.以下是本周的亮点. 文章.视频和博客文章 · 文章: Windows Azure 表存储简 ...
- 《Java并发编程实战》第二章 线程安全性 读书笔记
一.什么是线程安全性 编写线程安全的代码 核心在于要对状态訪问操作进行管理. 共享,可变的状态的訪问 - 前者表示多个线程訪问, 后者声明周期内发生改变. 线程安全性 核心概念是正确性.某个类的行为与 ...
- Java集合中对象排序
集合中的对象排序需求还是比較常见的.当然我们能够重写equals方法,循环比較:同一时候Java为我们提供了更易使用的APIs.当须要排序的集合或数组不是单纯的数字型时,通常能够使用Comparato ...