二进制数据、读写文件流、Base64编码

最近有个项目,是通过Selenium WebDriver驱动Chrome浏览报表生成截图,并把截图文件字节流以Base64编码成字符串写入DB的varchar(max)表字段,先看相关代码段:

......
var ss = webDriver.GetScreenshot();
imgString = ss.AsBase64EncodedString;
webDriver.ExecuteScript(string.Format("saveSnapshot('{0}','{1}')", jobId, imgString));
......

这里面还涉及到字符串的Base64编码、解码, 

C#提供的方法有:

  • string   Convert.ToBase64String(byte[] inArray);
  • byte[]   Convert.FromBase64String(string s)

javascript相关的方法有:

  • window.btoa(str)  —— base-64 编码字符串
  • window.atob(encodedStr) ——解码使用 base-64 编码的字符串
private System.IO.MemoryStream GetImage(string base64Img)
{
    if (string.IsNullOrWhiteSpace(base64Img)) return new System.IO.MemoryStream();
    var bty = Convert.FromBase64String(base64Img);
    return new System.IO.MemoryStream(bty);
}

静下来,突然想把平时项目中可能用得上的读取二进制文件流、编码保存做个收集汇总。

 

一、读取二进制数据文件(如图像),转换成BASE64字符串的方法

https://blog.csdn.net/jqrsdsy/article/details/5980543

1. 读取图像,进行Base64编码

            //编码例子
            System.IO.FileStream fs = System.IO.File.OpenRead("c://1.jpg");
            byte[] dt = new byte[fs.Length];
            fs.Read(dt, 0, (int)fs.Length);
            fs.Close();
            string s = Convert.ToBase64String(dt);
            fs = System.IO.File.OpenWrite("c://1.b64");
            dt = Encoding.Default.GetBytes(s);
            fs.Write(dt, 0, dt.Length);
            fs.Flush();
            fs.Close();
            

2. 解码Base64编码的文件,还原出图像文件 

            //解码例子
            System.IO.FileStream fs = System.IO.File.OpenRead("c://1.b64");
            byte[] dt = new byte[fs.Length];
            fs.Read(dt, 0, (int)fs.Length);
            string s = Encoding.Default.GetString(dt);
            dt = Convert.FromBase64String(s);
            fs = System.IO.File.OpenWrite("c://2.jpg");
            fs.Write(dt, 0, dt.Length);
            fs.Close();

 

二、对文件流的读写

https://www.cnblogs.com/lucky_dai/archive/2011/05/06/2038736.html

1. 写文件

            using (FileStream fs = File.Open("test.txt", FileMode.OpenOrCreate))
            {
                BinaryWriter bw = new BinaryWriter(fs, Encoding.UTF8);
                for (int i = 0; i < 10000000; i++)  // 写入一千万行数据
                {
                    bw.Write(100.8); // 写入一个Double
                    bw.Write("abdef");  // 写入一个String
                    bw.Write("poiu");  // 再写一个String
                    bw.Write((float)88.9);    // 写入一个float
                }
            }

2. 读取上面写入的文件,注意要对应写入数据类型要相匹配,读取顺序要相匹配

            using (FileStream fs = File.Open("test.txt", FileMode.Open))
            {
                BinaryReader br = new BinaryReader(fs, Encoding.UTF8);
                while (br.BaseStream.Length > br.BaseStream.Position) // 判断是否已读完
                {
                    Console.WriteLine(br.ReadDouble());
                    Console.WriteLine(br.ReadString());
                    Console.WriteLine(br.ReadString());
                    Console.WriteLine(br.ReadSingle());
                }
            }

“流”是读取文件的一般手段,那么你会用它读取文件中的数据了么?真的能读完全么?

通常我们读取一个文件使用如下的步骤:

1、声明并使用File的OpenRead实例化一个文件流对象,就像下面这样

       FileStream fs = File.OpenRead(filename);

或者

       FileStream fs = FileStream(filename, FileMode.Open, FileAccess.Read, FileShare.Read);

