C#實現圖片暗通道去霧演算法-Demo-提供代碼實例下載地址

来源:https://www.cnblogs.com/liyu3519/archive/2020/06/08/13064511.html
-Advertisement-
Play Games

C#實現圖片暗通道去霧演算法 代碼實例下載地址:https://www.90pan.com/b1915123 在圖像去霧這個領域,幾乎沒有人不知道《Single Image Haze Removal Using Dark Channel Prior》這篇文章,該文是2009年CVPR最佳論文。作者何凱 ...


 C#實現圖片暗通道去霧演算法

代碼實例下載地址:https://www.90pan.com/b1915123

在圖像去霧這個領域,幾乎沒有人不知道《Single Image Haze Removal Using Dark Channel Prior》這篇文章,該文是2009年CVPR最佳論文。作者何凱明博士,2007年清華大學畢業,2011年香港中文大學博士畢業,可謂是功力深厚,感嘆於國內一些所謂博士的水平,何這樣的博士才可以真正叫做Doctor。

關於何博士的一些資料和論文,大家可以訪問這裡:http://research.microsoft.com/en-us/um/people/kahe/

代碼(提供項目下載):https://www.90pan.com/b1915123

public class DefogHelper
    {
        public DefogHelper() { }

        /// <summary>
        /// 實現功能:實現基於暗通道的去霧演算法。(如果要用32位的將ImageMaster_64.dll改成ImageMaster_32.dll即可)
        /// </summary>
        /// <param name="Src">圖像數據在記憶體的起始地址</param>
        /// <param name="Dest">目標數據在記憶體的起始地址</param>
        /// <param name="Width">圖像的寬度</param>
        /// <param name="Height">圖像的高度</param>
        /// <param name="Stride">圖像的掃描行大小</param>
        /// <param name="BlockSize">用於計算暗通道圖像時的矩形半徑</param>
        /// <param name="GuideRadius">導向濾波的半徑</param>
        /// <param name="MaxAtom">為防止圖像天空部分出現holes,設置的最大大氣光值,預設240</param>
        /// <param name="Omega">控制去霧程度的一個參數,建議取值範圍[0.75,1],值越大,去霧越明顯,但可能出現局部過增強。</param>
        /// <param name="T0">用於控制最小透射率的一個參數,建議取值範圍[0.01,0.2]。</param>
        /// <param name="Gamma">調整亮度的參數,建議範圍[0.7,1]。</param>
        [DllImport("ImageMaster_64.dll", CallingConvention = CallingConvention.StdCall, CharSet = CharSet.Unicode, ExactSpelling = true)]
        private static extern int IM_HazeRemovalBasedOnDarkChannelPrior(IntPtr Src, IntPtr Dest, int Width, int Height, int Stride, int BlockSize = 5, int GuideRadius = 20, int MaxAtom = 220, float Omega = 0.9f, float T0 = 0.1f, float Gamma = 0.9f);

        /// <summary>
        /// 圖片緩存區
        /// </summary>
        private readonly byte[] bmpBuffer = new byte[1024 * 1024 * 64];
        private readonly IntPtr srcPtr = Marshal.AllocHGlobal(1024 * 1024 * 64);// 申請記憶體
        private readonly IntPtr destPtr = Marshal.AllocHGlobal(1024 * 1024 * 64);// 申請記憶體

        /// <summary>
        /// 圖片去霧
        /// </summary>
        /// <param name="scrBmp"></param>
        /// <param name="info"></param>
        /// <param name="result"></param>
        /// <param name="ms"></param>
        /// <param name="isFreed"></param>
        /// <returns></returns>
        public Bitmap ImageDefog(Bitmap scrBmp, DefogInfo info, out int result, out double ms, bool isFreed = false)
        {
            result = -1;
            ms = -1;
            if (scrBmp == null || info == null) return null;
            int w = scrBmp.Width, h = scrBmp.Height;
            System.Drawing.Rectangle bitmapRec = new System.Drawing.Rectangle(0, 0, w, h);
            BitmapData bmpData = scrBmp.LockBits(bitmapRec, ImageLockMode.ReadWrite, scrBmp.PixelFormat);
            int img_size = bmpData.Stride * h;
            if (img_size > bmpBuffer.Length) { result = 10; return null; }
            int stride = bmpData.Stride;
            try
            {
                Marshal.Copy(bmpData.Scan0, bmpBuffer, 0, img_size);
                Marshal.Copy(bmpBuffer, 0, srcPtr, img_size);
                DateTime dateTime = DateTime.Now;
                result = IM_HazeRemovalBasedOnDarkChannelPrior(srcPtr, destPtr, w, h, stride, info.BlockSize, info.GuideRadius, info.MaxAtom, info.Omega, info.T0, info.Gamma);
                ms = DateTime.Now.Subtract(dateTime).TotalMilliseconds;
                Marshal.Copy(destPtr, bmpBuffer, 0, img_size);
                Bitmap outBmp = BytesToBitmap(bmpBuffer, img_size, w, h);
                return outBmp;
            }
            catch(Exception ex)
            {
                return null;
            }
            finally
            {
                scrBmp.UnlockBits(bmpData);
                //Marshal.FreeHGlobal(srcPtr);
                //Marshal.FreeHGlobal(destPtr);
                if (isFreed) scrBmp.Dispose();
            }
        }
        /// <summary>
        /// 數組轉為Bitmap
        /// </summary>
        /// <param name="pixelData">數組</param>
        /// <returns>Bitmap圖像</returns>
        private Bitmap BytesToBitmap(byte[] pixelData, int length, int width, int height)
        {
            Bitmap img = new Bitmap(width, height, System.Drawing.Imaging.PixelFormat.Format24bppRgb);
            try
            {
                BitmapData data = img.LockBits(new Rectangle(0, 0, img.Width, img.Height), ImageLockMode.WriteOnly, System.Drawing.Imaging.PixelFormat.Format24bppRgb);
                Marshal.Copy(pixelData, 0, data.Scan0, length);//輸入顏色數據
                img.UnlockBits(data);//解鎖

                return img;
            }
            catch { img.Dispose(); return null; }
        }
        /// <summary>
        /// 從bitmap轉換成ImageSource
        /// </summary>
        /// <param name="icon"></param>
        /// <returns></returns>
        public ImageSource BitmapToImageSource(Bitmap bitmap)
        {
            return BitmapToBitmapImage(bitmap);
        }
        /// <summary>
        /// 從bitmap轉換成BitmapImage
        /// </summary>
        /// <param name="bitmap"></param>
        /// <returns></returns>
        public BitmapImage BitmapToBitmapImage(Bitmap bitmap)
        {
            BitmapImage bitmapImage = new BitmapImage();
            using (MemoryStream ms = new MemoryStream())
            {
                bitmap.Save(ms, System.Drawing.Imaging.ImageFormat.Bmp);
                bitmapImage.BeginInit();
                bitmapImage.StreamSource = new MemoryStream(ms.GetBuffer());
                bitmapImage.EndInit();
                ms.Close();
            }
            return bitmapImage;
        }
        /// <summary>
        /// 將數組轉化為bitmap,前54個數據是格式
        /// </summary>
        /// <param name="bitmap"></param>
        /// <returns></returns>
        public byte[] BitmapToBytes(Bitmap bitmap)
        {
            byte[] bytes;
            using (MemoryStream ms = new MemoryStream())
            {
                bitmap.Save(ms, System.Drawing.Imaging.ImageFormat.Bmp);
                bytes = ms.GetBuffer();
                ms.Close();
            }
            return bytes;
        }

        /// <summary>
        /// 去霧信息
        /// </summary>
        public class DefogInfo
        {
            /// <summary>
            /// 用於計算暗通道圖像時的矩形半徑,2-50
            /// </summary>
            public int BlockSize = 5;
            /// <summary>
            /// 導向濾波的半徑,2-200
            /// </summary>
            public int GuideRadius = 20;
            /// <summary>
            /// 為防止圖像天空部分出現holes,設置的最大大氣光值,預設202,190-255
            /// </summary>
            public int MaxAtom = 198;
            /// <summary>
            /// 控制去霧程度的一個參數,建議取值範圍[0.6,1],值越大,去霧越明顯,但可能出現局部過增強。
            /// </summary>
            public float Omega = 0.7f;
            /// <summary>
            /// 用於控制最小透射率的一個參數,建議取值範圍[0.01,0.2]。
            /// </summary>
            public float T0 = 0.01f;
            /// <summary>
            /// 調整亮度的參數,建議範圍[0.5,1]。
            /// </summary>
            public float Gamma = 0.5f;
        }
    }

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 作者:有數可據 見了鬼!工資竟然又跌了 #平均工資 2020年6月全國招收程式員313739人。2020年6月全國程式員平均工資14404元,工資中位數12500元,其中95%的人的工資介於5250元到35000元。 雖然收入又下降了,但是崗位比上個月多了起來,隨著經濟好轉,收入還是會漲回去的。大家 ...
  • 為什麼要使用依賴註入? 藉助依賴註入,可以管理類之間的依賴,幫助我們在構建應用時遵循設計原則,確保代碼可維護性和可擴展性 ASP.NET Core的整個架構中,依賴註入框架提供了對象創建和生命周期管理的核心能力,各個組件互相協作,也是依賴註入框架能力來實現的 兩個核心包: Microsft.Exte ...
  • 今天使用了npoi的HSSFWorkbook導出excel,卻出現文件損壞或擴展名不對錯誤,後來發現尾碼只要是“xls”就可以了。 npoi只是java poi的實現版,因此HSSFworkbook,XSSFworkbook,SXSSFworkbook這三種的區別在npoi中同理。 下麵轉自:htt ...
  • Java和C#都是編程的語言,它們是兩個不同方向的兩種語言 相同點: 他們都是面向對象的語言,也就是說,它們都能實現面向對象的思想(封裝,繼承,多態) 區別: 1.c#中的命名空間是namespace類似於Java中的package(包),在Java中導入包用import而c#中用using。2.c ...
  • 前言 什麼是async/await? await和async是.NET Framework4.5框架、C#5.0語法裡面出現的技術,目的是用於簡化非同步編程模型。 async和await的關係? async和await是成對出現的。 async出現在方法的聲明裡,用於批註一個非同步方法。光有async是 ...
  • 之前寫過一篇博客 關於 ScrollViewer 和滾動輪劫持(scroll-wheel-hijack),裡面介紹了 ScrollViewer 的滾動輪劫持問題,以及如果解決。當時的做法是繼承 ScrollViewer 並重寫 OnMouseWheel,全部代碼如下: public class Ex ...
  • 前言 為了節約大家的時間,長話短說,作為愛國青年,也作為技術前輩,我願意幫助大家,花最短時間學習c#入門,希望大家努力學習,多賺錢,多為社會做共用,以我做起。我開放了技術QQ群,為了讓大家可以交流,獲取源代碼跟視頻教程。只要你每天有固定的時間學習時間,你可以變成很牛的,我就是你的技術資源,一起努力吧 ...
  • 轉載地址:https://www.cnblogs.com/wuhuacong/archive/2013/02/05/2893191.html C#導出Word,Excel的方法有很多,這次因為公司的業務需求,需要導出內容豐富(文字,表格,圖片)的報告,以前的方法不好使,所以尋找新的導出方法,在網上找 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...