主頁 > .NET開發 > 如何縮短我的(非常基本的)蠻力演算法的處理時間?

如何縮短我的(非常基本的)蠻力演算法的處理時間?

2022-03-23 08:11:52 .NET開發

我正在學習 java atm,我有點卡在當前的任務上,我的教練昨天給了我...

練習是:

  1. 創建一個 4 位密碼(就像字串變數一樣)。(允許使用 0-9 之間的數字和一個或多個這些特殊字符:'!'、'#'、'%')
  2. 通過嘗試所有可能性,找到一種暴力破解密碼的方法。
  3. 測量并輸出處理所花費的時間。
  4. 此外,輸出查找密碼所需的“嘗試”次數。

(我的教練說我應該盡可能少用方法,因為我現在還不知道如何撰寫方法或如何使用類等)

我設法讓它作業,但現在它需要大約 60 毫秒才能完成回圈。我的教練現在告訴我,我應該試著讓它處理得更快,這樣至少需要大約 20 毫秒。

我現在已經是什么讓我的代碼變慢了。這是因為我總是經歷所有可能性,將所有這些都添加到 ArreyList 中,然后我檢查 pw 是否與 ArreyList 中的這些可能性之一匹配。浪費時間。現在我的目標是像以前一樣經歷各種可能性,但只在找到密碼之前。之后,它應該停止回圈,以便它也停止添加其余不必要的組合。我試了又試,但現在我決定尋求幫助:)

那是我的代碼:

        import java.util.ArrayList;
        import java.util.Scanner;

        public class trynew {
        public static void main(String[] args) {
        Scanner scn = new Scanner(System.in);

        char[] digits = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '#', '!', '%'};

        System.out.println("\nPlease create 4 digit password. Allowed are numbers between 0-9 and following characters: #,!,%");

        String passw = scn.nextLine();

        ArrayList<String> check_it = new ArrayList<String>();

        long start1 = System.currentTimeMillis();

        for (int i = 0; i < digits.length; i  ) {
            for (int j = 0; j < digits.length; j  ) {
                for (int k = 0; k < digits.length; k  ) {
                    for (int l = 0; l < digits.length; l  ) {
                        check_it.add(digits[i]   ""   digits[j]   ""   digits[k]   ""   digits[l]);
                    }
                }
            }
        }

        long end1 = System.currentTimeMillis();

        for (String sv : check_it) {
            if (sv.equals(passw)) {
                System.out.println("\nThe Password is: "   sv);
                System.out.println("It took "   check_it.indexOf(sv)   " tries, to find the password.");
            }
        }
        System.out.println("Process time was "   (end1 - start1)   " milliseconds.");


        }
    }




My approach  was to make the loop like that:

    for (int i = 0; i < digits.length; i  ) { 
        for (int j = 0; j < digits.length; j  ) {
            for (int k = 0; k < digits.length; k  ) {
                for (int l = 0; l < digits.length; l  ) {

                   if (!(digits[i]   ""   digits[j]   ""   digits[k]   ""   digits[l]).equals(passw)){
                        check_it.add(digits[i]   ""   digits[j]   ""   digits[k]   ""   digits[l]);
                    }

                }
            }
        }
    }

然后我嘗試使用while回圈而不是if,我嘗試設定布林值等。我還設法讓它只添加組合,直到找到pw,但不知何故處理時間不會下降:/

The reason why I want to add the combos into check_it ArreyList is because otherwise, I wouldn't now how I get the numbers of tries it took, until pw is found...

Can someone help me please, or poke me at the right direction?! Thanks & Greets!

uj5u.com熱心網友回復:

這就是我想出的。

private static final char[] digits = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '#', '!', '%'};

public static int guessPw(String pw) {
    int attempts = 0;
    char [] word = new char[pw.length()];
    for (char a : digits) {
        word[0] = a;
        for (char b : digits) {
            word[1] = b;
            for (char c : digits) {
                word[2] = c;
                for (char d : digits) {
                    word[3] = d;
                    attempts  ;
                    if (pw.equals(String.valueOf(word))) {
                        return attempts;
                    }
                }
            }
        }
    }
    return -1;
}

public static void timeAttempt(String pw) {     
    long start = System.nanoTime();
    int attempts = guessPw(pw);

    
    System.out.println(String.format("It took %dms and %d attempts to guess '%s'", TimeUnit.MILLISECONDS.toSeconds(System.nanoTime() - start), attempts, pw));
}
public static void main(String[] args) {
    timeAttempt("0000");
    timeAttempt("%%%%");
}

