iOS中 HTTP/Socket/TCP/IP通信协议详解 韩俊强的博客

简单介绍:

    // OSI(开放式系统互联), 由ISO(国际化标准组织)制定
    // 1. 应用层
    // 2. 表示层
    // 3. 会话层
    // 4. 传输层
    // 5. 网络层
    // 6. 数据链接层
    // 7. 物理层
    
    // TCP/IP, 由美国国防部制定
    // 1. 应用层, HTTP, FTP, SMTP, DNS
    // 2. 传输层, TCP, UDP
    // 3. 网络层, IP
    // 4. 链路层, ARP, RARP
    
    // HTTP(短连接)
    // 1. 建立链接, 三次握手
    // 2. 断开链接, 四次挥手
    
    // 数据报文->数据包->数据帧->比特流(二进制)-->比特流->数据帧->数据包->数据报文
    
    // socket, "插口", "套接字", 长连接, 存在于应用层和传输层之间, 提供一种封装, 方便进行通信

下面详解Socket通信:

布局如下:

              


引进框架:



服务端代码:

//  Created by 韩俊强 on 15/12/25.
//  Copyright © 2015年 韩俊强. All rights reserved.

#import "ViewController.h"
// 使用CocoPods使用<>, 可以指定路径
#import <CocoaAsyncSocket/CocoaAsyncSocket.h>
#import "GNASocket.h"

@interface ViewController ()<GCDAsyncSocketDelegate>

@property (weak, nonatomic) IBOutlet UITextField *portTF;
@property (weak, nonatomic) IBOutlet UITextView *message; // 多行文本输入框
@property (weak, nonatomic) IBOutlet UITextField *content;

@property (nonatomic, strong) GCDAsyncSocket *clientSocket;// 为客户端生成的socket

// 服务器socket
@property (nonatomic, strong) GCDAsyncSocket *serverSocket;

@end

@implementation ViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
}

// 服务端监听某个端口
- (IBAction)listen:(UIButton *)sender
{
    // 1. 创建服务器socket
    self.serverSocket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
    
    // 2. 开放哪些端口
    NSError *error = nil;
    BOOL result = [self.serverSocket acceptOnPort:self.portTF.text.integerValue error:&error];
    
    // 3. 判断端口号是否开放成功
    if (result) {
        [self addText:@"端口开放成功"];
    } else {
        [self addText:@"端口开放失败"];
    }
}

// 发送
- (IBAction)sendMessage:(UIButton *)sender
{
    NSData *data = [self.content.text dataUsingEncoding:NSUTF8StringEncoding];
    [self.clientSocket writeData:data withTimeout:-1 tag:0];
    
    GNASocket *socket = [GNASocket defaultScocket];
    [socket.mySocket readDataWithTimeout:-1 tag:0];
}

// 接收消息
- (IBAction)receiveMassage:(UIButton *)sender
{
    [self.clientSocket readDataWithTimeout:-1 tag:0];
}


// textView填写内容
- (void)addText:(NSString *)text
{
    self.message.text = [self.message.text stringByAppendingFormat:@"%@\n", text];
}

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    
}

#pragma mark - GCDAsyncSocketDelegate
// 当客户端链接服务器端的socket, 为客户端单生成一个socket
- (void)socket:(GCDAsyncSocket *)sock didAcceptNewSocket:(GCDAsyncSocket *)newSocket
{
    [self addText:@"链接成功"];
    //IP: newSocket.connectedHost
    //端口号: newSocket.connectedPort
    [self addText:[NSString stringWithFormat:@"链接地址:%@", newSocket.connectedHost]];
    [self addText:[NSString stringWithFormat:@"端口号:%hu", newSocket.connectedPort]];
    // short: %hd
    // unsigned short: %hu
    
    // 存储新的端口号
    self.clientSocket = newSocket;
}

- (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag
{
    NSString *message = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    [self addText:message];
}


@end

客户端代码:

//  Created by 韩俊强 on 15/12/25.
//  Copyright © 2015年 韩俊强. All rights reserved.
//

#import "SecondViewController.h"
#import <CocoaAsyncSocket.h>
#import "GNASocket.h"

@interface SecondViewController ()<GCDAsyncSocketDelegate>

@property (weak, nonatomic) IBOutlet UITextField *addressTF;
@property (weak, nonatomic) IBOutlet UITextField *portTF;
@property (weak, nonatomic) IBOutlet UITextField *message;

@property (weak, nonatomic) IBOutlet UITextView *content;

@property (nonatomic, strong) GCDAsyncSocket *socket;

@end

@implementation SecondViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
    
}

// 和服务器进行链接
- (IBAction)connect:(UIButton *)sender
{
    // 1. 创建socket
    self.socket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
    
    // 2. 与服务器的socket链接起来
    NSError *error = nil;
    BOOL result = [self.socket connectToHost:self.addressTF.text onPort:self.portTF.text.integerValue error:&error];
    
    // 3. 判断链接是否成功
    if (result) {
        [self addText:@"客户端链接服务器成功"];
    } else {
        [self addText:@"客户端链接服务器失败"];
    }
}

