利用终端模拟群聊功能

2016-03-09  本文已影响95人  Alexander

前言


#import <Foundation/Foundation.h>

@interface WGServicerListener : NSObject

/** 定义一个对象方法,用于QQ开启服务器 */
- (void)startServeSocket;

@end

#import <Foundation/Foundation.h>
#import "GCDAsyncSocket.h"
#import "WGServicerListener.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {

        // 创建一个服务器的监听对象
        WGServicerListener *serveListener = [[WGServicerListener alloc] init];
        
        // 监听开启服务器
        [serveListener startServeSocket];
        
        // 保证服务器一直开启
        [[NSRunLoop mainRunLoop] run];
        
    }
    return 0;
}


#import "WGServicerListener.h"
#import "GCDAsyncSocket.h"

@interface WGServicerListener () <GCDAsyncSocketDelegate>

/** 服务器的Socket对象 */
@property(nonatomic, strong) GCDAsyncSocket *serverSocket;

/** 保存所有客户端对象 */
@property(nonatomic, strong) NSMutableArray *clientSockets;

@end

@implementation WGServicerListener

#pragma mark - 懒加载
- (NSMutableArray *)clientSockets
{
    if (!_clientSockets) {
        _clientSockets = [NSMutableArray array];
    }
    return _clientSockets;
}


- (void)startServeSocket {

    // 创建一个服务器的Socket对象
    GCDAsyncSocket *serverSocket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_global_queue(0, 0)];
    
    // 绑定并监听serverSocket对象
    NSError *error = nil;
    [serverSocket acceptOnPort:1886 error:&error];
    
    // 判断是否开启QQ服务器
    if (!error) {
        NSLog(@"QQ服务器已经开启");
    } else
    {
        NSLog(@"QQ服务器开启失败");
    }
    
    // 保存创建的服务器Socket对象
    self.serverSocket = serverSocket;
    
}

/**
 *  只要有客户端连接服务器就会调用该代理方法. 第一个Socket表示服务器的Socket对象,第二个是客户端的Socket对象
 *  在该方法中的参数中:serverSocket就是服务器端的Socket,所以需要定义一个属性强引用着它,第二个参数是服务器
 *  端的clientSocket,用于读取客户端上传的数据.所以需要定义一个数组保存它.
 */

- (void)socket:(GCDAsyncSocket *)serverSocket didAcceptNewSocket:(GCDAsyncSocket *)clientSocket
{
    
//    NSLog(@"%@",serverSocket);
//    NSLog(@"%@",clientSocket);
    // 保存和QQ服务器连接的客户端
    [self.clientSockets addObject:clientSocket];
    
    // 监听客户端有没有上传数据
    /**
     * -1 表示不要超时
     */
    [clientSocket readDataWithTimeout:-1 tag:0];
    
    NSLog(@"客户端%ld已经连接到服务器了",self.clientSockets.count);
}

/**
 *   监听客户端有没有上传数据,只要和服务器连接的客户端发送了消息,那么就一定会调用该方法
 *   第一个参数是客户端(因为监听的是客户端是否发送消息)
 */
- (void)socket:(GCDAsyncSocket *)clientSocket didReadData:(NSData *)data withTag:(long)tag
{
    // 传进来的是一个NSData类型,需要将它转为字符串类型
    NSString *responeStr = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    NSLog(@"%@",responeStr);
    
    // 发送消息,在发送消息之前需要判断当前监听的对象(客户端)是不是自己,如果是自己那么就不要发送消息给自己了
    for (GCDAsyncSocket *socket in self.clientSockets) {
        if (socket != clientSocket) {
            
            [socket writeData:data withTimeout:-1 tag:0];
        }
    }

    // 每次发送完毕消息,都需要监听客户端是否上传了信息,如果不监听,永远发送不了下一条消息
    [clientSocket readDataWithTimeout:-1 tag:0];

    
}
@end
// 箭头表示发送消息的方向
QQ1(端口号01)—> QQ服务器 —> QQ3
QQ2(端口号02)—> QQ服务器 —> QQ3
       —>QQ服务器—> 根据端口号01 —> QQ1
       |
  QQ3  恢复消息
       | 
       —>QQ服务器—> 根据端口号02—>QQ2

发送消息的原理 : 客户端的数据—>服务器 —> 发给好友(转发的过程)

上一篇 下一篇

猜你喜欢

热点阅读