首页 > 其他分享 >【专栏精选】Unity中的HTTP网络通信

【专栏精选】Unity中的HTTP网络通信

时间:2023-04-28 16:34:16浏览次数:36  
标签:网络通信 www HTTP 请求 Unity var UnityWebRequest


本文节选自洪流学堂公众号技术专栏《大话Unity2019》,未经允许不可转载。

洪流学堂公众号回复专栏,查看更多专栏文章。


洪流学堂,让你快人几步。你好,我是郑洪智。

大智:“小新,今天开始,我们来学习Unity中的HTTP通信。”
小新:“为啥不先学Tcp和Udp呢?这俩不是基础么?”
大智:“虽然这俩是基础,但是难度也更大。Http是应用层协议,使用起来会比较方便,可以快速入门。”

Unity中的HTTP编程

Unity中的HTTP通信主要依赖的是Unity自带的UnityWebRequest类(虽然也有一些第三方插件,但一般使用这个就足够了)。

小新:“UnityWebRequest类我们之前学过呀,是不是复习一下就行啦”
大智:“之前我们使用UnityWebRequest类主要用file协议来加载本地的文件,和真正的HTTP请求还有所不同。”

我们用一个最简单的HTTP请求,来学习整个请求的流程。

using System.Collections;
using UnityEngine;
using UnityEngine.Networking;

public class UnityPageRequest : MonoBehaviour
{
    // Start is called before the first frame update
    IEnumerator Start()
    {
        var url = "https://www.baidu.com";
        var www = UnityWebRequest.Get(url);
        yield return www.SendWebRequest();

        if (www.isHttpError || www.isNetworkError)
        {
            Debug.Log(www.error);
        }
        else
        {
            Debug.Log(www.downloadHandler.text);
        }
    }
}

HTTP是请求-响应模型,所以通信的过程主要分为请求部分和响应部分。

HTTP请求(Request)

客户端发送一个HTTP请求到服务器的请求消息包括以下格式:

请求行(request line)、请求头部(header)、空行和请求数据四个部分组成。

HTTP请求方法

本例中我们使用了UnityWebRequest.Get(url)来获取对应的信息。
这里的Get方法对应了HTTP方法中的GET。

什么是HTTP方法呢?

根据HTTP标准,HTTP请求可以使用多种请求方法。
HTTP1.0定义了三种请求方法: GET, POST 和 HEAD方法。
HTTP1.1新增了五种请求方法:OPTIONS, PUT, DELETE, TRACE 和 CONNECT 方法。

  • GET 请求指定的页面信息,并返回实体主体。
  • POST 向指定资源提交数据进行处理请求(例如提交表单或者上传文件)。数据被包含在请求体中。POST请求可能会导致新的资源的建立和/或已有资源的修改。
  • HEAD 类似于get请求,只不过返回的响应中没有具体的内容,用于获取报头
  • PUT 从客户端向服务器传送的数据取代指定的文档的内容。
  • DELETE 请求服务器删除指定的页面。
  • CONNECT HTTP/1.1协议中预留给能够将连接改为管道方式的代理服务器。
  • OPTIONS 允许客户端查看服务器的性能。
  • TRACE 回显服务器收到的请求,主要用于测试或诊断。

通常我们用的比较多的是GET和POST方法。GET一般用来获取信息,POST一般用来向服务器上传信息,比如表单、文件等。

UnityWebRequest类中封装了几种HTTP方法的高级操作:

  • GET对应UnityWebRequest.Get
  • POST对应UnityWebRequest.Post
  • HEAD对应UnityWebRequest.Head
  • PUT对应UnityWebRequest.Put
  • DELETE对应UnityWebRequest.Delete

此外,还能使用底层的API来手动创建UnityWebRequest。

URL

上面的几行代码请求了https://www.baidu.com这一URL对应的资源,返回的数据是一个html文件的文本内容。

URL我们上一节中学过,它是一个资源地址,就本例来说:

  1. 协议部分:该URL的协议部分为“https:”,这代表网页使用的是HTTPS协议,S代表Secure,安全的,是一种加密传输的HTTP协议,在"https"后面的“//”为分隔符。
  2. 域名部分:该URL的域名部分为“www.baidu.com”。一个URL中,也可以使用IP地址作为域名使用
  3. 端口部分:端口不是一个URL必须的部分,如果省略,将采用默认端口80。本例中就省略了端口号。
  4. 虚拟目录部分:从域名后的第一个“/”开始到最后一个“/”为止,是虚拟目录部分。虚拟目录也不是一个URL必须的部分。本例中的虚拟目录是根目录“/”
  5. 文件名部分:文件名部分也不是一个URL必须的部分,可以在服务端配置和开发。

如果你不参与服务端开发的话,一般只需要知道URL地址是什么就可以了。

HTTP headers(头部)

请求头部经常用来设置请求的一些属性参数。

比如我们想告诉服务器给我们返回json格式的数据,通常要设置:

Content-type:application/json

在Unity中需要这么设置:

www.SetRequestHeader("Content-Type", "application/json");

请求数据

有些请求方法如POST和PUT还可以携带请求数据,通常用于上传文件。

在Unity中,请求数据通常以这种方式添加,例如上传一张图片:

IEnumerator UploadTexture()
{
    var tex = new Texture2D(1,1);
    tex.ReadPixels(new Rect(0, 0, Screen.width, Screen.height), 0, 0);
    tex.Apply();
    // tex.Apply() 对性能影响较大,故等待一帧再执行
    yield return null;
    var bytes = tex.EncodeToPNG();
    var form = new WWWForm();
    form.AddBinaryData("screenshot", bytes);
    var www = UnityWebRequest.Post("server url", form);
    yield return www.SendWebRequest();
    // ...
}

HTTP响应(Response)

Http请求的结果都是字节流。

那么为什么我们能看到在浏览器上有的能呈现文字,还有图片、视频呢?

这是因为响应结果中都有Content-Type报头,浏览器会根据报头来解析接收到的数据,显示为不同的内容。

HTTP状态码

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 //服务器当前不能处理客户端的请求,一段时间后可能恢复正常

在Unity中可以使用如下代码获取状态码:

Debug.Log(www.responseCode);

这个状态码也决定了www.isHttpError的返回值

isHttpError = reponseCode >= 400;

Unity中的HTTP响应

但是在Unity中,UnityWebRequest并没有自动帮我们处理这个过程,我们需要手动来处理。

通常我们使用:

var text = www.downloadHandler.text; // 文本信息,使用UTF8编码解析
var bytes = www.downloadHandler.data; // 字节数组

注意www.downloadHandler.text实际上是自动使用UTF8编码解析了字节数组,如果发送的数据不是采用UTF8编码,需要获取字节数组自行解析。

另外在Unity中,Unity结合游戏开发中的常见需求,也加入了请求图片、音频请求的接口。

请求图片:

IEnumerator RequestTexture()
{
    var url = "https://www.baidu.com/img/bd_logo1.png";
    var www = UnityWebRequestTexture.GetTexture(url);
    yield return www.SendWebRequest();
    Debug.Log("status code:" + www.responseCode);
    if (www.isHttpError || www.isNetworkError)
    {
        Debug.Log(www.error);
    }
    else
    {
        var tex = DownloadHandlerTexture.GetContent(www);
    }
}

请求音频:

IEnumerator RequestMedia()
{
    var url = "https://www.xxx.com/music.ogg";
    var www = UnityWebRequestMultimedia.GetAudioClip(url, AudioType.OGGVORBIS);
    yield return www.SendWebRequest();
    Debug.Log("status code:" + www.responseCode);
    if (www.isHttpError || www.isNetworkError)
    {
        Debug.Log(www.error);
    }
    else
    {
        var audio = DownloadHandlerAudioClip.GetContent(www);
    }
}

请求视频:

目前Unity的VideoPlayer中支持直接从file或http协议直接播放视频,如果你想将这个视频下载到本地,可以使用UnityWebRequest.Get方法获取到所有字节再将字节保存到本地。

请求Assetbundle:

IEnumerator InstantiateObject(string assetBundleName)
{
    string uri = "file:///" + Application.dataPath + "/AssetBundles/" + assetBundleName;
    UnityWebRequest request = UnityWebRequestAssetBundle.GetAssetBundle(uri, 0);
    yield return request.SendWebRequest();
    AssetBundle bundle = DownloadHandlerAssetBundle.GetContent(request);
    GameObject cube = bundle.LoadAsset<GameObject>("Cube");
    GameObject sprite = bundle.LoadAsset<GameObject>("Sprite");
    Instantiate(cube);
    Instantiate(sprite);
}

总结

大智:“Unity有一个旧的Web请求API是WWW,但是对HTTP请求不完善。建议你还是学习新的UnityWebRequest类,拥有更好的性能和更完善的HTTP请求。”

今日思考题

大智:“试一试本文中的这些例子。”
小新:“好嘞!”
大智:“收获别忘了分享出来!也别忘了分享给你学Unity的朋友,也许能够帮到他。”


洪流学堂公众号回复专栏,查看更多专栏文章。

《大话Unity2019》,大智带小新学Unity2019的有趣经历,让你学Unity更简单。



标签:网络通信,www,HTTP,请求,Unity,var,UnityWebRequest
From: https://blog.51cto.com/u_5746184/6234805

相关文章

  • Unity开发《一起来捉妖》教程 | 1.陀螺仪控制相机
    洪流学堂,让你快人几步。你好,我是郑洪智。洪流学堂公众号回复捉妖,可以获取本教程的源码工程。大智:“小新,你小子最近是不是谈恋爱了,怎么天天往外跑?”小新:“嘿嘿”大智:“嘿嘿你个鬼啊,从实招来,是不是要请我吃饭了?”小新:“最近有一款非常火的AR游戏,叫《一起来捉妖》,你玩了没?”大智:“听说......
  • [Unity脚本运行时更新]C#7.1新特性
    洪流学堂微信公众号。本文是该系列《Unity脚本运行时更新带来了什么?》的第6篇。洪流学堂公众号回复runtime,获取本系列所有文章。Unity2017-2018.2中的4.x运行时已经支持到C#6,之前的文章已经介绍完毕。Unity2018.3将支持到C#7.3,今天我们先来看看C#7.1新特性能给代码带来什么吧,不过......
  • [Unity脚本运行时更新]C#7.2新特性
    洪流学堂微信公众号。本文是该系列《Unity脚本运行时更新带来了什么?》的第7篇。洪流学堂公众号回复runtime,获取本系列所有文章。Unity2017-2018.2中的4.x运行时已经支持到C#6,之前的文章已经介绍完毕。Unity2018.3将支持到C#7.3,今天我们来看看C#7.2新特性能给代码带来什么吧,不过这......
  • 【解决】axios 下载文件 Failed to read the 'responseText' property from 'XMLHttp
    主要解决以下两个问题问题一:idm一些网站不允许请求同一文件两次故障原因:IDM在发神经因为它检测到浏览器集成插件未安装,所以诱导你安装。实际上,装了插件问题也会出现。改参数都没用。1.很可能是你点击网页的下载链接有问题(换个网页下载试试,就不提示了),Edge浏览器一直会欺......
  • Unity动画系统详解9:Target Matching是什么?
    摘要:在游戏中,经常有这种情况:角色的手或者脚需要在特定时间放在特定的位置。比如角色需要用手撑着跳过一个石头或一堵墙,或者跳起抓住房梁。TargetMatch就是让动画的特定片段去匹配特定的位置。洪流学堂,让你快人几步。你好,我是跟着大智学Unity的萌新,我叫小新,这几周一起来复(yu)习(xi)动......
  • Unity2020.1新功能探路:Profiler相关更新
    洪流学堂,让你快人几步。你好,我是你的技术探路者郑洪智,你可以叫我大智。大智作为探路者带你一块探索一下Unity2020.1里面有什么好玩的东西。这一次Profiler的更新比较大,咱们专门用一篇来看看Profiler方面的更新。主要包含以下几个方面:Profiler作为单独程序启动ProfileAnalyer包的更......
  • Unity+SenseAR教程:用手势发射爱心【源码】
    摘要:Unity+SenseAR2.3开发的AR应用,基于手势识别功能,发射你的爱心~洪流学堂,让你快人几步。你好,我是你的技术探路者郑洪智,你可以叫我大智。今天开头就不絮叨了,一句“名言”送给大家,然后咱们把代码微微的一写。“虽然今天是情人节,但怎会比学习带来更多的多巴胺呢?”——澹深苟最终效果......
  • [独家放送]Unity2020规划预览,可视化编程又双叒叕来了!
    你好,我是你的技术探路者郑洪智,你可以叫我大智。欢迎一起进入2020年,在新的一年里Unity有什么大动作呢?本文带你速览你最关心的Unity2020的核心功能!你最可能关心的功能有哪些呢?Unity2019.3在哪里???(乱入)下面从四个方面来看Unity有哪些更新:核心功能和性能更多的DOTS(Data-OrientedTechSt......
  • 为HttpClient开启HTTP/2
    .NetCore在调用其他服务时,调用通常使用HttpClient,而HttpClient默认使用HTTP/1.1。  配置HttpClient以使用HTTP/2h2连接自.NETCore3.0发布以来,.NET开发人员可以使用HttpClient启用HTTP/2。 设置HTTP/2的方法DefaultRequestVersion和DefaultVersionP......
  • golang常用的http请求操作
    之前用python写各种网络请求的时候写的非常顺手,但是当打算用golang写的时候才发现相对来说还是python的那种方式用的更加顺手,习惯golang的用法之后也就差别不大了,下面主要整理了常用的通过golang发起的GET请求以及POST请求的代码例子golang发起GET请求基本的GET请求//基本的GE......