带压缩的可保存图像库?

时间:2012-06-11 11:46:11

标签: c# image compression image-compression imagelibrary

这可能是不可能的,或者更确切地说,不是很受欢迎,但我想知道如何为图像创建数据文件,实际上会压缩它们?假设我有200MB的图像文件,是否有一些系统可以用来将它们存储在一个文件中,这会将它们压缩到150MB的总大小? (只是示例数字,比率并不重要)。

我知道我可以使用Base64对图像进行编码,然后将它们存储在SQLite数据库中,但我读到以编码形式存储图像实际上会导致比原始图像略大的图像文件本身。

我还在考虑一个ZIP文件,但我不确定它是否可以用作“库”?

如果这样的事情不是预定义的类,那么有人可以引导我走上正轨吗?

这是我正在寻找的模拟:

class ImageLibrary {
  //this is where the code would go for the library?
}

class MyProgram{
  public MyProgram() 
  {
    ImageLibrary library = new ImageLibrary();
    library.Add(<Image object here, with an ID of sorts>);
    library.Add(<Another image object here, also with an ID>);
    Load += new FormLoadEventHandler(MyProgram_Load);
  }

  void MyProgram_Load(object sender, EventArgs e)
  {
    PictureBox.Image = library.Get(<image id here>);
  }
}

我希望这是可能的。否则,我只会忍受稍大的文件大小,Base64对它们进行编码。但是,因为我现在有将要存储的近500张图像,所以保存的kB是一个获得的kB。 :)此外,请不要判断我的代码示例的质量,它只是一个模拟,我把它从袖口写下来。

干杯,并提前谢谢。

2 个答案:

答案 0 :(得分:3)

如果将图像保存为二进制文件将有助于这是我用来将它们转换为二进制文件然后保存到SQLite中的代码:

public byte[] ImageToByte(Image image, System.Drawing.Imaging.ImageFormat format){
        using (MemoryStream ms = new MemoryStream())
        {
            // Convert Image to byte[]
            image.Save(ms, format);
            byte[] imageBytes = ms.ToArray();
            return imageBytes;
        }
    }
    public Image ByteToImage(byte[] imageBytes)
    {
        // Convert byte[] to Image
        MemoryStream ms = new MemoryStream(imageBytes, 0, imageBytes.Length);
        ms.Write(imageBytes, 0, imageBytes.Length);
        Image image = new Bitmap(ms); 
        return image;
    }

然后保存二进制文件:

void SaveImage(byte[] image){
        string conStringDatosUsuarios = @" Data Source = \Program Files\GPS___CAM\Data\DatosUsuarios.s3db ";            
        SQLiteConnection con = new SQLiteConnection(conStringDatosUsuarios); 
        SQLiteCommand cmd = con.CreateCommand();
        cmd.CommandText = String.Format("INSERT INTO Users (Foto) VALUES (@0);");
        SQLiteParameter p = new SQLiteParameter("@0", System.Data.DbType.Binary);
        p.Value = image;
        cmd.Parameters.Add(p);            
        con.Open(); 
        try
        {
            cmd.ExecuteNonQuery();
        }
        catch (Exception exc1)
        {
            MessageBox.Show(exc1.Message);
        }
        con.Close();
    }

希望有所帮助

编辑正如您所问,我正在使用加载图片代码进行更新: (要将字节转换为图像,必须使用ByteToImage函数)

void LoadImage(string tag){
        string query = "SELECT Foto FROM Users;";
        string conString = @" conection to your database ";
        SQLiteConnection con = new SQLiteConnection(conString); 
        SQLiteCommand cmd = new SQLiteCommand(query, con);            
        con.Open();
        try
        {
            SQLiteDataReader rdr = cmd.ExecuteReader();
            try
            {
                while (rdr.Read())
                {
                    pictureBox1.Image = ByteToImage((System.Byte[])rdr[0]);
                }
            }
            catch (Exception exc) { MessageBox.Show(exc.Message); }
        }
        catch (Exception ex) { MessageBox.Show(ex.Message); }
        con.Close();
    }

编辑2 尝试此操作以查看您要加载的数据类型:

System.Type checkType = rdr[0].GetType();
pictureBox1.Image = ByteToImage((System.Byte[])rdr[0]);

在代码中添加第一行并在该行中添加断点。检查checkType的类型。可能它不是二元的。让我知道结果可以帮到你。

答案 1 :(得分:1)

查看此文章:http://msdn.microsoft.com/en-us/library/aa479502.aspx

我希望它有所帮助。从ZIP访问文件的想法。它不是现成的解决方案,而是一个很好的概念解释。