zoukankan      html  css  js  c++  java
  • 浅析GET和POST请求的本质区别以及关于get请求的长度限制到底是多少的问题

      GET和POST是HTTP请求的两种基本方法,要说它们的区别,接触过WEB开发的人都能说出一二。最直观的区别就是GET把参数包含在URL中,POST通过request body传递参数。你可能自己写过无数个GET和POST请求,或者已经看过很多权威网站总结出的他们的区别,你非常清楚知道什么时候该用什么。当你在面试中被问到这个问题,你的内心充满了自信和喜悦。你轻轻松松的给出了一个“标准答案”:

    • GET在浏览器回退时是无害的,而POST会再次提交请求。
    • GET产生的URL地址可以被Bookmark,而POST不可以。
    • GET请求会被浏览器主动cache,而POST不会,除非手动设置。
    • GET请求只能进行url编码,而POST支持多种编码方式。
    • GET请求参数会被完整保留在浏览器历史记录里,而POST中的参数不会被保留。
    • GET请求在URL中传送的参数是有长度限制的,而POST么有。
    • 对参数的数据类型,GET只接受ASCII字符,而POST没有限制。
    • GET比POST更不安全,因为参数直接暴露在URL上,所以不能用来传递敏感信息。
    • GET参数通过URL传递,POST放在Request body中。

      (本标准答案参考自w3schools)

      “很遗憾,这不是我们要的回答!”,请告诉我真相。。。

      如果我告诉你GET和POST本质上没有区别你信吗?让我们扒下GET和POST的外衣,坦诚相见吧!

    一、GET和POST是什么?

      GET和POST是HTTP协议中的两种发送请求的方法。

      HTTP是什么?HTTP是基于TCP/IP的关于数据如何在万维网中如何通信的协议。

      HTTP的底层是TCP/IP。所以GET和POST的底层也是TCP/IP,也就是说,GET/POST都是TCP链接。GET和POST能做的事情是一样一样的。你要给GET加上request body,给POST带上url参数,技术上是完全行的通的。

      那么,“标准答案”里的那些区别是怎么回事?

    1、交通规则HTTP与运输汽车TCP

      在我大万维网世界中,TCP就像汽车,我们用TCP来运输数据,它很可靠,从来不会发生丢件少件的现象。但是如果路上跑的全是看起来一模一样的汽车,那这个世界看起来是一团混乱,送急件的汽车可能被前面满载货物的汽车拦堵在路上,整个交通系统一定会瘫痪。

      为了避免这种情况发生,交通规则HTTP诞生了。HTTP给汽车运输设定了好几个服务类别,有GET, POST, PUT, DELETE等等,HTTP规定,当执行GET请求的时候,要给汽车贴上GET的标签(设置method为GET),而且要求把传送的数据放在车顶上(url中)以方便记录。如果是POST请求,就要在车上贴上POST的标签,并把货物放在车厢里。

      当然,你也可以在GET的时候往车厢内偷偷藏点货物,但是这是很不光彩;也可以在POST的时候在车顶上也放一些数据,让人觉得傻乎乎的。HTTP只是个行为准则,而TCP才是GET和POST怎么实现的基本

      但是,我们只看到HTTP对GET和POST参数的传送渠道(url还是requrest body)提出了要求。“标准答案”里关于参数大小的限制又是从哪来的呢?

    2、运输公司

      在我大万维网世界中,还有另一个重要的角色:运输公司不同的浏览器(发起http请求)和服务器(接受http请求)就是不同的运输公司。虽然理论上,你可以在车顶上无限的堆货物(url中无限加参数)。

      但是运输公司可不傻,装货和卸货也是有很大成本的,他们会限制单次运输量来控制风险,数据量太大对浏览器和服务器都是很大负担。业界不成文的规定是,(大多数)浏览器通常都会限制url长度在2K个字节,而(大多数)服务器最多处理64K大小的url。

      超过的部分,恕不处理。如果你用GET服务,在request body偷偷藏了数据,不同服务器的处理方式也是不同的,有些服务器会帮你卸货,读出数据,有些服务器直接忽略,所以,虽然GET可以带request body,也不能保证一定能被接收到哦。

      好了,现在你知道,GET和POST本质上就是TCP链接,并无差别。但是由于HTTP的规定和浏览器/服务器的限制,导致他们在应用过程中体现出一些不同。

      你以为本文就这么结束了?我们的大BOSS还等着出场呢。。。这位BOSS有多神秘?当你试图在网上找“GET和POST的区别”的时候,那些你会看到的搜索结果里,从没有提到他。他究竟是什么呢。。。

    二、GET和POST还有一个重大区别

    1、简单的说:GET产生一个TCP数据包;POST产生两个TCP数据包。

    2、长的说:

    • 对于GET方式的请求,浏览器会把http header和data一并发送出去,服务器响应200(返回数据);
    • 而对于POST,浏览器先发送header,服务器响应100 continue,浏览器再发送data,服务器响应200 ok(返回数据)。

      也就是说,GET只需要汽车跑一趟就把货送到了,而POST得跑两趟,第一趟,先去和服务器打个招呼“嗨,我等下要送一批货来,你们打开门迎接我”,然后再回头把货送过去。

      因为POST需要两步,时间上消耗的要多一点,看起来GET比POST更有效。因此Yahoo团队有推荐用GET替换POST来优化网站性能。但这是一个坑!跳入需谨慎。为什么?因为:

    (1)GET与POST都有自己的语义,不能随便混用。

    (2)据研究,在网络环境好的情况下,发一次包的时间和发两次包的时间差别基本可以无视。而在网络环境差的情况下,两次包的TCP在验证数据包完整性上,有非常大的优点。

    (3)并不是所有浏览器都会在POST中发送两次包,Firefox就只发送一次。

    作者:巅峰大词典

    https://segmentfault.com/a/1190000038556123

    三、总结

    1、常见区别:回退是否再次请求、是否被bookmark、是否主动缓存、是否支持多种编码方式、请求参数是否保留在历史记录里、传送参数的长度限制、对参数的数据类型、安全性差别、请求参数的放置位置

    2、通过概念理解:http - 交通规则、tcp - 运输汽车、请求方式(get、post、delete等)- 不同运输汽车不同紧急货物、浏览器和服务器通讯 - 两家运输公司对接、参数长度限制 - 堆货卸货成本

    3、重大区别:get只发一个tcp包(一次性将header和data带过去)、post要发两个tcp包(先带header过去-我要送货来了,请准备;再将data带过去-送货来了)

    四、关于get请求的长度限制到底是多少

      经常看到听到说 get 有长度限制,1024B。其实是错误的。下面把从权威官网的解释复制过来,以做更正:

    1、Http get 方法提交的数据大小长度并没有限制,Http协议规范没有对URL长度进行限制

      目前说的get长度有限制,是特定的浏览器及服务器对它的限制。各种浏览器和服务器的最大处理能力如下:
            IE:对URL的最大限制为2083个字符,若超出这个数字,提交按钮没有任何反应。
            Firefox:对Firefox浏览器URL的长度限制为:65536个字符。
            Safari:URL最大长度限制为80000个字符。
            Opera:URL最大长度限制为190000个字符。
            Google(chrome):URL最大长度限制为8182个字符。
            Apache(Server):能接受的最大url长度为8192个字符(这个准确度待定???)
            Microsoft Internet Information Server(IIS):能接受最大url的长度为16384个字符。

    2、理论上讲,post 是没有大小限制的。Http协议规范也没有进行大小限制,起限制作用的是服务器处理程序的处理能力

      Tomcat下默认post长度为2M,可通过修改 conf/server.xml 中的“maxPostSize=0”来取消对post大小的限制。
      注意:(若长度超限,则服务端返回414标识)
    (1)首先即使有长度限制,也是限制的是整个URI长度,而不仅仅是你的参数值数据长度。
    (2)HTTP协议从未规定GET/POST的请求长度限制是多少
    (3)所谓的请求长度限制是由浏览器和web服务器决定和设置的,浏览器和web服务器的设定均不一样,这依赖于各个浏览器厂家的规定或者可以根据web服务器的处理能力来设定。

    五、GET VS POST扩展

    1、多数浏览器对于POST采用两阶段发送数据的,先发送请求头,再发送请求体,即使参数再少再短,也会被分成两个步骤来发送(相对于GET),也就是第一步发送header数据,第二部再发送body部分。Http是应用层的协议,而再传输层有些情况TCP会出现两次连结的过程,http协议本身不保存状态信息,一次请求一次响应。对于TCP而言,通信次数越多反而可靠性越低,能在一次连结中传输完需要的信息是最可靠的,所以尽量使用GET请求来减少网络耗时。如果通信时间增加,这段时间客户端于服务器端一直保持连接状态,在服务器侧负载可能会增加,可靠性会下降。

    2、GET请求能够被cache,GET请求能够被保存在浏览器的浏览历史里面(密码等重要数据GET提交,别人查看历史记录,就可以直接看到这些私密数据)POST不进行缓存。

    3、GET参数是带在URL后面,传统IE中URL的最大可用长度为2048字符,其他浏览器对URL长度限制实现上有所不同。POST请求无长度限制(目前理论上是这样)。

    4、GET提交的数据大小,不同浏览器的限制不同,一般在2k-8k之间,POST提交数据比较大,大小靠服务器的设定值限制,而且某些数据只能用POST方法【携带】,比如file。

    5、全部用POST不是十分合理,最好先把请求按功能和场景分下类,对数据请求频繁,数据不敏感且数据量在普通浏览器最小限定的2k范围内,这种情况使用GET。其他地方使用POST。

    6、GET的本质是【得】,而POST的本质是【给】。而且,GET是【幂等】的,在这一点上,GET被认为是【安全的】。实际上server端也可以用作资源更新,但是这种用法违反了约定,容易造成CSRF(跨站请求伪造)。

  • 相关阅读:
    PyQt(Python+Qt)学习随笔:使用pyqtConfigure建立信号和槽的连接
    PyQt(Python+Qt)学习随笔:调用disconnect进行信号连接断开时的信号签名与断开参数的匹配要求
    PyQt(Python+Qt)学习随笔:什么是信号绑定(Unbound and Bound Signals)?
    PyQt(Python+Qt)学习随笔:信号签名(signature of the signal)是什么?
    第六章、信号和槽进阶--自定义信号及其他信号、槽的高级特性
    第15.19节 PyQt(Python+Qt)入门学习:自定义信号与槽连接
    第五章、信号和槽的实战应用--一个计算器的实现
    第四章 、PyQt中的信号(signal)和槽(slot)机制以及Designer中的使用
    第三章 、使用Qt Designer进行GUI设计
    织梦通过 phpmyadmin 导出的数据,再次导入的时候报错
  • 原文地址:https://www.cnblogs.com/goloving/p/14321819.html
Copyright © 2011-2022 走看看