IOS

一、Delegate为什么要用weak或者assign修饰,不能用strong?
说明:weak、assign修饰的不会使引用计数器加1,strong会使引用计数器加1。
在Dog类中定义一个代理,先用Strong修饰:

#import <Foundation/Foundation.h>

@protocol DogDelegate<NSObject>
-(void)testDog;
@end

@interface Dog : NSObject
@property(nonatomic,strong)id<DogDelegate>delegate;
@end

在person类中使用Dog:


image.png

在viewController类中:


image.png

这种情况下,示意图为:


image.png

可以看出:
1、

@property(nonatomic,strong)Dog *dog;

person对dog有强引用
2、

@property(nonatomic,strong)id<DogDelegate>delegate;

self.dog.delegate = self,这句使delegate又对Person有强引用。

这样一来,当viewController不对Person引用后,dog.delegate对Person还有强引用,所以person不会释放,当然Person中的dog也不会释放,这样就有循环引用的情况。

第二情况:
在Dog类中,如果用weak修饰的话:

#import <Foundation/Foundation.h>

@protocol DogDelegate<NSObject>
-(void)testDog;
@end

@interface Dog : NSObject
@property(nonatomic,weak)id<DogDelegate>delegate;
@end

那么上面的示意图中,delegate对Person的引用为弱引用。
当viewController不对Person引用后,Person的retainCount为0,即person会被释放,因此,Person中的dog也会被释放了。这样就可以避免循环引用的问题了。

二、block的循环引用
文章参考了:https://www.jianshu.com/p/4384a42778bc
首先,不是在block中使用强引用,例如self,都会引起循环引用,这需要视情况而定。
比如: block不是self的属性或者变量时,在block内使用self也不会循环引用:
例如定义一个block:

@interface ViewController ()
{
    void(^myBlock)(void);
}

如果想要在block中,使用self,一般会这样解决循环引用:

__weak typeof(self) weakself = self;
    myBlock = ^(void){
        NSLog(@"----value111-%@====address111--%p-----sel111f=%p",weakself.person,weakself.person,weakself);

    };

这样的确可以解决block中的循环引用的问题,但是也有可能产生另外一个问题:
block中的使用的self已经被释放了,也就是为nil了。
假如:
新建一个继承NSObject的类testObject

#import "testObject.h"

@implementation testObject

-(void)dealloc{
    NSLog(@"本对象已经被销毁了--");
}
@end

我们知道判断一个对象是否被释放最可靠的方法就是看释放执行dealloc。
那么在ViewController中:

@interface ViewController ()
{
    void(^myBlock)(void);
}
@property(nonatomic,strong)testObject *test;
@property(nonatomic,strong)NSString *person;

@end

然后:

__weak typeof (_test) weakself = _test;
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        for (int i=0; i<10; i++) {
            NSLog(@"-----------llllaaa===%@====%d",weakself,i);
            sleep(1);
        }
    });
    // 3秒后,释放对象
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3*NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        self.test = nil;
    });

也就是说3秒后将test释放,子线程中的block的test就为nil了,打印结果:

2018-03-10 17:02:03.748705+0800 aaa[3856:207467] -----------llllaaa===<testObject: 0x60400001c7b0>====1
2018-03-10 17:02:04.752324+0800 aaa[3856:207467] -----------llllaaa===<testObject: 0x60400001c7b0>====2
2018-03-10 17:02:05.754304+0800 aaa[3856:207467] -----------llllaaa===<testObject: 0x60400001c7b0>====3
2018-03-10 17:02:06.042539+0800 aaa[3856:207349] 本对象已经被销毁了--
2018-03-10 17:02:06.757422+0800 aaa[3856:207467] -----------llllaaa===(null)====4
2018-03-10 17:02:07.759369+0800 aaa[3856:207467] -----------llllaaa===(null)====5
2018-03-10 17:02:08.764179+0800 aaa[3856:207467] -----------llllaaa===(null)====6
2018-03-10 17:02:09.765000+0800 aaa[3856:207467] -----------llllaaa===(null)====7
2018-03-10 17:02:10.769800+0800 aaa[3856:207467] -----------llllaaa===(null)====8
2018-03-10 17:02:11.774028+0800 aaa[3856:207467] -----------llllaaa===(null)====9