uj5u.com熱心網友回復:

這段代碼應該可以作業。如果找到密碼,它只會打破每個回圈。

   int tries = 0;
   boolean found = false;
   for (int i = 0; i < digits.length; i  ) {
            for (int j = 0; j < digits.length; j  ) {
                for (int k = 0; k < digits.length; k  ) {
                    for (int l = 0; l < digits.length; l  ) {
                        tries  ;
                        if ((digits[i]   ""   digits[j]   ""   digits[k]   ""   digits[l]).equals(passw)) {
                           found = true;
                           break;
                        }
                    }
                    if (found)
                      break;
                }
               if (found)
                  break;
            }
            if (found)
                  break;
        }

我不建議使用此代碼!這是相當粗略的。你為什么不把 for 回圈變成一個函式呢?找到密碼后,您可以從那里回傳嘗試。但我認為你可以自己解決這個問題:)

編輯:我會給你一個沒有功能的解決方案的提示。你覺得這個while回圈怎么樣?

while(!(digits[i]   ""   digits[j]   ""   digits[k]   ""   digits[l]).equals(passw))

在那里你只需要以正確的順序'增長' l、k、j 和 i :)

uj5u.com熱心網友回復:

與 Ryan 發布的內容非常相似。基本上你想避免創建一個字串。

在這里,我將輸入的密碼轉換為 char 陣列,并使用以下方法將輸入的密碼與生成的密碼進行比較Arrays.equals()

  public static void main(String[] args) {       
    Scanner scn = new Scanner(System.in);

    char[] digits = {'0', '1', '2', '3', '4', '5', '6', 
                     '7', '8', '9', '#', '!', '%'};

    System.out.println("\nPlease create 4 digit password.");
    System.out.println("Allowed are numbers between 0-9 and following characters: #,!,%");
    System.out.print("Password: ");
    
    String response = scn.nextLine();
    char[] password = response.toCharArray();
    
    long start1 = System.currentTimeMillis();

    int counter = 0;
    char[] pw = new char[4];
    boolean foundIt = false;
    for (int i = 0; i < digits.length && !foundIt; i  ) {
        pw[0] = digits[i];
        for (int j = 0; j < digits.length && !foundIt; j  ) {
            pw[1] = digits[j];
            for (int k = 0; k < digits.length && !foundIt; k  ) {
                pw[2] = digits[k];
                for (int l = 0; l < digits.length && !foundIt; l  ) {
                  counter  ;
                  pw[3] = digits[l];
                  foundIt = Arrays.equals(pw, password);
                }
            }
        }
    }
  
    long end1 = System.currentTimeMillis();

    System.out.println("\nThe Password is: "   response);
    System.out.println("It took "   counter   " tries, to find the password.");

    System.out.println("Process time was "   (end1 - start1)   " milliseconds.");
  }

uj5u.com熱心網友回復:

其處理速度如此緩慢的原因是因為您的所有代碼都在單個執行緒中運行。您可以通過使蠻力代碼多執行緒來大大減少整體處理時間。

您的應用程式運行緩慢的另一個原因是它使用許多嵌套回圈來確定digits. 與其使用那些嵌套回圈,不如計算digits.

獲得所有可能排列的串列后,您可以使用 java 8parallelStream()anyMatch()以多執行緒方式檢查哪個排列與密碼匹配。換句話說,計算機將能夠同時匹配 2 個或更多排列,pass 將大大減少處理時間。下面是一個使用parallelStream()/的例子anyMatch()

    AtomicInteger threadSafeCounter = new AtomicInteger(); //Always use a thread safe integer when multiple threads will be manipulating it. This is to avoid race conditions.

    char[] digits = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '#', '!', '%'};

    Scanner scn = new Scanner(System.in);

    System.out.println("\nPlease create 4 digit password. Allowed are numbers between 0-9 and following characters: #,!,%");

    String passw = scn.nextLine();

List<String> permutationList = new ArrayList(); //Calculate the 4-digit permutations from `digits` and add them to this list.

    long start1 = System.currentTimeMillis();
    permutationList.parallelStream().anyMatch(combination -> {
        threadSafeCounter.incrementAndGet();
        if(combination.equals(passw)){
            System.out.println("\nthe password is "   passw);
            System.out.println("It took "   threadSafeCounter.get()   " tries, to find the password.");
            return true;
        }
        else return false;
    });

    long end1 = System.currentTimeMillis();

    System.out.println("Process time was "   (end1 - start1)   " milliseconds.");

