大家好,我是阿赵。
这次来讲一下在unity引擎里面,图片和base64字符串的互相转换问题。
一、图片传输的多种方式
有时候我们需要把图片通过网络传输发送。
在Unity里面,有不止一种方式可以实现,比如说,把图片的bytes通过WWWForm的方式来发送,这是Unity官方文档里面的方法:
using UnityEngine;
using UnityEngine.Networking;
using System.Collections;
public class WWWFormImage : MonoBehaviour
{
public string screenShotURL= "http://www.my-server.com/cgi-bin/screenshot.pl";
// Use this for initialization
void Start()
{
StartCoroutine(UploadPNG());
}
IEnumerator UploadPNG()
{
// We should only read the screen after all rendering is complete
yield return new WaitForEndOfFrame();
// Create a texture the size of the screen, RGB24 format
int width = Screen.width;
int height = Screen.height;
var tex = new Texture2D( width, height, TextureFormat.RGB24, false );
// Read screen contents into the texture
tex.ReadPixels( new Rect(0, 0, width, height), 0, 0 );
tex.Apply();
// Encode texture into PNG
byte[] bytes = tex.EncodeToPNG();
Destroy( tex );
// Create a Web Form
WWWForm form = new WWWForm();
form.AddField("frameCount", Time.frameCount.ToString());
form.AddBinaryData("fileUpload", bytes, "screenShot.png", "image/png");
// Upload to a cgi script
using (var w = UnityWebRequest.Post(screenShotURL, form))
{
yield return w.SendWebRequest();
if (w.isNetworkError || w.isHttpError) {
print(w.error);
}
else {
print("Finished Uploading Screenshot");
}
}
}
}
但有时候传输的双方并不是都能自己决定,比如和第三方的服务商做通讯,他们只能支持明文字符串的发送。这个时候,也可以把图片转换成字符串,通过发送字符串来达到目的。
二、图片转Base64字符串
比如我现在在Unity里面有这么一张图片
值得注意的是,在Unity里面,需要读取图片的字节内容,必须是把Read/Write Enabled勾上,然后为了转换的时候是无损的,最好是把图片压缩关掉。把Power of 2关掉,以保持图片是原始分辨率。
然后上代码:
static public string TexToString(Texture2D tex)
{
byte[] bs = tex.EncodeToPNG();
string base64String = System.Convert.ToBase64String(bs);
base64String = “data:image/png;base64,” + base64String;
return base64String;
}
代码很简单,用EncodeToPNG把Texture2D转成byte[],然后再通过System.Convert.ToBase64String转成字符串而已。
需要注意的有2点:
1、为什么要加上”data:image/png;base64,”?
这是一个标准,加上这个头信息,会让接收方知道,现在收到的字符串是一张图片,png格式,并且转换成base64字符串了。如果收发双方都是你自己,你也可以不用加这个信息。
这个信息对于图片解码本身是无意义的,所以接收方在把字符串转图片的时候,还需要把这段去掉。
2、base64包含特殊字符的问题
由于base64字符串会包含一些特殊字符,比如”+”、”/”、”=”,这些字符串如果作为url的明文来发送,是会有问题的。
为了解决这个问题,在发送的时候可以:
1.使用URLEncode转换
2.做字符替换
我们这里不讨论这个特殊字符的问题。
指定图片,运行之后,刚才那张图片就变成了这堆字符串了:

三、Base64转图片
从字符串转换回Texture2D:
static public Texture2D Base64StringToTex(string base64String)
{
base64String = base64String.Replace("data:image/png;base64,", "").Replace("data:image/jgp;base64,", "").Replace("data:image/jpg;base64,", "").Replace("data:image/jpeg;base64,", "");
byte[] bytes = System.Convert.FromBase64String(base64String);
Texture2D tex = new Texture2D(2, 2);
tex.LoadImage(bytes);
return tex;
}
代码同样很简单,一开始的时候,将base64头部信息替换掉,保证下面转换的是纯base64的字符串。
然后通过System.Convert.FromBase64String把字符串转换回byte[],最后通过LoadImage把byte[]加载成Texture2D。
四、当Base64字符串不完整的处理
上面都是常规方法,下面是关于一些原理上的东西了。
假如在某些情况下,base64的字符串的末尾出现了缺失,那么将会出现问题。在System.Convert.FromBase64String的时候,会报错:
FormatException: Invalid length for a Base-64 char array or string.
这是因为Base64字符串的长度应该是4的倍数的,但由于缺失了字符串,导致长度不对。
这个时候,就需要把base64字符串补齐:
base64String = base64String.Replace("data:image/png;base64,", "").Replace("data:image/jgp;base64,", "").Replace("data:image/jpg;base64,", "").Replace("data:image/jpeg;base64,", "");
int mod = base64String.Length % 4;
if(mod>0&&mod>2)
{
base64String += new string('=', 4 - mod);
}
else
{
if(mod>0)
{
base64String += new string('A', 4 - mod);
}
}
为什么是这样补?在base64里面,等号=代表的是空,A代表的是0。当字符串长度不是4的倍数,我们求余数,如果余数是2,我们就补2个空,也就是加2个等号。但这里有一个规则,等号最多只能是2个,如果超过2个,就会报错:
FormatException: The input is not a valid Base-64 string as it
contains a non-base 64 character, more than two padding characters, or
an illegal character among the padding characters.
所以如果缺超过2个字符的时候,就已经不是补等号可以解决了,所以我这里直接补0。补0当然也是不对的,不过不用急,由于补空已经不能解决的时候,其实补什么都是不行的,所以下面还需要后续的处理。
先来了解一下,假如一张图片EncodeToPNG转换成byte[]时,这堆byte[]有什么特点?
首先,这堆byte[]会是以这些byte开头的:
137,80,78,71,13,10,26,10,0,0,0,13,73,72,68,82,0,0
然后,到了最后,是以这些byte结尾的:
0,0,0,0,73,69,78,68,174,66,96,130
知道了这个规则之后,我们就可以做这样的处理了:
1、由于开头一般不会缺失,所以我们可以不管
2、后面这一段固定的结尾,如果缺了,Unity会解析不出来,多了无所谓。所以我们可以针对转换出来的byte[]作检查,看看是不是包含0,0,0,0,73,69,78,68,174,66,96,130的。只要包含就行,不需要一定以它结尾。后面多出来的byte不影响解析。
3、如果发现不包含完整的结尾byte,那么就检查一下是不是缺了某一段,或者是整段没有了。然后,把它们补齐就行了。
4、由于结尾是12个字节,对应的是9个字符,所以base64缺失在9个字符以内,都可以这样补齐。
于是代码会变成这样:
static public Texture2D Base64StringToTex(string base64String)
{
base64String = base64String.Replace("data:image/png;base64,", "").Replace("data:image/jgp;base64,", "").Replace("data:image/jpg;base64,", "").Replace("data:image/jpeg;base64,", "");
int mod = base64String.Length % 4;
if (mod > 0 && mod > 2)
{
base64String += new string('=', 4 - mod);
}
else
{
if (mod > 0)
{
base64String += new string('A', 4 - mod);
}
}
byte[] bytes = System.Convert.FromBase64String(base64String);
bytes = CheckFormatBytes(bytes);
Texture2D tex = new Texture2D(2, 2);
tex.LoadImage(bytes);
return tex;
}
static private byte[] endBs = new byte[]{ 73, 69, 78, 68, 174, 66, 96, 130 };
static public byte[] CheckFormatBytes(byte[] bs)
{
List<byte> origList = new List<byte>();
for(int i = 0;i<bs.Length;i++)
{
origList.Add(bs[i]);
}
int lastIndex = -1;
int checkIndex = -1;
bool isFormat = true;
int firstIndex = -1;
for(int i = 0;i<endBs.Length;i++)
{
if(i==0)
{
lastIndex = origList.LastIndexOf(endBs[i]);
if(lastIndex<0)
{
isFormat = false;
break;
}
else
{
firstIndex = lastIndex;
checkIndex = i;
}
}
else
{
int curIndex = origList.LastIndexOf(endBs[i]);
if(curIndex<0||curIndex-lastIndex!=1)
{
isFormat = false;
break;
}
else
{
checkIndex = i;
lastIndex = curIndex;
}
}
}
if(isFormat == true)
{
return bs;
}
else
{
if(checkIndex>=0)
{
origList.RemoveRange(firstIndex, origList.Count - firstIndex);
}
for (int i = 0; i < endBs.Length; i++)
{
origList.Add(endBs[i]);
}
}
return origList.ToArray();
}