怎么解决?在block中再用strong修饰符,

  dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        __strong typeof (_test) strongSelf = _test; // 这样处理下
        for (int i=0; i<10; i++) {
            NSLog(@"-----------llllaaa===%@====%d",strongSelf,i);
            sleep(1);
        }
    });
    // 3秒后,释放对象
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3*NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        self.test = nil;
    });

打印:

2018-03-10 17:07:18.166733+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====1
2018-03-10 17:07:19.168924+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====2
2018-03-10 17:07:20.169278+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====3
2018-03-10 17:07:21.170645+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====4
2018-03-10 17:07:22.173204+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====5
2018-03-10 17:07:23.177067+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====6
2018-03-10 17:07:24.181829+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====7
2018-03-10 17:07:25.186076+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====8
2018-03-10 17:07:26.190384+0800 aaa[3905:211388] -----------llllaaa===<testObject: 0x60000000e9c0>====9
2018-03-10 17:07:27.192283+0800 aaa[3905:211388] 本对象已经被销毁了--

三、ios的编译跟链接
编译:只是单纯的检查每个源文件的语法是否合理,并不会检查每个源文件之前的关联关系,一个源文件编译成功就会产生一个目标文件。
链接:检查目标文件之前的关联关系,将相关联的目标文件组合在一起,生成可执行的文件。

四、runtime使用。
说起这个,最容易想起的应该是:交换方法、给分类动态关联属性了。

使用一、交换两个方法。

场景:在保持系统原有方法功能的基础上,添加额外的功能。
需求:加载一张图片:[UIImage imageNamed:@"image"]是无法知道这张图片是否已经加载成功的。
实现:使用runtime将两个方法进行交换
步骤:
1.定义一个Image的分类
2.在分类中定义一个方法,用于跟系统的imageNamed方法交换
3.交换方法。
定义一个UIImage+Image的分类

#import "UIImage+Image.h"
#import <objc/message.h>
@implementation UIImage (Image)

在load方法中用runtime将两个方法进行交换


image.png
image.png

完整代码:

#import "UIImage+Image.h"
#import <objc/message.h>
@implementation UIImage (Image)

// 在类加载进入内存的时候调用,并且只会调用一次
+(void)load{
    // 1.获取imageNamed方法的地址
    Method imageNamedMethod = class_getClassMethod(self, @selector(imageNamed:));
    // 2.获取自己写的replace_imageNamed方法地址
    Method replaceImageMethod = class_getClassMethod(self, @selector(replace_imageNamed:));
    // 3.交换方法地址,相当于交换实现的方式
    
    method_exchangeImplementations(imageNamedMethod, replaceImageMethod
                                   );
}


// 这样就相当于不改变原来功能的基础上增加了新的功能
/**
 注意:这里是不会引起死循环的
 调用:imageNamed: 相当于 调用 replace_imageNamed:
 调用: replace_imageNamed: 相当于 调用 imageNamed:
 */
+(UIImage *)replace_imageNamed:(NSString *)name{
    UIImage *image = [UIImage replace_imageNamed:name];
    if (image) {
        NSLog(@"---method交换成功");
    }else{
        NSLog(@"---method交换失败");
    }
    return image;
}
@end

使用:

#import "ViewController.h"
#import "UIImage+Image.h"
#import "Person.h"
#import "Student.h"
#import "NSObject+testRuntime.h"
#import "NSObject+DictToModel.h"
@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(100, 100, 100, 100)];
    imageView.backgroundColor = [UIColor redColor];
    imageView.image = [UIImage imageNamed:@"8.png"];
    [self.view addSubview:imageView];
    

打印

---method交换成功

使用二、给分类动态添加属性
场景:例如给NSObject添加一个分类,在分类中是不能够添加成员属性的,虽然我们用了@property,但是仅仅会自动生成get和set方法的声明,并没有带下划线的属性和方法实现生成。但是我们可以通过runtime就可以做到给它方法的实现。