digits您可以在此 SO 執行緒中找到有關如何從中生成排列的示例如果您調整 Subash 在上述 SO 執行緒中發布的代碼,則排列生成代碼本身也可以成為多執行緒的。為此,您需要使用Fork/Join framework.

編輯:這是一篇很好的文章permutations,解釋了和之間的區別combinations在我的示例中我計算排列而不是組合的原因是因為在這種情況下順序/排列很重要。

uj5u.com熱心網友回復:

這是相當有效的。

  • 檢查是否存在特殊字符。如果是,則回傳 -1。
  • 它將 4 位數字轉換為 int。
  • 然后只需從 0 迭代到 9999 并與目標進行比較
  • 它回傳嘗試次數,如果無效則回傳 -1
String [] data = { "9999", "9999%" };
for (String pwd : data) {
long start = System.nanoTime();
int tries = check(pwd);
long  time = (long)(((System.nanoTime() - start)/1e9)*1000);
System.out.printf("Took %d tries in %d ms to validate `%s`%n", tries, time, pwd); 

印刷

Took 10000 tries in 3 ms to validate `9999`
Took -1 tries in 0 ms to validate `9999%`

檢查方法

public static int check(String pwd) {
    if (pwd.replaceAll("[!%#]", "").equals(pwd)) {
        int pass = Integer.parseInt(pwd);
        for (int i = 0; i < 10_000; i  ) {
            if (i == pass) {
                return i   1;
            }
        }
    }
    return -1;
}

轉載請註明出處,本文鏈接:https://www.uj5u.com/net/447072.html

標籤:java for-loop arraylist nested-loops brute-force

上一篇:XAML 給資源起個好名字 用 StaticResource 起一個別名

下一篇:如何從列中的某一行中多次減去第N行

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • WebAPI簡介

    Web體系結構: 有三個核心:資源(resource),URL(統一資源識別符號)和表示 他們的關系是這樣的:一個資源由一個URL進行標識,HTTP客戶端使用URL定位資源,表示是從資源回傳資料,媒體型別是資源回傳的資料格式。 接下來我們說下HTTP. HTTP協議的系統是一種無狀態的方式,使用請求/ ......

    uj5u.com 2020-09-09 22:07:47 more
  • asp.net core 3.1 入口:Program.cs中的Main函式

    本文分析Program.cs 中Main()函式中代碼的運行順序分析asp.net core程式的啟動,重點不是剖析原始碼,而是理清程式開始時執行的順序。到呼叫了哪些實體,哪些法方。asp.net core 3.1 的程式入口在專案Program.cs檔案里,如下。ususing System; us ......

    uj5u.com 2020-09-09 22:07:49 more
  • asp.net網站作為websocket服務端的應用該如何寫

    最近被websocket的一個問題困擾了很久,有一個需求是在web網站中搭建websocket服務。客戶端通過網頁與服務器建立連接,然后服務器根據ip給客戶端網頁發送資訊。 其實,這個需求并不難,只是剛開始對websocket的內容不太了解。上網搜索了一下,有通過asp.net core 實作的、有 ......

    uj5u.com 2020-09-09 22:08:02 more
  • ASP.NET 開源匯入匯出庫Magicodes.IE Docker中使用

    Magicodes.IE在Docker中使用 更新歷史 2019.02.13 【Nuget】版本更新到2.0.2 【匯入】修復單列匯入的Bug,單元測驗“OneColumnImporter_Test”。問題見(https://github.com/dotnetcore/Magicodes.IE/is ......

    uj5u.com 2020-09-09 22:08:05 more
  • 在webform中使用ajax

    如果你用過Asp.net webform, 說明你也算是.NET 開發的老兵了。WEBform應該是2011 2013左右,當時還用visual studio 2005、 visual studio 2008。后來基本都用的是MVC。 如果是新開發的專案,估計沒人會用webform技術。但是有些舊版 ......

    uj5u.com 2020-09-09 22:08:50 more
  • iis添加asp.net網站,訪問提示:由于擴展配置問題而無法提供您請求的

    今天在iis服務器配置asp.net網站,遇到一個問題,記錄一下: 問題:由于擴展配置問題而無法提供您請求的頁面。如果該頁面是腳本,請添加處理程式。如果應下載檔案,請添加 MIME 映射。 WindowServer2012服務器,添加角色安裝完.netframework和iis之后,運行aspx頁面 ......

    uj5u.com 2020-09-09 22:10:00 more
  • WebAPI-處理架構

    帶著問題去思考,大家好! 問題1:HTTP請求和回傳相應的HTTP回應資訊之間發生了什么? 1:首先是最底層,托管層,位于WebAPI和底層HTTP堆疊之間 2:其次是 訊息處理程式管道層,這里比如日志和快取。OWIN的參考是將訊息處理程式管道的一些功能下移到堆疊下端的OWIN中間件了。 3:控制器處理 ......

    uj5u.com 2020-09-09 22:11:13 more
  • 微信門戶開發框架-使用指導說明書

    微信門戶應用管理系統,采用基于 MVC + Bootstrap + Ajax + Enterprise Library的技術路線,界面層采用Boostrap + Metronic組合的前端框架,資料訪問層支持Oracle、SQLServer、MySQL、PostgreSQL等資料庫。框架以MVC5,... ......

    uj5u.com 2020-09-09 22:15:18 more
  • WebAPI-HTTP編程模型

    帶著問題去思考,大家好!它是什么?它包含什么?它能干什么? 訊息 HTTP編程模型的核心就是訊息抽象,表示為:HttPRequestMessage,HttpResponseMessage.用于客戶端和服務端之間交換請求和回應訊息。 HttpMethod類包含了一組靜態屬性: private stat ......

    uj5u.com 2020-09-09 22:15:23 more
  • 部署WebApi隨筆

    一、跨域 NuGet參考Microsoft.AspNet.WebApi.Cors WebApiConfig.cs中配置: // Web API 配置和服務 config.EnableCors(new EnableCorsAttribute("*", "*", "*")); 二、清除默認回傳XML格式 ......

    uj5u.com 2020-09-09 22:15:48 more
最新发布
  • C#多執行緒學習(二) 如何操縱一個執行緒

    <a href="https://www.cnblogs.com/x-zhi/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2943582/20220801082530.png" alt="" /></...

    uj5u.com 2023-04-19 09:17:20 more
  • C#多執行緒學習(二) 如何操縱一個執行緒

    C#多執行緒學習(二) 如何操縱一個執行緒 執行緒學習第一篇:C#多執行緒學習(一) 多執行緒的相關概念 下面我們就動手來創建一個執行緒,使用Thread類創建執行緒時,只需提供執行緒入口即可。(執行緒入口使程式知道該讓這個執行緒干什么事) 在C#中,執行緒入口是通過ThreadStart代理(delegate)來提供的 ......

    uj5u.com 2023-04-19 09:16:49 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    <a href="https://www.cnblogs.com/huangxincheng/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/214741/20200614104537.png" alt="" /&g...

    uj5u.com 2023-04-18 08:39:04 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    一:背景 1. 講故事 前段時間協助訓練營里的一位朋友分析了一個程式卡死的問題,回過頭來看這個案例比較經典,這篇稍微整理一下供后來者少踩坑吧。 二:WinDbg 分析 1. 為什么會卡死 因為是表單程式,理所當然就是看主執行緒此時正在做什么? 可以用 ~0s ; k 看一下便知。 0:000> k # ......

    uj5u.com 2023-04-18 08:33:10 more
  • SignalR, No Connection with that ID,IIS

    <a href="https://www.cnblogs.com/smartstar/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/u36196.jpg" alt="" /></a>...

    uj5u.com 2023-03-30 17:21:52 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:15:33 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:13:31 more
  • C#遍歷指定檔案夾中所有檔案的3種方法

    <a href="https://www.cnblogs.com/xbhp/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/957602/20230310105611.png" alt="" /></a&...

    uj5u.com 2023-03-27 14:46:55 more
  • C#/VB.NET:如何將PDF轉為PDF/A

    <a href="https://www.cnblogs.com/Carina-baby/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2859233/20220427162558.png" alt="" />...

    uj5u.com 2023-03-27 14:46:35 more
  • 武裝你的WEBAPI-OData聚合查詢

    <a href="https://www.cnblogs.com/podolski/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/616093/20140323000327.png" alt="" /><...

    uj5u.com 2023-03-27 14:46:16 more