日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Net core中使用System.Drawing對上傳的圖片流進行壓縮(示例代碼)

瀏覽:363日期:2022-06-08 13:57:32
目錄
  • 直接壓縮圖片
  • 通過文件流壓縮圖片
  • 上傳到七牛云前壓縮圖片
    • 部署問題
    • 在Linux中安裝
  • 產生原因
    • 解決方案

      由于net core 中默認沒有System.Drawing,可以通過nuget下載一個來代替System.Drawing.Common

      直接壓縮圖片

      /// <summary>/// 圖片壓縮/// </summary>/// <param name="sFile">原圖片位置</param>/// <param name="dFile">壓縮后圖片位置</param>/// <param name="dHeight">圖片壓縮后的高度</param>/// <param name="dWidth">圖片壓縮后的寬度</param>/// <param name="flag">圖片壓縮比0-100,數值越小壓縮比越高,失真越多</param>/// <returns></returns>public static bool GetPicThumbnailTest(string sFile, string dFile, int dHeight, int dWidth, int flag){    System.Drawing.Image iSource = System.Drawing.Image.FromFile(sFile);    //如果為參數為0就保持原圖片的高寬嘛(不然想保持原圖外面還要去讀取一次)    if (dHeight == 0)    {dHeight = iSource.Height;    }    if (dWidth == 0)    {dWidth = iSource.Width;    }      ImageFormat tFormat = iSource.RawFormat;    int sW = 0, sH = 0;     //按比例縮放    Size tem_size = new Size(iSource.Width, iSource.Height);     if (tem_size.Width > dHeight || tem_size.Width > dWidth)    {if ((tem_size.Width * dHeight) > (tem_size.Width * dWidth)){    sW = dWidth;    sH = (dWidth * tem_size.Height) / tem_size.Width;}else{    sH = dHeight;    sW = (tem_size.Width * dHeight) / tem_size.Height;}    }    else    {sW = tem_size.Width;sH = tem_size.Height;    }     Bitmap ob = new Bitmap(dWidth, dHeight);    Graphics g = Graphics.FromImage(ob);     g.Clear(Color.WhiteSmoke);    g.CompositingQuality = CompositingQuality.HighQuality;    g.SmoothingMode = SmoothingMode.HighQuality;    g.InterpolationMode = InterpolationMode.HighQualityBicubic;     g.DrawImage(iSource, new Rectangle((dWidth - sW) / 2, (dHeight - sH) / 2, sW, sH), 0, 0, iSource.Width, iSource.Height, GraphicsUnit.Pixel);     g.Dispose();    //以下代碼為保存圖片時,設置壓縮質量     EncoderParameters ep = new EncoderParameters();    long[] qy = new long[1];    qy[0] = flag;//設置壓縮的比例1-100     EncoderParameter eParam = new EncoderParameter(System.Drawing.Imaging.Encoder.Quality, qy);    ep.Param[0] = eParam;    try    {ImageCodecInfo[] arrayICI = ImageCodecInfo.GetImageEncoders();ImageCodecInfo jpegICIinfo = null;for (int x = 0; x < arrayICI.Length; x++){    if (arrayICI[x].FormatDescription.Equals("JPEG"))    {jpegICIinfo = arrayICI[x];break;    }}if (jpegICIinfo != null){    ob.Save(dFile, jpegICIinfo, ep);//dFile是壓縮后的新路徑 }else{    ob.Save(dFile, tFormat);}return true;    }    catch    {return false;    }    finally    {iSource.Dispose();ob.Dispose();    }}

      通過文件流壓縮圖片

      有些時候我們不想先把圖片保存后,然后在去讀取壓縮,我們想通過文件流就直接對圖片進行壓縮了,比如我們要把圖片上傳到七牛云

      先把流進行壓縮在上傳到七牛云就比較科學了

      1:首先我們需要通過圖片上傳的流來獲取圖片

      foreach (IFormFile file in files)//獲取多個文件列表集合   {       if (file.Length > 0)       {   //獲取圖片上傳的流   Stream stream = file.OpenReadStream();   //直接從流里邊變成圖片   System.Drawing.Image iSource = System.Drawing.Image.FromStream(stream);       }   }

      2:通過圖片壓縮算法把圖片進行壓縮

      這里有一個參數是輸入流,后面還有一個是壓縮后的輸出流

      /// <summary>/// 上傳圖片文件/// </summary>/// <returns></returns>[HttpPost]public async Task<IActionResult> UploadImageFile_WeChat(){    var file = IHttpContextAccessor.HttpContext.Request.Form.Files;    if (file == null || file.Count == 0)    {return Fail("未上傳有效文件");    }    var result = new List<dynamic>();    foreach (var item in file)    {var ExtensionName = Path.GetExtension(item.FileName).ToLower();var RemotePath = getRemotePath(ExtensionName);if (string.IsNullOrEmpty(RemotePath) || !"image".Equals(RemotePath)){    return Fail("不支持此類型文件的上傳");}string remotePath = PathFormatter.Format(item.FileName + "." + ExtensionName, "/upload/" + RemotePath + "/image" + "/{yyyy}{mm}/{dd}{time}{rand:6}");string savePath = AppDomain.CurrentDomain.BaseDirectory + "/wwwroot/" + remotePath;MemoryStream memoryStream = new MemoryStream();//ob.Save(memoryStream, jpegICIinfo, ep);//這里的ob就是壓縮后的Bitmap對象var k = GetPicThumbnail(item.OpenReadStream(), 0, 0, 70, memoryStream);System.Drawing.Image imgSource = System.Drawing.Image.FromStream(memoryStream);imgSource.Save(savePath);if (k){    result.Add(new { url = Config.FileConfig.fileUrl + remotePath, remoteUrl = remotePath, name = item.FileName });}    }    return Success("上傳成功", result);}private bool GetPicThumbnail(Stream stream, int dHeight, int dWidth, int flag, Stream outstream){    //可以直接從流里邊得到圖片,這樣就可以不先存儲一份了    System.Drawing.Image iSource = System.Drawing.Image.FromStream(stream);    //如果為參數為0就保持原圖片    if (dHeight == 0)    {dHeight = iSource.Height;    }    if (dWidth == 0)    {dWidth = iSource.Width;    }    ImageFormat tFormat = iSource.RawFormat;    int sW = 0, sH = 0;    //按比例縮放    Size tem_size = new Size(iSource.Width, iSource.Height);    if (tem_size.Width > dHeight || tem_size.Width > dWidth)    {if ((tem_size.Width * dHeight) > (tem_size.Width * dWidth)){    sW = dWidth;    sH = (dWidth * tem_size.Height) / tem_size.Width;}else{    sH = dHeight;    sW = (tem_size.Width * dHeight) / tem_size.Height;}    }    else    {sW = tem_size.Width;sH = tem_size.Height;    }    Bitmap ob = new Bitmap(dWidth, dHeight);    Graphics g = Graphics.FromImage(ob);    g.Clear(Color.WhiteSmoke);    g.CompositingQuality = CompositingQuality.HighQuality;    g.SmoothingMode = SmoothingMode.HighQuality;    g.InterpolationMode = InterpolationMode.HighQualityBicubic;    g.DrawImage(iSource, new Rectangle((dWidth - sW) / 2, (dHeight - sH) / 2, sW, sH), 0, 0, iSource.Width, iSource.Height, GraphicsUnit.Pixel);    g.Dispose();    //以下代碼為保存圖片時,設置壓縮質量     EncoderParameters ep = new EncoderParameters();    long[] qy = new long[1];    qy[0] = flag;//設置壓縮的比例1-100     EncoderParameter eParam = new EncoderParameter(System.Drawing.Imaging.Encoder.Quality, qy);    ep.Param[0] = eParam;    try    {ImageCodecInfo[] arrayICI = ImageCodecInfo.GetImageEncoders();ImageCodecInfo jpegICIinfo = null;for (int x = 0; x < arrayICI.Length; x++){    if (arrayICI[x].FormatDescription.Equals("JPEG"))    {jpegICIinfo = arrayICI[x];break;    }}if (jpegICIinfo != null){    //可以存儲在流里邊;    ob.Save(outstream, jpegICIinfo, ep);}else{    ob.Save(outstream, tFormat);}return true;    }    catch    {return false;    }    finally    {iSource.Dispose();ob.Dispose();    }}

        3:把壓縮后的圖片轉化成流,很簡單用一個內存流來中轉一下就好了

      MemoryStream memoryStream = new MemoryStream(); ob.Save(memoryStream, jpegICIinfo, ep);//這里的ob就是壓縮后的Bitmap對象

         為了驗證一下轉化是否正確,我們可以把流在轉化成圖片然后在圖片進行存儲

      System.Drawing.Image imgSource = System.Drawing.Image.FromStream(memoryStream);imgSource.Save("url");

        如果能夠成功壓縮并成功保存就說明這些步驟都成功了!

      這里說一下圖片傳輸的思路:

      圖片文件這種本身是無法進行傳輸的,就像跨語言的對象也是無法進行傳輸。但是我們可以事先約定一種標準,

      讓雙方都可以認識都可以解析的一種標準,比如base64,比如對象的json序列化,比如光纖信號的光波表示,其實原理都是一樣。

      上傳到七牛云前壓縮圖片

      通過上面的方法可以得到一個輸出流,我們可以通過它進行圖片的保存,但是如果直接把這個輸出流傳遞到七牛云的方法中去,圖片是不能被上傳成功的,存儲大小會是0kb,說明我們這個流七牛云的接口識別不到,也就是約定的內容不一樣,我們要改造成七牛云能夠被識別的狀態

      換一個方法嘗試,直接用流不行,就從流里邊讀出來字節數組試試

      //實例化一個內存流,存放壓縮后的圖片   MemoryStream ysstream = new MemoryStream();   bool issuc = ImageTool.GetPicThumbnail(stream, 300, 300, 80, ysstream);    if (issuc)   {       //通過流上傳圖片到七牛云       //HttpResult result = um.UploadStream(stream, saveKey, uploadToken);       //從內存流里邊讀出來字節數組上傳到七牛云       HttpResult result = um.UploadData(ysstream.ToArray(), saveKey, uploadToken);       if (result.Code == 200)       {   return Json(result.Text);       }       else       {   throw new Exception(result.RefText);//上傳失敗錯誤信息       }   }   else   {       throw new Exception("圖片壓縮失敗");//上傳失敗錯誤信息   }

        成功了

      換回流試試呢,不應該啊。傳遞流進去他里邊也應該是讀取的直接哇,本質上都一樣哇

      還是不行,看來得看一下他這個源碼了,看一下他拿到這個流過后是怎么去用的,就能針對性解決問題了

      部署問題

      在Windows環境下直接運行是沒問題的,但是發布到Linux上就會報錯

      在Linux中安裝

      開始安裝libgdiplus,執行【docker ps -a 】查看所有容器

      【docker start 容器ID】 將容器運行起來

      【docker exec -it e90f2b9d448d /bin/bash】進入該容器bash界面

      執行【apt-get update】

      【apt-get install -y libgdiplus】安裝libgdiplus類庫

      【ln -s /usr/lib/libgdiplus.so /usr/lib/gdiplus.dll】創建鏈接文件

      【eixt】退出docker bash到宿主機的bash,執行 【docker restart 容器ID】,此時接口已經能正確訪問了

      上面的方法有個弊端,假如容器被誤刪,又要重新給容器安裝libgdiplus庫。

      我們可以把修改好的容器制作成鏡像,執行【docker commit e90f2b9d448d skyapi_libgdiplus】,然后執行【docker images】,

      可以看到名字叫skyapi_libgdiplus的Docker鏡像已經制作好了。今后只需要在 docker run -t 參數后面指定skyapi_libgdiplus鏡像即可。

      當前還可以將鏡像保存到docker hub,本地硬盤都可以。

      喜聞樂見的是,.NET 6發布了,但是避免不了新框架帶來各種問題。在以往的跨平臺應用中,往往采用System.Drawing.Common這個庫作為圖形編輯組件。

      在.NET 6之前,在Linux操作系統中需要用到這個庫時,只需要安裝libgdiplus和libc6-dev這兩個依賴即可。但是在.NET 6中,System.Drawing.Common被歸為Windows特定的庫,編譯時產生“'Image.xxx()' is only supported on: 'windows'.”這樣的警告。這不是最重要的,嚴重的是,在Linux中調用時,會產生“The type initializer for 'Gdip' threw an exception.”這樣的異常。

      產生原因

      在設計上System.Drawing.Common 是 Windows 技術的精簡包裝器,因此其跨平臺實現欠佳。

      具微軟文檔中描述,在舊的行為上,libgdiplus 是本機端 System.Drawing.Common 跨平臺實現的主要提供程序。 libgdiplus 實際上是對 System.Drawing.Common 所依賴的 Windows 部分的重新實現。 該實現使 libgdiplus 成為一個重要的組件。 它大約有 30,000 行 C 代碼,大部分未經測試,而且缺少很多功能。 libgdiplus 還具有許多用于圖像處理和文本呈現的外部依賴項,例如 cairo、pango 和其他本機庫。 這些依賴項使得維護和交付組件更具挑戰性。 自從包含 Mono 跨平臺實現以來,我們已將許多從未得到修復的問題重定向到 libgdiplus。 相比之下,我們采用的其他外部依賴項,例如 icu 或 openssl,都是高質量的庫。 使 libgdiplus 的功能集和質量與 .NET 堆棧的其余部分相媲美是不可行的。

      在這之后,System.Drawing.Common 將僅在 Windows 窗體和 GDI+ 項目中使用。

      解決方案

      1、項目不會在Linux平臺運行,僅在Windows中運行

      可以忽略這個警告。

      2、通過將 runtimeconfig.json 文件中的 System.Drawing.EnableUnixSupport 運行時配置開關設置為 true 來啟用對非 Windows 平臺的支持。

      {   "runtimeOptions": {      "configProperties": { "System.Drawing.EnableUnixSupport": true      }   }}

      3、換用其它支持跨平臺的圖像處理庫

      如:

      • ImageSharp
      • SkiaSharp

      需要注意的是,這些庫并不與System.Drawing.Common的API兼容,所以更換相應的庫之后需要重新編寫相關代碼。

      到此這篇關于Net core中使用System.Drawing對上傳的圖片流進行壓縮的文章就介紹到這了,更多相關Net core圖片壓縮內容請搜索以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持!

      標簽: ASP.NET
      日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
      国产精品黄色| 欧美日韩国产探花| 天堂俺去俺来也www久久婷婷| 婷婷综合六月| 免费国产自久久久久三四区久久| 黄色欧美日韩| 日韩精品91亚洲二区在线观看| 国产亚洲久久| 天堂√中文最新版在线| 国产精品日本| 国产精品流白浆在线观看| 精品国产精品国产偷麻豆| 欧美亚洲日本精品| 亚洲丝袜美腿一区| 日韩成人精品一区| 少妇高潮一区二区三区99| 麻豆高清免费国产一区| 欧美不卡高清一区二区三区| 蜜臀av性久久久久蜜臀aⅴ流畅| 日韩高清在线一区| 另类中文字幕国产精品| 日韩av一级片| 99成人超碰| 国产日韩欧美在线播放不卡| 99精品视频在线| 久久字幕精品一区| 亚洲人亚洲人色久| 日本蜜桃在线观看视频| 在线看片一区| 国产综合欧美| 国产a亚洲精品| 日韩国产欧美在线播放| 国产一区二区三区自拍| 麻豆精品久久| 久久亚洲精品伦理| 国产综合视频| 欧美日一区二区| 福利在线一区| 国产成人精选| 国产精品久久久网站| 午夜久久av| 免费人成网站在线观看欧美高清| 蜜桃视频在线网站| 久久三级毛片| 国产精品成人3p一区二区三区| 综合色一区二区| 蜜臀久久久久久久| 天堂av在线一区| 欧美日韩国产探花| 99久精品视频在线观看视频| 伊人久久高清| 国产传媒在线| 成人一区不卡| 久久精品亚洲人成影院 | 中文久久精品| 免费久久精品| 日韩精品久久理论片| 久久a爱视频| 中文日韩在线| 国产精品日本一区二区不卡视频| 国产精品久久久一区二区| 三上悠亚国产精品一区二区三区 | 精品日韩视频| 三级在线观看一区二区 | 日韩中文在线电影| 免费的成人av| 欧美日韩中文字幕一区二区三区| 日本成人中文字幕在线视频| 国产中文字幕一区二区三区| 91久久视频| 精品美女久久| 天堂va蜜桃一区二区三区| 国产欧美日韩综合一区在线播放| 日韩精品看片| 日本午夜精品| 国产亚洲精品自拍| 国产一区2区| 欧美午夜网站| 亚洲在线免费| 韩国久久久久久| 久久国际精品| 亚洲一二三区视频| 国产一区日韩欧美| 色欧美自拍视频| 国产精品一区二区美女视频免费看 | 在线日韩电影| 日韩精品成人| 男女激情视频一区| 欧美1区2区3区| 日韩欧美中文| 亚洲欧洲美洲av| 麻豆国产精品一区二区三区| 日韩一区免费| 天堂va蜜桃一区二区三区| 婷婷亚洲综合| 欧美亚洲激情| 亚洲福利久久| 日韩伦理福利| av资源新版天堂在线| 精品一区二区三区中文字幕在线| 日本aⅴ免费视频一区二区三区| 亚洲专区欧美专区| 国产二区精品| 狠狠色综合网| 首页国产欧美日韩丝袜| 丝袜诱惑制服诱惑色一区在线观看 | 精品视频高潮| 牛牛精品成人免费视频| 久久国内精品| 国产精品777777在线播放| 麻豆国产精品| 丝袜美腿一区| 九九久久电影| 久久国产精品久久久久久电车| 六月丁香综合| 亚洲精选久久| 国产欧美成人| 日韩成人精品一区二区| 激情欧美日韩一区| 亚洲欧美不卡| 欧美日韩一区自拍| 国产一区二区三区久久 | 精品国产亚洲日本| 亚洲a在线视频| 亚洲一区欧美| 国产成人久久精品麻豆二区| 不卡在线一区二区| 国产精品主播在线观看| 久久久久久久欧美精品| 久久精品播放| 久久久噜噜噜| 六月婷婷一区| 麻豆91精品视频| 久久精品av| 国产精品亚洲一区二区在线观看| 都市激情国产精品| 在线免费观看亚洲| 久久电影tv| 日韩av网站在线免费观看| а√天堂8资源在线| 久久不见久久见中文字幕免费| 神马日本精品| 日韩激情网站| 久久人人88| 国产乱人伦丫前精品视频| 久久人人88| 精品一区二区三区免费看| 亚洲资源av| 欧美成人基地 | 成人日韩在线观看| 香蕉久久国产| 免费日韩一区二区三区| 亚洲精品2区| 欧美黄页在线免费观看| 一区二区视频欧美| 国产一区二区三区天码| 日韩国产在线观看一区| 亚洲特色特黄| 四虎成人av| 国产精品亚洲欧美日韩一区在线| 亚洲午夜在线| 麻豆久久久久久久| 日韩高清电影免费| 男女激情视频一区| 欧美福利专区| 国产精品亚洲一区二区三区在线观看| 7m精品国产导航在线| 日韩精品一区第一页| 九色精品91| 免费观看久久av| 亚洲天堂黄色| 亚洲一级网站| 99国产精品| 成人av动漫在线观看| 成人免费电影网址| 日韩毛片视频| 麻豆高清免费国产一区| 日韩激情中文字幕| 国产精品一国产精品k频道56| 日本视频一区二区| 国产精品一区二区精品视频观看| 免费成人性网站| 一二三区精品| 欧美日韩午夜| 欧美黑人巨大videos精品| 欧美激情久久久久久久久久久| 精品九九久久| 在线成人动漫av| 红桃视频国产一区| 亚洲一区av| 国产欧美日韩亚洲一区二区三区| 午夜性色一区二区三区免费视频| 亚洲精品伊人| 欧美1区2区3| 日韩免费av| 国产精品美女久久久浪潮软件| 视频一区在线视频| 欧美激情网址| 欧美精品一卡| 国产精品久久久久av蜜臀 |