定义了一个NSObject+testRumtime

#import "NSObject+testRuntime.h"
#import <objc/message.h>
@implementation NSObject (testRuntime)

在.h文件中

#import <Foundation/Foundation.h>

@interface NSObject (testRuntime)

@property(nonatomic,copy)NSString *name;
-(void)obtainMessage;

@end

.m文件中

#import "NSObject+testRuntime.h"
#import <objc/message.h>
@implementation NSObject (testRuntime)
/**
 动态添加属性关联
 objc_setAssociatedObject  将某个值跟某个对象关联起来,将某个值存储到某个对象中
 object  给那个对象添加属性
 key 属性的名称
 value 属性值
 policy 保存策略
 */
-(void)setName:(NSString *)name{
    
    objc_setAssociatedObject(self, @"name", name,OBJC_ASSOCIATION_COPY_NONATOMIC);
}

-(void)obtainMessage{
    NSString *aa = self.name;
    NSLog(@"adasd-=--------%@",aa);
}

-(NSString *)name{
    return objc_getAssociatedObject(self, @"name");
}
@end

使用:

 Person *pp  = [[Person alloc]init];
 pp.name = @"xiaoming";
 NSLog(@"log--the--print--%@",pp.name);

打印:

log--the--print--xiaoming

使用三、将字典转成模型
思路:利用运行时,遍历模型的所有的属性,然后找到属性的名字,作为key到字典中查找value进行赋值。
出现的情况:
1.字典的key跟模型的属性匹配不上(key比模型的属性多,或者模型的属性比key多)
2.模型中套有模型
3.模型属性是数组,数组有模型

注解:根据上面的三种特殊情况,先是字典的key和模型的属性不对应的情况。不对应有两种,一种是字典的键值大于模型属性数量,这时候我们不需要任何处理,因为runtime是先遍历模型所有属性,再去字典中根据属性名找对应值进行赋值,多余的键值对也当然不会去看了;另外一种是模型属性数量大于字典的键值对,这时候由于属性没有对应值会被赋值为nil,就会导致crash,我们只需加一个判断即可。考虑三种情况下面一一注解;

步骤:提供一个NSObject分类,专门字典转模型,以后所有模型都可以通过这个分类实现字典转模型。

实现:
创建一个专门用来字典转模型的分类:NSObject+DicToModel

#import "NSObject+DictToModel.h"
#import <objc/message.h>
@implementation NSObject (DictToModel)

.m文件实现

#import "NSObject+DictToModel.h"
#import <objc/message.h>
@implementation NSObject (DictToModel)

+(instancetype)modelWithDict:(NSDictionary *)dict
{
    // 1.创建对应的对象
    id objc = [[self alloc]init];
    
    /**
     class_copyIvarList 获取某个类中的所有成员变量
     第一个参数: 表示获取哪一个类中的成员变量
     第二个参数: 表示这个类有多少的成员变量,传入的是Int变量的地址,会自动给变量地址赋值的
     返回值:Ivar *  是一个ivar数组来的,会把所有成员属性放在一个数组中,然后通过返回的数组就能全部获取到。
     count 成员变量的个数
     */
    unsigned int count = 0;
    // 获取类中的所有成员变量
    Ivar *ivarList =  class_copyIvarList(self, &count);
    
    for (int i=0; i<count; i++) {
        Ivar ivar = ivarList[I];
        // 获取成员变量的名字
        NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
        // 获取变量的类型
        NSString *ivarType = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];
        
        // 处理成员变量名--->字典中的key(去掉_,从第一个角标开始截取)
        NSString *key = [ivarName substringFromIndex:1];
        // 根据属性名到字典中查找对应的value
        id value = dict[key];
        
        // 二级转换:如果字典中还有字典,也需要把对应字典进行转换
        if ([value isKindOfClass:[NSDictionary class]] && ![ivarType hasPrefix:@"NS"]) {
            
            // 根据字符串的类型生成类对象
            Class modelClass = NSClassFromString(ivarType);
            if (modelClass) {// 有对应的模型才需要转换
                value = [modelClass modelWithDict:value];
            }
        }
        // 三级转换,NSArray中也是字典,把数组中的字典转换成模型
        if ([value isKindOfClass:[NSArray class]]) {
            if ([self respondsToSelector:@selector(arrayContainModelClass)]) {
                // 这里先转换为id类型,方便调用任何对象的方法
                id idSelf = self;
                // 获取数组中字典对应的模型
                NSString *type = [idSelf arrayContainModelClass][key];
        
                // 生成模型
                Class classModel = NSClassFromString(type);
                NSMutableArray *mutableArray = [[NSMutableArray alloc]init];
                // 遍历字典数组,生成模型数组
                for (NSDictionary *dict in value) {
                    // 字典转模型
                    id model = [classModel modelWithDict:dict];
                    [mutableArray addObject:model];
                }
            // 把模型数组赋值给value
                value = mutableArray;
            }
        }
        // 要做一个判断,避免如果模型的属性大于字典的,就会产生nil
        if (value) {
            [objc setValue:value forKey:key];
        }
    }
    return objc;
}