2、准备一个存放文件内容的字节数组,fs.Length将得到文件的实际大小,就像下面这样

       byte[] data = new byte[fs.Length];

3、哇!开始读了,调用一个文件流的一个方法读取数据到data数组中

       fs.Read (data, 0, data.Length);

只写了3句就可以把文件里面的内容原封不动的读出来,真是太简洁了!可以这段代码真的能像你预期的那样工作么?答案是:几乎可以!
在大部分情况下上面的代码工作的很好,但是我们应该注意Read方法是有返回值的,既然有返回值那么一定有其道理,如果按照上面的写法完全可以是一个没有返回值的函数。我想返回值的目的是,为了给我们一个机会判断实际读 取文件的大小,从而来判断文件是否已经完全读完。所以上面的代码不能保证我们一定读完了文件里面的所有字节(虽然在很多情况下是读完了)。下面的方法提供 了一个比上面方法更安全的方法,来保证文件被完全读出

public static void SafeRead (Stream stream, byte[] data){
    int offset=0;
    int remaining = data.Length;
    // 只要有剩余的字节就不停的读
    while (remaining > 0){
        int read = stream.Read(data, offset, remaining);
        if (read <= 0)
            throw new EndOfStreamException("文件读取到"+read.ToString()+"失败!");
        
        remaining -= read;   // 减少剩余的字节数        
        offset += read;      // 增加偏移量
    }
}

有些情况下你不知道流实际的长度比如:网络流。此时可以使用类似的方法读取流直到流里面的数据完全读取出来为止。我们可以先初始化一段缓存,再将流读出来的流信息写到内存流里面,就像下面这样:

public static byte[] ReadFully (Stream stream){
    // 初始化一个32k的缓存
    byte[] buffer = new byte[32768];
    using (MemoryStream ms = new MemoryStream()){ //返回结果后会自动回收调用该对象的Dispose方法释放内存
        // 不停的读取
        while (true){
            int read = stream.Read (buffer, 0, buffer.Length);
            // 直到读取完最后的3M数据就可以返回结果了
            if (read <= 0)
                return ms.ToArray();
            ms.Write (buffer, 0, read);
        }
    }
}

虽然上面的例子都比较简单,效果也不是很明显(大部分都是对的),也许你早就会了,没关系这篇文章本来就是写给初学者的。

下面的方法提供了一种使用指定缓存长度的方式读取流,虽然在很多情况下你可以直接使用Stream.Length得到流的长度,但是不是所有的流都可以得到。

public static byte[] Read2Buffer (Stream stream, int BufferLen){
       // 如果指定的无效长度的缓冲区,则指定一个默认的长度作为缓存大小
       if (BufferLen < 1){
              BufferLen = 0x8000;
       }
       // 初始化一个缓存区
       byte[] buffer = new byte[BufferLen];
       int read=0;  
       int block;
       // 每次从流中读取缓存大小的数据,知道读取完所有的流为止
       while ( (block = stream.Read(buffer, read, buffer.Length-read)) > 0){
              // 重新设定读取位置
              read += block;    

              // 检查是否到达了缓存的边界,检查是否还有可以读取的信息
              if (read == buffer.Length){
                     // 尝试读取一个字节
                     int nextByte = stream.ReadByte();         

                     // 读取失败则说明读取完成可以返回结果
                     if (nextByte==-1){
                            return buffer;
                     }         

                     // 调整数组大小准备继续读取
                     byte[] newBuf = new byte[buffer.Length*2];
                     Array.Copy(buffer, newBuf, buffer.Length);
                     newBuf[read]=(byte)nextByte;
                     buffer = newBuf;// buffer是一个引用(指针),这里意在重新设定buffer指针指向一个更大的内存
                     read++;
              }
       }

       // 如果缓存太大则使用ret来收缩前面while读取的buffer,然后直接返回
       byte[] ret = new byte[read];
       Array.Copy(buffer, ret, read);
       return ret;
}

 

你可能感兴趣的:(.Net,Javascript,文件操作)