// 接收数据
- (IBAction)receiveMassage:(UIButton *)sender
{
    [self.socket readDataWithTimeout:-1 tag:0];
}

// 发送消息
- (IBAction)sendMassage:(UIButton *)sender
{
    [self.socket writeData:[self.message.text dataUsingEncoding:NSUTF8StringEncoding] withTimeout:-1 tag:0];
}


// textView填写内容
- (void)addText:(NSString *)text
{
    self.content.text = [self.content.text stringByAppendingFormat:@"%@\n", text];
}


- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
}

#pragma mark - GCDAsyncSocketDelegate

// 客户端链接服务器端成功, 客户端获取地址和端口号
- (void)socket:(GCDAsyncSocket *)sock didConnectToHost:(NSString *)host port:(uint16_t)port
{
    [self addText:[NSString stringWithFormat:@"链接服务器%@", host]];
    
    GNASocket *socket = [GNASocket defaultScocket];
    socket.mySocket = self.socket;
}

// 客户端已经获取到内容
- (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag
{
    NSString *content = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    [self addText:content];
}

@end



为通信传值写一个单例:

//.h
#import <Foundation/Foundation.h>
#import <CocoaAsyncSocket.h>

@interface GNASocket : NSObject

@property (nonatomic, strong) GCDAsyncSocket *mySocket;

+ (GNASocket *)defaultScocket;

@end


//.m

#import "GNASocket.h"

@implementation GNASocket

+ (GNASocket *)defaultScocket
{
    static GNASocket *socket = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        socket = [[GNASocket alloc] init];
    });
    return socket;
}

@end

最终效果:



iOS开发者交流群:446310206


GitHub 项目地址喜欢记得start一下哦!




  • 6
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
### 回答1: func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { let cell = tableView.cellForRow(at: indexPath) cell?.imageView?.image = UIImage(named: "highlightedImage") } ### 回答2: 在 Swift ,可以使用 `UITableViewDelegate` 协议的 `didSelectRowAt` 方法来实现点击 UITableViewCell 进行高亮图片的效果。以下是一个示例代码: 首先,在你的视图控制器设置 `UITableViewDelegate`: ```swift class ViewController: UIViewController, UITableViewDelegate { // ... } ``` 在 `viewDidLoad` 方法,将 `UITableViewDelegate` 设置为你的 `UITableView` 的代理: ```swift override func viewDidLoad() { super.viewDidLoad() tableView.delegate = self } ``` 然后,实现 `didSelectRowAt` 方法来处理选 cell 的操作,可以在这个方法获取 cell 的索引并更改其状态: ```swift func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { guard let cell = tableView.cellForRow(at: indexPath) else { return } // 选时的高亮效果,可以自行修改 cell.contentView.backgroundColor = .yellow } ``` 最后,为了取消高亮效果,还需要实现 `didDeselectRowAt` 方法: ```swift func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) { guard let cell = tableView.cellForRow(at: indexPath) else { return } // 取消高亮效果 cell.contentView.backgroundColor = .white } ``` 以上是一个点击 UITableViewCell 高亮图片的 Swift 代码示例,你可以根据自己的需求修改高亮效果的样式和图片。 ### 回答3: 下面是一个 Swift 的代码示例,用于在点击表格单元格时高亮显示图片。 ```swift import UIKit class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate { let cellID = "cellIdentifier" let colors = [UIColor.red, UIColor.green, UIColor.blue] var tableView: UITableView! override func viewDidLoad() { super.viewDidLoad() tableView = UITableView(frame: self.view.frame) tableView.delegate = self tableView.dataSource = self tableView.register(UITableViewCell.self, forCellReuseIdentifier: cellID) self.view.addSubview(tableView) } func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { return colors.count } func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let cell = tableView.dequeueReusableCell(withIdentifier: cellID, for: indexPath) cell.selectionStyle = .none cell.textLabel?.text = "Row \(indexPath.row)" cell.textLabel?.textColor = UIColor.white cell.backgroundColor = colors[indexPath.row] return cell } func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { let cell = tableView.cellForRow(at: indexPath) cell?.imageView?.highlightedImage = UIImage(named: "highlighted_image") cell?.setHighlighted(true, animated: true) } func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) { let cell = tableView.cellForRow(at: indexPath) cell?.imageView?.highlightedImage = nil cell?.setHighlighted(false, animated: true) } } ``` 上述代码,我们创建了一个包含了不同颜色单元格的表格视图。当点击单元格时,我们设置`imageView`的`highlightedImage`为一个高亮图片,并将单元格的高亮状态设置为`true`。当取消选单元格时,我们将`highlightedImage`设置为`nil`,并将单元格的高亮状态设置为`false`。请确保你有一个名为`highlighted_image`的图片文件,且存在于项目

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

韩俊强

奖励一杯咖啡☕️

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值