@end

.h

#import <Foundation/Foundation.h>

@protocol ModelDelegate<NSObject>
+(NSDictionary *)arrayContainModelClass;
@end

@interface NSObject (DictToModel)
+(instancetype)modelWithDict:(NSDictionary *)dict;
@end

使用四、动态添加方法
OC是一门动态语言,一个函数是由一个selector(SEL)和一个implement(IML)组成的。其中SEL相当于门牌号,而IML就是住户(函数的实现),同理,门牌号是可以乱发,但不一定能找到住户的。
ios中,根据门牌号(SEL)查找方法实现是有一个过程的。如图所示:
盗图的,参考http://www.cnblogs.com/biosli/p/NSObject_inherit_2.html

image.png

也就是,当调用了没有对应的IML时,

1.系统首先会执行对象中的resolveInstanceMethod函数来返回BOOL
2.如果调用resolveInstanceMethod还是没找到SEL实现,则会执行对象中的forwardingTargetForSelector来返回一个id
3.如果forwardingTargetForSelector返回的是nil或者self,则还有一次机会,
会执行系统中的methodSignatureForSelector来返回NSMethodSignature,
紧跟着执行对象中的forwardInvocation,
因此一般来说,methodSignatureForSelector跟forwardInvocation是成对出现的。
4.如果第3步还是没有可执行的IML,那么则会调用doesNotRecognizeSelector并且抛出异常。

应用的场景可分为:

在一个函数找不到时,Objective-C提供了三种方式去补救:

1、调用resolveInstanceMethod给个机会让类添加这个实现这个函数

2、调用forwardingTargetForSelector让别的对象去执行这个函数

3、调用methodSignatureForSelector(函数符号制造器)和forwardInvocation(函数执行器)灵活的将目标函数以其他形式执行。

如果都不中,调用doesNotRecognizeSelector抛出异常。

五、HTTP协议:
参考:https://www.cnblogs.com/ranyonsue/p/5984001.html
HTTP:超文本传输协议
一个http请求,包括请求、响应。
HTTP请求之URL

例如:http://www.aspxfans.com:8080/news/index.asp?boardID=5&ID=24618&page=1#name
一个完整的URL包括:
1.协议部分:该URL的协议部分为:"http:",这代表网页使用的是HTTP协议。在Internet中可以使用多种协议,如HTTP,FTP等等本例中使用的是HTTP协议。在"HTTP"后面的“//”为分隔符
2.域名部分:该URL的域名部分为“www.aspxfans.com”。一个URL中,也可以使用IP地址作为域名使用。
3.端口部分:跟在域名后面的是端口,域名和端口之间使用“:”作为分隔符。端口不是一个URL必须的部分,如果省略端口部分,将采用默认端口
4.虚拟目录部分:从域名后的第一个“/”开始到最后一个“/”为止,是虚拟目录部分。虚拟目录也不是一个URL必须的部分。本例中的虚拟目录是“/news/”
5.文件名部分:从域名后的最后一个“/”开始到“?”为止,是文件名部分,如果没有“?”,则是从域名后的最后一个“/”开始到“#”为止,是文件部分,如果没有“?”和“#”,那么从域名后的最后一个“/”开始到结束,都是文件名部分。本例中的文件名是“index.asp”。文件名部分也不是一个URL必须的部分,如果省略该部分,则使用默认的文件名
6.锚部分:从“#”开始到最后,都是锚部分。本例中的锚部分是“name”。锚部分也不是一个URL必须的部分
7.参数部分:从“?”开始到“#”为止之间的部分为参数部分,又称搜索部分、查询部分。本例中的参数部分为“boardID=5&ID=24618&page=1”。参数可以允许有多个参数,参数与参数之间用“&”作为分隔符。

HTTP请求之消息Request
客户端发送一个请求消息包括以下格式:

请求行,请求头,空行,请求数据
QQ20180328-0.png
Http请求消息结构.png
请求行以一个方法符号开头,以空格分开,后面跟着请求的URI和协议的版本。
Get请求例子,使用Charles抓取的request:

GET /562f25980001b1b106000338.jpg HTTP/1.1
Host    img.mukewang.com
User-Agent    Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/51.0.2704.106 Safari/537.36
Accept    image/webp,image/*,*/*;q=0.8
Referer    http://www.imooc.com/
Accept-Encoding    gzip, deflate, such
Accept-Language    zh-CN,zh;q=0.8
第一部分:请求行,用来说明请求类型,要访问的资源以及所使用的HTTP版本.

GET说明请求类型为GET,[/562f25980001b1b106000338.jpg]为要访问的资源,该行的最后一部分说明使用的是HTTP1.1版本。

第二部分:请求头部,紧接着请求行(即第一行)之后的部分,用来说明服务器要使用的附加信息

从第二行起为请求头部,HOST将指出请求的目的地.User-Agent,服务器端和客户端脚本都能访问它,它是浏览器类型检测逻辑的重要基础.该信息由你的浏览器来定义,并且在每个请求中自动发送等等

第三部分:空行,请求头部后面的空行是必须的

即使第四部分的请求数据为空,也必须有空行。

第四部分:请求数据也叫主体,可以添加任意的其他数据。

这个例子的请求数据为空。
POST请求例子,使用Charles抓取的request:

POST / HTTP1.1
Host:www.wrox.com
User-Agent:Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; .NET CLR 2.0.50727; .NET CLR 3.0.04506.648; .NET CLR 3.5.21022)
Content-Type:application/x-www-form-urlencoded
Content-Length:40
Connection: Keep-Alive

name=Professional%20Ajax&publisher=Wiley
第一部分:请求行,第一行明了是post请求,以及http1.1版本。
第二部分:请求头部,第二行至第六行。
第三部分:空行,第七行的空行。
第四部分:请求数据,第八行。

HTTP之响应消息Response

HTTP响应也由四个部分组成,分别是:状态行、消息报头、空行和响应正文。
111.png
例子

HTTP/1.1 200 OK
Date: Fri, 22 May 2009 06:07:21 GMT
Content-Type: text/html; charset=UTF-8

<html>
      <head></head>
      <body>
            <!--body goes here-->
      </body>
</html>
第一部分:状态行,由HTTP协议版本号, 状态码, 状态消息 三部分组成。

第一行为状态行,(HTTP/1.1)表明HTTP版本为1.1版本,状态码为200,状态消息为(ok)

第二部分:消息报头,用来说明客户端要使用的一些附加信息

第二行和第三行为消息报头,
Date:生成响应的日期和时间;Content-Type:指定了MIME类型的HTML(text/html),编码类型是UTF-8

第三部分:空行,消息报头后面的空行是必须的

第四部分:响应正文,服务器返回给客户端的文本信息。

空行后面的html部分为响应正文。
HTTP之状态码
状态代码有三位数字组成,第一个数字定义了响应的类别,共分五种类别:

1xx:指示信息--表示请求已接收,继续处理

2xx:成功--表示请求已被成功接收、理解、接受

3xx:重定向--要完成请求必须进行更进一步的操作

4xx:客户端错误--请求有语法错误或请求无法实现

5xx:服务器端错误--服务器未能实现合法的请求

常见状态码:

200 OK                        //客户端请求成功
400 Bad Request               //客户端请求有语法错误,不能被服务器所理解
401 Unauthorized              //请求未经授权,这个状态代码必须和WWW-Authenticate报头域一起使用 
403 Forbidden                 //服务器收到请求,但是拒绝提供服务
404 Not Found                 //请求资源不存在,eg:输入了错误的URL
500 Internal Server Error     //服务器发生不可预期的错误
503 Server Unavailable        //服务器当前不能处理客户端的请求,一段时间后可能恢复正常
更多状态码http://www.runoob.com/http/http-status-codes.html

TCP三次握手
参考:https://www.jianshu.com/p/ef892323e68f
所谓三次握手(Three-Way Handshake)即建立TCP连接,就是指建立一个TCP连接时,需要客户端和服务端总共发送3个包以确认连接的建立。在socket编程中,这一过程由客户端执行connect来触发,整个流程如下图所示:

]、

(1)第一次握手:Client将标志位SYN置为1,随机产生一个值seq=J,并将该数据包发送给Server,Client进入SYN_SENT状态,等待Server确认。

(2)第二次握手:Server收到数据包后由标志位SYN=1知道Client请求建立连接,Server将标志位SYN和ACK都置为1,ack=J+1,随机产生一个值seq=K,并将该数据包发送给Client以确认连接请求,Server进入SYN_RCVD状态。

(3)第三次握手:Client收到确认后,检查ack是否为J+1,ACK是否为1,如果正确则将标志位ACK置为1,ack=K+1,并将该数据包发送给Server,Server检查ack是否为K+1,ACK是否为1,如果正确则连接建立成功,Client和Server进入ESTABLISHED状态,完成三次握手,随后Client与Server之间可以开始传输数据了。
简单来说,就是

1、建立连接时,客户端发送SYN包(SYN=i)到服务器,并进入到SYN-SEND状态,等待服务器确认

2、服务器收到SYN包,必须确认客户的SYN(ack=i+1),同时自己也发送一个SYN包(SYN=k),即SYN+ACK包,此时服务器进入SYN-RECV状态

3、客户端收到服务器的SYN+ACK包,向服务器发送确认报ACK(ack=k+1),此包发送完毕,客户端和服务器进入ESTABLISHED状态,完成三次握手,客户端与服务器开始传送数据。

SYN攻击:

在三次握手过程中,Server发送SYN-ACK之后,收到Client的ACK之前的TCP连接称为半连接(half-open connect),此时Server处于SYN_RCVD状态,当收到ACK后,Server转入ESTABLISHED状态。SYN攻击就是Client在短时间内伪造大量不存在的IP地址,并向Server不断地发送SYN包,Server回复确认包,并等待Client的确认,由于源地址是不存在的,因此,Server需要不断重发直至超时,这些伪造的SYN包将产时间占用未连接队列,导致正常的SYN请求因为队列满而被丢弃,从而引起网络堵塞甚至系统瘫痪。SYN攻击时一种典型的DDOS攻击,检测SYN攻击的方式非常简单,即当Server上有大量半连接状态且源IP地址是随机的,则可以断定遭到SYN攻击了,使用如下命令可以让之现行:

#netstat -nap | grep SYN_RECV

TCP跟UDP的区别?

TCP (传输控制协议)
建立连接,形成传输通道
通过三次握手完成连接,
必须建立连接,效率较为低下
UDP(用户数据报协议)
将数据源和目的地封装成数据包中,
每个数据报的大小是有限制,在64k之内
因为不需要建立连接,所以是不可靠的
不需要建立连接,速度快。

举例:老师讲课,共享屏幕都学生的电脑上,老师电脑的屏幕就是数据源,学生电脑的ip就是目的地,封装成数据包后,直接扔到学生电脑上,但因为是不需要建立连接的,所以根本不知道学生是否有接收到数据,所以就会出现有时学生屏幕卡顿的情况。

