Создать новый образ в файловой системе из System.Drawing.Image?

2

Хорошо, извините, это, наверное, вопрос о нобе, но я немного застрял.

Итак, что я делаю (в моем приложении asp.net) загружается изображение из файловой системы:

System.Drawing.Image tempImage;
tempImage = System.Drawing.Image.FromFile(HttpContext.Server.MapPath(originalPath));

Затем я немного изменяю размер:

tempImage = my awesomeResizingFunction(tempImage, newSize);

и намерены сохранить его в файловой системе в другом месте, используя следующее:

string newPath = "/myAwesomePath/newImageName.jpg";
tempImage.Save(newPath);

и я получаю эту ошибку:

"A generic error occurred in GDI+."

Я знаю, что изображение "нормально", потому что я могу написать его в браузере и посмотреть изображение с измененным размером, я получаю сообщение об ошибке, когда пытаюсь его сохранить. Я новичок и застрял, я делаю это совершенно неправильно? (Ну, я думаю, это очевидно, но вы знаете, что я имею в виду...)

Теги:
thumbnails
gdi+
image-resizing

3 ответа

5
Лучший ответ

PLZ попробовать этот код... Я использовал тот же код для изменения размера изображения и сохранения. если у вас возникнут какие-либо проблемы, тогда plz сообщит мне.

System.Drawing.Bitmap bmpOut = new System.Drawing.Bitmap(NewWidth, NewHeight);
    System.Drawing.Graphics g = System.Drawing.Graphics.FromImage(bmpOut);
    g.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.HighQualityBicubic;
    g.FillRectangle(System.Drawing.Brushes.White, 0, 0, NewWidth, NewHeight);
    g.DrawImage(new System.Drawing.Bitmap(fupProduct.PostedFile.InputStream), 0, 0, NewWidth, NewHeight);
    MemoryStream stream = new MemoryStream();
    switch (fupProduct.FileName.Substring(fupProduct.FileName.IndexOf('.') + 1).ToLower())
    {
        case "jpg":
            bmpOut.Save(stream, System.Drawing.Imaging.ImageFormat.Jpeg);
            break;
        case "jpeg":
            bmpOut.Save(stream, System.Drawing.Imaging.ImageFormat.Jpeg);
            break;
        case "tiff":
            bmpOut.Save(stream, System.Drawing.Imaging.ImageFormat.Tiff);
            break;
        case "png":
            bmpOut.Save(stream, System.Drawing.Imaging.ImageFormat.Png);
            break;
        case "gif":
            bmpOut.Save(stream, System.Drawing.Imaging.ImageFormat.Gif);
            break;
    }
    String saveImagePath = Server.MapPath("../") + "Images/Thumbnail/" + fupProduct.FileName.Substring(fupProduct.FileName.IndexOf('.'));
    bmpOut.Save(saveImagePath);

где fupProduct - идентификатор управления файловой загрузкой

  • 1
    Вы можете улучшить это, используя Path.GetExtension вместо IndexOf ('.'), Хотя вам придется немного изменить переключатель msdn.microsoft.com/en-us/library/…
  • 0
    Спасибо РичардОД .....
Показать ещё 1 комментарий
2

Вы уверены, что originalPath и newPath указывают на разные файлы? Когда вы используете Image.FromFile, файл остается заблокированным, пока вы не вызовете Dispose на изображении, что может привести к указанному вами исключению. Вместо этого вы можете загрузить изображение:

Image tempImage = null;
using (FileStream fs = new FileStream(originalPath, FileMode.Open, FileAccess.Read))
{
    tempImage = Image.FromStream(fs);
}
...

Этот подход гарантирует, что файл будет закрыт в конце блока использования

0

Возможно ли, что исходный поток, поддерживающий исходное изображение, был закрыт? Если поток за Bitmap был закрыт, вы начинаете получать ошибки GDI+. Я столкнулся с этим, когда мы добавили обработку изображений на наш сайт.

Если вы открываете объект Bitmap в отладчике Visual Studio, вы видите исключения вместо значений свойств? Если это так, это не проблема с операцией сохранения, но уровень GDI + потерял способность обрабатывать объект, период.

Что я нашел, мне нужно было отслеживать MemoryStreams, принадлежащие моим растровым изображениям, и держать их вместе. Изменение размера изображения привело к созданию нового MemoryStream с новым растровым изображением.

В итоге я создал этот простой класс (обрезал некоторые дополнительные свойства, которые здесь не нужны):

public class UploadedImage : IDisposable
{
    private Bitmap _img = null;
    private Stream _baseStream = null;


    /// <summary>
    /// The image object.  This must always belong to BaseStream, or weird things can happen.
    /// </summary>
    public Bitmap Img
    {
        [DebuggerStepThrough]
        get { return _img; }
        [DebuggerStepThrough]
        set { _img = value; }
    }

    /// <summary>
    /// The stream that stores the image.  This must ALWAYS belong to Img, or weird things can happen.
    /// </summary>
    public Stream BaseStream
    {
        [DebuggerStepThrough]
        get { return _baseStream; }
        [DebuggerStepThrough]
        set { _baseStream = value; }
    }

    [DebuggerStepThrough]
    public void Dispose()
    {
        if (Img != null)
            Img.Dispose();

        if (BaseStream != null)
            BaseStream.Close();

        _attached = false;
    }
}

Теперь я имел дело с изображениями, загруженными на наш веб-сайт, и я обнаружил, что, когда Asp.Net переработал поток, связанный с запросом, все внезапные операции с изображениями начали сбрасываться. Итак, мое решение, было ли это лучшим способом сделать это или нет, состояло в том, чтобы скопировать данные из загружаемого потока в мой собственный MemoryStream, загрузить изображение из этого и вставить в этот контейнер. И везде, где я создал новое изображение из старого, я всегда держал поток и изображение вместе.

Надеюсь, это поможет.

EDIT: Мне также интересно узнать, как вы изменяете размер изображения. Это фрагмент того, как я сделал:

temp = new Bitmap(newWidth, newHeight, PIXEL_FORMAT);
temp.SetResolution(newHorizontalRes, newVerticalRes);
gr = Graphics.FromImage(temp);

//
// This copies the active frame from 'img' to the new 'temp' bitmap.
// Also resizes it and makes it super shiny.  Sparkle on, mr image dude.
// 
Rectangle rect = new Rectangle(0, 0, newWidth, newHeight);
gr.InterpolationMode = InterpolationMode.HighQualityBicubic;
gr.SmoothingMode = SmoothingMode.HighSpeed;
gr.PageUnit = GraphicsUnit.Pixel;
gr.DrawImage(img, rect);

//
// Image copied onto the new bitmap.  Save the bitmap to a fresh memory stream.
//
retval = new UploadedImage();
retval.BaseStream = (Stream)(new MemoryStream());

temp.Save(retval.BaseStream, ImageFormat.Jpeg);
retval.Img = temp;
  • 0
    Обратите внимание: ( msdn.microsoft.com/en-us/library/system.drawing.aspx )> Классы в пространстве имен System.Drawing не поддерживаются для использования в службе Windows или ASP.NET.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню