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

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

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

瀏覽:359日期: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久久,三上悠亚国产精品一区二区三区
      日韩精选在线| 国产精品v日韩精品v欧美精品网站| 999国产精品永久免费视频app| 91精品推荐| 精品少妇av| 国产亚洲网站| 国产精品4hu.www| 久久uomeier| 国产日本精品| 婷婷成人在线| 国产欧美三级| 日韩在线卡一卡二| 国产精品日本一区二区不卡视频 | 99精品网站| 欧美视频二区| 国产精品腿扒开做爽爽爽挤奶网站| 蜜桃av一区二区三区电影| 好吊日精品视频 | 国产成人精品一区二区三区免费 | 一区二区小说| 国产一区二区三区亚洲| 欧美影院精品| 亚洲高清成人| 老司机久久99久久精品播放免费| 成年男女免费视频网站不卡| 精品理论电影在线| 久久国产麻豆精品| 国产精品jk白丝蜜臀av小说| 国产精品白浆| 美女视频黄 久久| 日本一区二区免费高清| 韩国久久久久久| 91久久国产| 亚洲专区视频| 另类综合日韩欧美亚洲| 国产精品久久久久蜜臀 | 综合激情一区| 国产自产自拍视频在线观看| 日韩精品欧美| 国产美女久久| 亚洲欧洲美洲国产香蕉| 日本一区二区三区中文字幕| 久久字幕精品一区| 日韩精品成人在线观看| 成人精品亚洲| 久久午夜影院| 国产资源在线观看入口av| 国产精品a级| 欧美日一区二区| 日韩精选在线| 日韩va亚洲va欧美va久久| 夜夜嗨网站十八久久| 国产成人精品一区二区三区在线| 日韩中文欧美在线| 美女尤物国产一区| 久久精品国内一区二区三区水蜜桃| 日韩一区二区三区四区五区| 国产66精品| 福利精品在线| 日韩亚洲在线| 国产图片一区| 日韩午夜电影| 不卡在线一区| 国产高清不卡| 日韩在线综合| 亚洲午夜精品久久久久久app| 日韩欧美久久| 久久久五月天| 影音先锋久久| 国产亚洲久久| 亚洲免费激情| 国产美女高潮在线观看| 麻豆高清免费国产一区| 蜜臀av性久久久久蜜臀aⅴ四虎| 另类av一区二区| 日本欧美韩国一区三区| 麻豆成人在线| 日韩精品第二页| 日韩在线中文| 高清在线一区| 欧美成人aaa| 国产精品视频一区二区三区综合 | 欧美va亚洲va日韩∨a综合色| 欧美a一区二区| 日本亚洲欧美天堂免费| 久久精品午夜| 国产欧美综合一区二区三区| 久久久久久久久久久9不雅视频| 日韩一区精品| 日韩网站在线| www成人在线视频| 国产精品毛片久久久| 日本欧洲一区二区| 模特精品在线| 日本成人在线一区| 国内一区二区三区| 亚洲午夜国产成人| 日韩欧美国产精品综合嫩v| 国产日产精品_国产精品毛片| 伊人久久国产| 日韩精品一区二区三区中文字幕| 国产精品成人a在线观看| 丝袜美腿诱惑一区二区三区 | 国模 一区 二区 三区| 另类av一区二区| 亚洲精品少妇| 欧美在线综合| 成人自拍av| 蜜桃视频在线网站| 国产成人免费av一区二区午夜| 免费在线欧美黄色| 欧美不卡高清一区二区三区| 精品一区二区三区中文字幕视频| 国产欧美高清| 国产精品一区二区精品视频观看| 日韩精彩视频在线观看| 日韩av一区二| 国产探花一区在线观看| 国产乱码精品一区二区亚洲| 国产精品尤物| 91免费精品| 亚欧洲精品视频在线观看| 亚洲婷婷在线| 综合激情婷婷| 一区免费视频| 69堂精品视频在线播放| 四季av一区二区凹凸精品| 视频在线在亚洲| 麻豆网站免费在线观看| 日本在线视频一区二区| 日韩欧美少妇| 国产精品99久久久久久董美香| 国产精品av久久久久久麻豆网| 精品久久久网| 久久精品伊人| 一区二区三区国产在线| 国产伦久视频在线观看| 久久精品播放| 日韩av一二三| 视频一区二区欧美| 久草免费在线视频| 日韩综合一区二区| 色黄视频在线观看| 香蕉久久久久久| www.51av欧美视频| 亚洲欧美一级| 久久爱www.| 中文字幕av亚洲精品一部二部 | 国产精品激情电影| 日韩高清二区| 免费看黄色91| 久久久久国产| 国产h片在线观看| 国产视频一区二区在线播放| 日韩欧美一区二区三区在线观看 | 日韩专区在线视频| 国产精品观看| 欧美一级网站| 国产乱子精品一区二区在线观看 | 日本免费一区二区视频| 国产精品久久久久久模特| 韩国三级一区| 日韩一区电影| 日本aⅴ亚洲精品中文乱码 | 欧美日韩国产探花| 国产精品一区2区3区| 欧美日韩第一| 在线亚洲免费| 国产欧美日韩视频在线| 久久久久久黄| www.51av欧美视频| 国产精品密蕾丝视频下载| 麻豆国产一区| 日韩在线麻豆| 一区在线免费观看| 亚洲一区av| 好看的av在线不卡观看| 午夜av不卡| 国产精品原创| 国产一区二区高清| 久久激五月天综合精品| 欧美va天堂在线| 视频一区视频二区在线观看| 亚洲精品日本| 日韩福利视频网| 美女久久久久久| 国产精品一国产精品k频道56| 日本成人在线网站| 中文字幕一区日韩精品| 久久精品超碰| 水蜜桃精品av一区二区| 99国产精品99久久久久久粉嫩| 欧美黑人巨大videos精品| 岛国av在线播放| 久久精品免费一区二区三区| 免费看欧美美女黄的网站| 视频二区不卡| 不卡在线一区| 久久精品凹凸全集| 久久精品国产www456c0m|