TCP跟HTTP的区别
TCP:传输协议(决定用什么方式进行交互)
HTTP:协议(决定数据的格式) 例如请求头有规定"content-type"为son或者xml,编码格式为url或者utf8等
举例:从广州到北京 可以坐飞机 高铁 火车(指的是传输协议TCP)
到了北京后,进行交流用英语或者国语(指的是HTTP)

六、Socket、HTTP、TCP联系与区别
TCP连接:

手机能够使用联网功能是因为手机底层实现了TCP/IP协议,可以使手机终端通过无线网络建立TCP连接。TCP协议可以对上层网络提供接口,使上层网络数据的传输建立在“无差别”的网络之上。
建立起一个TCP连接需要经过“三次握手”:
第一次握手:客户端发送syn包(syn=j)到服务器,并进入SYN_SEND状态,等待服务器确认;
第二次握手:服务器收到syn包,必须确认客户的SYN(ack=j+1),同时自己也发送一个SYN包(syn=k),即SYN+ACK包,此时服务器进入SYN_RECV状态;
第三次握手:客户端收到服务器的SYN+ACK包,向服务器发送确认包ACK(ack=k+1),此包发送完毕,客户端和服务器进入ESTABLISHED状态,完成三次握手。

HTTP连接:

HTTP协议即超文本传送协议(HypertextTransfer Protocol ),是Web联网的基础,也是手机联网常用的协议之一,HTTP协议是建立在TCP协议之上的一种应用。
HTTP连接最显著的特点是客户端发送的每次请求都需要服务器回送响应,在请求结束后,会主动释放连接。从建立连接到关闭连接的过程称为“一次连接”。
1)在HTTP 1.0中,客户端的每次请求都要求建立一次单独的连接,在处理完本次请求后,就自动释放连接。
2)在HTTP 1.1中则可以在一次连接中处理多个请求,并且多个请求可以重叠进行,不需要等待一个请求结束后再发送下一个请求。

Socket(套接字):

Socket是对TCP/IP协议的封装,Socket本身并不是协议,而是一个调用接口(API),通过Socket,我们才能使用TCP/IP协议。

建立socket连接:

建立Socket连接至少需要一对套接字,其中一个运行于客户端,称为ClientSocket,另一个运行于服务器端,称为ServerSocket。
套接字之间的连接过程分为三个步骤:服务器监听,客户端请求,连接确认。
服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求。
客户端请求:指客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述它要连接的服务器的套接字,指出服务器端套接字的地址和端口号,然后就向服务器端套接字提出连接请求。
连 接确认:当服务器端套接字监听到或者说接收到客户端套接字的连接请求时,就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发给客户 端,一旦客户端确认了此描述,双方就正式建立连接。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。

区别:
这三者在TCP/IP协议族中的位置关系:


image
HTTP是应用层的协议,更靠近用户端;
TCP是传输层的协议;
而socket是从传输层上抽象出来的一个抽象层,本质是接口。

1、TCP连接与HTTP连接的区别
HTTP是基于TCP的,客户端往服务器发送一个HTTP请求时
七、修改127.0.0.1访问localhost
命令:cat /etc/hosts ----查看hosts文件的内容
修改hosts文件的内容:

sudo vi /etc/hosts
输入i
输入127.0.0.1 www.xjl.com
按esc 按: 然后输入wq ----保存并且退出

判断服务器是否开启的命令:

telnet 127.0.0.1 80
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容

  • 1.项目经验 2.基础问题 3.指南认识 4.解决思路 ios开发三大块: 1.Oc基础 2.CocoaTouch...
    阳光的大男孩儿阅读 4,969评论 0 13
  • 版权声明:本文为博主原创文章,未经博主允许不得转载。 1. OC中,与alloc语义相反的方法是dealloc还是...
    LeaderBiao阅读 2,040评论 0 6
  • 循环引用:http://ios.jobbole.com/82077/类别的作用功能:1.扩充现有类的功能2.对现有...
    得一切从简阅读 492评论 0 1
  • 1.自定义一个delegate 代理的作用:完成委托方交给的任务,委托方有一些任务自己不想完成,但是还需要要实现,...
    yangliangliang阅读 436评论 0 0
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,121评论 29 470