IOS

作者: xiaofengl | 来源:发表于2018-03-01 18:53 被阅读0次

    一、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
    

    相关文章

      网友评论

        本文标题:IOS

        本文链接:https://www.haomeiwen.com/subject/bohnxftx.html