生成随机密码


229

当我们网站上的用户丢失密码并转到“丢失密码”页面时,我们需要为他提供一个新的临时密码。我真的不介意这有多随机,或者如果它与所有“需要的”强密码规则匹配,我要做的就是给他们提供一个密码,以后可以更改。

该应用程序是用C#编写的Web应用程序。所以我当时想变得刻薄,并选择使用Guid的简单方法。即

Guid.NewGuid().ToString("d").Substring(1,8)

建议?有什么想法吗?


12
这里有一些不错的解决方案,但有几点建议:不要生成包含以下任何字符的密码:Oo0Ili(您知道为什么):)
stian.net 2013年

2
我添加了一个使用KeePass作为密码生成器的答案,在公开的众多选项中,我还包括了排除相似字符的选项,如@ stian.net所述。
彼得

Answers:


570

7
不知道框架有这种方法!太棒了!将为此交换我当前的代码!
FryHard

35
我花了近一天的时间完善自己的pw gen代码后才找到它。形象我的感受;)
瑞克

16
AFAIK此方法不会生成符合域密码策略的密码,因此它并不适合每种用法。
teebot '04

19
此解决方案的主要问题是您无法控制字符集,因此无法消除在视觉上模糊的字符(0oOl1i!|),这在实际中非常重要。
David Hammond 2014年

19
有什么用ASP.NET Core吗?
shashwat

113
public string CreatePassword(int length)
{
        const string valid = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890";
        StringBuilder res = new StringBuilder();
        Random rnd = new Random();
        while (0 < length--)
        {
            res.Append(valid[rnd.Next(valid.Length)]);
        }
        return res.ToString();
}

能够从可用字符列表中为生成的密码进行选择(例如,仅数字,仅大写或仅小写等),这具有很好的好处。


2
此方法(基于62)在强度方面优于GUID(基于16):8字符的十六进制字符串等效于4-5字符的字母数字字符串
Jimmy

57
Random不是加密安全的;System.Security.Cryptography.RNGCryptoServiceProvider是更好的选择。
anaximander

3
每次实例化Random类时,每次调用该方法都会生成相同的密码。通过将Random移出该方法并重新使用该实例,可以确保此方法的安全。
2014年

6
不,不会。除非两个人决定在同一时钟时间更改密码。
Radu094 2014年

10
问题引用:不在乎“如果它与所有“必需的”强密码规则都匹配” ...尽管感谢
投票否决

35

我的代码的主要目标是:

  1. 琴弦的分布几乎是均匀的(只要它们很小,就不必担心微小的偏差)
  2. 每个参数集输出超过数十亿个字符串。如果PRNG仅生成20亿个(熵的31位)不同的值,则生成8个字符串(约47位的熵)是没有意义的。
  3. 这是安全的,因为我希望人们将其用于密码或其他安全令牌。

第一个属性是通过将64位值乘以字母大小取模来实现的。对于小字母(例如问题中的62个字符),这导致的偏差可以忽略不计。通过使用RNGCryptoServiceProvider代替实现第二和第三属性System.Random

using System;
using System.Security.Cryptography;

public static string GetRandomAlphanumericString(int length)
{
    const string alphanumericCharacters =
        "ABCDEFGHIJKLMNOPQRSTUVWXYZ" +
        "abcdefghijklmnopqrstuvwxyz" +
        "0123456789";
    return GetRandomString(length, alphanumericCharacters);
}

public static string GetRandomString(int length, IEnumerable<char> characterSet)
{
    if (length < 0)
        throw new ArgumentException("length must not be negative", "length");
    if (length > int.MaxValue / 8) // 250 million chars ought to be enough for anybody
        throw new ArgumentException("length is too big", "length");
    if (characterSet == null)
        throw new ArgumentNullException("characterSet");
    var characterArray = characterSet.Distinct().ToArray();
    if (characterArray.Length == 0)
        throw new ArgumentException("characterSet must not be empty", "characterSet");

    var bytes = new byte[length * 8];
    new RNGCryptoServiceProvider().GetBytes(bytes);
    var result = new char[length];
    for (int i = 0; i < length; i++)
    {
        ulong value = BitConverter.ToUInt64(bytes, i * 8);
        result[i] = characterArray[value % (uint)characterArray.Length];
    }
    return new string(result);
}

(这是我的答案的副本,该答案如何在C#中生成随机的8个字符的字母数字字符串?


1
如果UInt64.MaxValue不能被characterArray.Length整除,则随机选择的字符将不会均匀分布(尽管这将是很小的效果)。
杰夫·沃克

1
@JeffWalkerCodeRanger这就是为什么我说的偏差可以忽略,而不是没有偏差。即使具有PB级的输出,您也只能将其与完全无偏的字符串生成器区分开来,不足1%。完全无偏的额外复杂性显然不值得在这里从理论上获得随机性。
CodesInChaos

9
对于使用.NET Core的用户,请替换为“ new RNGCryptoServiceProvider()。GetBytes(bytes);”。与“ System.Security.Cryptography.RandomNumberGenerator.Create()。GetBytes(bytes);”一起使用
NPNelson

24
public string GenerateToken(int length)
{
    using (RNGCryptoServiceProvider cryptRNG = new RNGCryptoServiceProvider())
    {
        byte[] tokenBuffer = new byte[length];
        cryptRNG.GetBytes(tokenBuffer);
        return Convert.ToBase64String(tokenBuffer);
    }
}

(您也可以使用该方法所在的类来实现IDisposable,保留对的引用RNGCryptoServiceProvider,并对其进行正确处理,以避免重复实例化它。)

注意,由于它返回一个以64为基数的字符串,因此输出长度始终是4的倍数,多余的空格=用作填充字符。该length参数指定字节缓冲区的长度,而不是输出字符串的长度(因此,现在考虑的可能不是该参数的最佳名称)。这控制密码将具有多少个字节。但是,由于base-64使用4个字符的块来对输入的每3个字节进行编码,因此,如果您要求的长度不是3的倍数,则会有一些额外的“空格”,它将使用=用于填充额外。

如果由于某种原因不喜欢使用base-64字符串,则可以用Convert.ToBase64String()转换为常规字符串或使用任何Encoding方法来替换调用。例如。Encoding.UTF8.GetString(tokenBuffer)-只需确保选择一个可以代表RNG值的完整范围的字符集,并且该字符集会产生与您要发送或存储它的位置兼容的字符。例如,使用Unicode往往会提供很多中文字符。使用base-64可以保证广泛兼容的字符集,并且只要您使用体面的哈希算法,这种字符串的特性就不会使其安全性降低。


我认为您的意思是将tokenBuffer放在具有linkBuf的位置。
PIntag 2015年

当我使用此代码并传递10个长度时,返回的字符串始终为16个字符长,最后2个字符始终为“ ==”。我使用不正确吗?长度以十六进制指定吗?
PIntag,2015年

1
指定的长度是随机性(或“熵”)的字节数,这在技术上是已知的。但是,返回值是base-64编码的,这意味着由于base-64编码的工作原理,输出长度始终是4的倍数。有时,它的字符数超过了对所有字符进行编码所需的字符数,因此它使用=字符填充其余部分。
anaximander

1
RNGCryptoServiceProvider是一个IDisposable,因此我将使用using模式(即,使用(var cryptRNG = new RNGCryptoServiceProvider()){...})来实现它
kloarubeek

@kloarubeek一个正确的观点,一个很好的建议。我已经将其添加到我的代码示例中。
anaximander

20

它更大了,但我认为它看起来更全面:http : //www.obviex.com/Samples/Password.aspx

///////////////////////////////////////////////////////////////////////////////
// SAMPLE: Generates random password, which complies with the strong password
//         rules and does not contain ambiguous characters.
//
// To run this sample, create a new Visual C# project using the Console
// Application template and replace the contents of the Class1.cs file with
// the code below.
//
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND,
// EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED
// WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR PURPOSE.
// 
// Copyright (C) 2004 Obviex(TM). All rights reserved.
// 
using System;
using System.Security.Cryptography;

/// <summary>
/// This class can generate random passwords, which do not include ambiguous 
/// characters, such as I, l, and 1. The generated password will be made of
/// 7-bit ASCII symbols. Every four characters will include one lower case
/// character, one upper case character, one number, and one special symbol
/// (such as '%') in a random order. The password will always start with an
/// alpha-numeric character; it will not start with a special symbol (we do
/// this because some back-end systems do not like certain special
/// characters in the first position).
/// </summary>
public class RandomPassword
{
    // Define default min and max password lengths.
    private static int DEFAULT_MIN_PASSWORD_LENGTH  = 8;
    private static int DEFAULT_MAX_PASSWORD_LENGTH  = 10;

    // Define supported password characters divided into groups.
    // You can add (or remove) characters to (from) these groups.
    private static string PASSWORD_CHARS_LCASE  = "abcdefgijkmnopqrstwxyz";
    private static string PASSWORD_CHARS_UCASE  = "ABCDEFGHJKLMNPQRSTWXYZ";
    private static string PASSWORD_CHARS_NUMERIC= "23456789";
    private static string PASSWORD_CHARS_SPECIAL= "*$-+?_&=!%{}/";

    /// <summary>
    /// Generates a random password.
    /// </summary>
    /// <returns>
    /// Randomly generated password.
    /// </returns>
    /// <remarks>
    /// The length of the generated password will be determined at
    /// random. It will be no shorter than the minimum default and
    /// no longer than maximum default.
    /// </remarks>
    public static string Generate()
    {
        return Generate(DEFAULT_MIN_PASSWORD_LENGTH, 
                        DEFAULT_MAX_PASSWORD_LENGTH);
    }

    /// <summary>
    /// Generates a random password of the exact length.
    /// </summary>
    /// <param name="length">
    /// Exact password length.
    /// </param>
    /// <returns>
    /// Randomly generated password.
    /// </returns>
    public static string Generate(int length)
    {
        return Generate(length, length);
    }

    /// <summary>
    /// Generates a random password.
    /// </summary>
    /// <param name="minLength">
    /// Minimum password length.
    /// </param>
    /// <param name="maxLength">
    /// Maximum password length.
    /// </param>
    /// <returns>
    /// Randomly generated password.
    /// </returns>
    /// <remarks>
    /// The length of the generated password will be determined at
    /// random and it will fall with the range determined by the
    /// function parameters.
    /// </remarks>
    public static string Generate(int   minLength,
                                  int   maxLength)
    {
        // Make sure that input parameters are valid.
        if (minLength <= 0 || maxLength <= 0 || minLength > maxLength)
            return null;

        // Create a local array containing supported password characters
        // grouped by types. You can remove character groups from this
        // array, but doing so will weaken the password strength.
        char[][] charGroups = new char[][] 
        {
            PASSWORD_CHARS_LCASE.ToCharArray(),
            PASSWORD_CHARS_UCASE.ToCharArray(),
            PASSWORD_CHARS_NUMERIC.ToCharArray(),
            PASSWORD_CHARS_SPECIAL.ToCharArray()
        };

        // Use this array to track the number of unused characters in each
        // character group.
        int[] charsLeftInGroup = new int[charGroups.Length];

        // Initially, all characters in each group are not used.
        for (int i=0; i<charsLeftInGroup.Length; i++)
            charsLeftInGroup[i] = charGroups[i].Length;

        // Use this array to track (iterate through) unused character groups.
        int[] leftGroupsOrder = new int[charGroups.Length];

        // Initially, all character groups are not used.
        for (int i=0; i<leftGroupsOrder.Length; i++)
            leftGroupsOrder[i] = i;

        // Because we cannot use the default randomizer, which is based on the
        // current time (it will produce the same "random" number within a
        // second), we will use a random number generator to seed the
        // randomizer.

        // Use a 4-byte array to fill it with random bytes and convert it then
        // to an integer value.
        byte[] randomBytes = new byte[4];

        // Generate 4 random bytes.
        RNGCryptoServiceProvider rng = new RNGCryptoServiceProvider();
        rng.GetBytes(randomBytes);

        // Convert 4 bytes into a 32-bit integer value.
        int seed = BitConverter.ToInt32(randomBytes, 0);

        // Now, this is real randomization.
        Random  random  = new Random(seed);

        // This array will hold password characters.
        char[] password = null;

        // Allocate appropriate memory for the password.
        if (minLength < maxLength)
            password = new char[random.Next(minLength, maxLength+1)];
        else
            password = new char[minLength];

        // Index of the next character to be added to password.
        int nextCharIdx;

        // Index of the next character group to be processed.
        int nextGroupIdx;

        // Index which will be used to track not processed character groups.
        int nextLeftGroupsOrderIdx;

        // Index of the last non-processed character in a group.
        int lastCharIdx;

        // Index of the last non-processed group.
        int lastLeftGroupsOrderIdx = leftGroupsOrder.Length - 1;

        // Generate password characters one at a time.
        for (int i=0; i<password.Length; i++)
        {
            // If only one character group remained unprocessed, process it;
            // otherwise, pick a random character group from the unprocessed
            // group list. To allow a special character to appear in the
            // first position, increment the second parameter of the Next
            // function call by one, i.e. lastLeftGroupsOrderIdx + 1.
            if (lastLeftGroupsOrderIdx == 0)
                nextLeftGroupsOrderIdx = 0;
            else
                nextLeftGroupsOrderIdx = random.Next(0, 
                                                     lastLeftGroupsOrderIdx);

            // Get the actual index of the character group, from which we will
            // pick the next character.
            nextGroupIdx = leftGroupsOrder[nextLeftGroupsOrderIdx];

            // Get the index of the last unprocessed characters in this group.
            lastCharIdx = charsLeftInGroup[nextGroupIdx] - 1;

            // If only one unprocessed character is left, pick it; otherwise,
            // get a random character from the unused character list.
            if (lastCharIdx == 0)
                nextCharIdx = 0;
            else
                nextCharIdx = random.Next(0, lastCharIdx+1);

            // Add this character to the password.
            password[i] = charGroups[nextGroupIdx][nextCharIdx];

            // If we processed the last character in this group, start over.
            if (lastCharIdx == 0)
                charsLeftInGroup[nextGroupIdx] = 
                                          charGroups[nextGroupIdx].Length;
            // There are more unprocessed characters left.
            else
            {
                // Swap processed character with the last unprocessed character
                // so that we don't pick it until we process all characters in
                // this group.
                if (lastCharIdx != nextCharIdx)
                {
                    char temp = charGroups[nextGroupIdx][lastCharIdx];
                    charGroups[nextGroupIdx][lastCharIdx] = 
                                charGroups[nextGroupIdx][nextCharIdx];
                    charGroups[nextGroupIdx][nextCharIdx] = temp;
                }
                // Decrement the number of unprocessed characters in
                // this group.
                charsLeftInGroup[nextGroupIdx]--;
            }

            // If we processed the last group, start all over.
            if (lastLeftGroupsOrderIdx == 0)
                lastLeftGroupsOrderIdx = leftGroupsOrder.Length - 1;
            // There are more unprocessed groups left.
            else
            {
                // Swap processed group with the last unprocessed group
                // so that we don't pick it until we process all groups.
                if (lastLeftGroupsOrderIdx != nextLeftGroupsOrderIdx)
                {
                    int temp = leftGroupsOrder[lastLeftGroupsOrderIdx];
                    leftGroupsOrder[lastLeftGroupsOrderIdx] = 
                                leftGroupsOrder[nextLeftGroupsOrderIdx];
                    leftGroupsOrder[nextLeftGroupsOrderIdx] = temp;
                }
                // Decrement the number of unprocessed groups.
                lastLeftGroupsOrderIdx--;
            }
        }

        // Convert password characters into a string and return the result.
        return new string(password);
     }
}

/// <summary>
/// Illustrates the use of the RandomPassword class.
/// </summary>
public class RandomPasswordTest
{
    /// <summary>
    /// The main entry point for the application.
    /// </summary>
    [STAThread]
    static void Main(string[] args)
    {
        // Print 100 randomly generated passwords (8-to-10 char long).
        for (int i=0; i<100; i++)
            Console.WriteLine(RandomPassword.Generate(8, 10));
    }
}
//
// END OF FILE
///////////////////////////////////////////////////////////////////////////////

2
事实证明,框架对此提供了支持。所以我宁愿接受那个答案!
FryHard

1
即使仅生成长输出大小的密码,也只能生成2 ^ 31个不同的密码,这有点不足。抵御在线攻击可能就足够了,但是对于脱机攻击肯定很小。=>我不建议这样做。
CodesInChaos

这仍然是一个很好的答案,因为“内置”支持实际上是成员资格,如果您决定不使用ASP.NET成员资格怎么办?由于依赖项为System.Web.dll,因此它仍然有效,但是由于该方法不是自包含的,因此有点尴尬。@GEOCHET:感谢您发布此替代方法。
克里斯·戈麦斯

8

我知道这是一个旧线程,但是对于某些人来说,我有一个相当简单的解决方案。易于实施,易于理解和易于验证。

请考虑以下要求:

我需要生成一个随机密码,该密码至少包含2个小写字母,2个大写字母和2个数字。密码的长度也必须至少为8个字符。

以下正则表达式可以验证这种情况:

^(?=\b\w*[a-z].*[a-z]\w*\b)(?=\b\w*[A-Z].*[A-Z]\w*\b)(?=\b\w*[0-9].*[0-9]\w*\b)[a-zA-Z0-9]{8,}$

它不在此问题的范围内-但正则表达式基于lookahead / lookbehindlookaround

以下代码将创建一个符合此要求的随机字符集:

public static string GeneratePassword(int lowercase, int uppercase, int numerics) {
    string lowers = "abcdefghijklmnopqrstuvwxyz";
    string uppers = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
    string number = "0123456789";

    Random random = new Random();

    string generated = "!";
    for (int i = 1; i <= lowercase; i++)
        generated = generated.Insert(
            random.Next(generated.Length), 
            lowers[random.Next(lowers.Length - 1)].ToString()
        );

    for (int i = 1; i <= uppercase; i++)
        generated = generated.Insert(
            random.Next(generated.Length), 
            uppers[random.Next(uppers.Length - 1)].ToString()
        );

    for (int i = 1; i <= numerics; i++)
        generated = generated.Insert(
            random.Next(generated.Length), 
            number[random.Next(number.Length - 1)].ToString()
        );

    return generated.Replace("!", string.Empty);

}

为了满足上述要求,只需调用以下命令:

String randomPassword = GeneratePassword(3, 3, 3);

该代码以无效字符开头("!") -字符串具有可插入新字符的长度。

然后,它从1循环到所需的小写字母#,并在每次迭代中从小写字母列表中获取随机项,并将其插入字符串中的随机位置。

然后,对大写字母和数字重复循环。

这将使您返回length =的字符串,lowercase + uppercase + numerics其中您想要的计数的小写,大写和数字字符已按随机顺序放置。


3
请勿System.Random用于安全性至关重要的内容,例如密码。使用RNGCryptoServiceProvider
CodesInChaos

lowers[random.Next(lowers.Length - 1)].ToString() 此代码永远不会生成“ z”。random.Next产生的整数小于提供的整数,因此您不应从长度中减去多余的整数。
notbono

6

对于这种密码,我倾向于使用一种系统,该系统很可能会生成更容易使用的密码。简短,通常由可发音的片段和一些数字组成,并且没有字符间的歧义(是0还是O?A 1或I?)。就像是

string[] words = { 'bur', 'ler', 'meh', 'ree' };
string word = "";

Random rnd = new Random();
for (i = 0; i < 3; i++)
   word += words[rnd.Next(words.length)]

int numbCount = rnd.Next(4);
for (i = 0; i < numbCount; i++)
  word += (2 + rnd.Next(7)).ToString();

return word;

(直接在浏览器中键入,因此仅用作准则。此外,添加更多单词)。


6

我不喜欢Membership.GeneratePassword()创建的密码,因为它们太丑陋且包含太多特殊字符。

此代码生成一个10位数字,不太难看。

string password = Guid.NewGuid().ToString("N").ToLower()
                      .Replace("1", "").Replace("o", "").Replace("0","")
                      .Substring(0,10);

当然,我可以使用Regex进行所有替换,但这是更易读和可维护的IMO。


2
一个GUID不应被滥用为加密PRNG
CodesInChaos

3
如果要使用此方法,则可以使用.ToString(“ N”),而不必替换“-”。也不必替换“ l”,因为它不是十六进制数字。
JackAce 2014年

完全了解您为什么这样做。我只需要一个短暂的密码(不到一天),每个用户的密码不必太独特。剔除1和0是消除混乱的好方法。我也只是将我的代码转换为大写字母,然后将其长度削减为6。就像“ N”建议一样。谢谢!
比尔·诺埃尔

6

我创建了这个类使用RNGCryptoServiceProvider,它是灵活的。例:

var generator = new PasswordGenerator(minimumLengthPassword: 8,
                                      maximumLengthPassword: 15,
                                      minimumUpperCaseChars: 2,
                                      minimumNumericChars: 3,
                                      minimumSpecialChars: 2);
string password = generator.Generate();

太好了,该课程的许可证是多少?我可以使用吗?
codeulike

随心所欲使用它!
亚历克斯·席普曼

3

我创建了类似于成员资格提供程序中可用方法的方法。如果您不想在某些应用程序中添加Web参考,这将很有用。

效果很好。

public static string GeneratePassword(int Length, int NonAlphaNumericChars)
    {
        string allowedChars = "abcdefghijkmnopqrstuvwxyzABCDEFGHJKLMNOPQRSTUVWXYZ0123456789";
        string allowedNonAlphaNum = "!@#$%^&*()_-+=[{]};:<>|./?";
        Random rd = new Random();

        if (NonAlphaNumericChars > Length || Length <= 0 || NonAlphaNumericChars < 0)
            throw new ArgumentOutOfRangeException();

            char[] pass = new char[Length];
            int[] pos = new int[Length];
            int i = 0, j = 0, temp = 0;
            bool flag = false;

            //Random the position values of the pos array for the string Pass
            while (i < Length - 1)
            {
                j = 0;
                flag = false;
                temp = rd.Next(0, Length);
                for (j = 0; j < Length; j++)
                    if (temp == pos[j])
                    {
                        flag = true;
                        j = Length;
                    }

                if (!flag)
                {
                    pos[i] = temp;
                    i++;
                }
            }

            //Random the AlphaNumericChars
            for (i = 0; i < Length - NonAlphaNumericChars; i++)
                pass[i] = allowedChars[rd.Next(0, allowedChars.Length)];

            //Random the NonAlphaNumericChars
            for (i = Length - NonAlphaNumericChars; i < Length; i++)
                pass[i] = allowedNonAlphaNum[rd.Next(0, allowedNonAlphaNum.Length)];

            //Set the sorted array values by the pos array for the rigth posistion
            char[] sorted = new char[Length];
            for (i = 0; i < Length; i++)
                sorted[i] = pass[pos[i]];

            string Pass = new String(sorted);

            return Pass;
    }

6
请勿System.Random用于安全性至关重要的内容,例如密码。使用RNGCryptoServiceProvider
CodesInChaos

3

我对KeePass内置的密码生成器一直很满意。由于KeePass是.Net程序和开源程序,因此我决定对代码进行深入研究。我最终只是引用了KeePass.exe(标准应用程序安装中提供的副本)作为我项目中的参考,并编写了以下代码。借助KeePass,您可以看到它有多灵活。您可以指定长度,要包含/不包含的字符等。

using KeePassLib.Cryptography.PasswordGenerator;
using KeePassLib.Security;


public static string GeneratePassword(int passwordLength, bool lowerCase, bool upperCase, bool digits,
        bool punctuation, bool brackets, bool specialAscii, bool excludeLookAlike)
    {
        var ps = new ProtectedString();
        var profile = new PwProfile();
        profile.CharSet = new PwCharSet();
        profile.CharSet.Clear();

        if (lowerCase)
            profile.CharSet.AddCharSet('l');
        if(upperCase)
            profile.CharSet.AddCharSet('u');
        if(digits)
            profile.CharSet.AddCharSet('d');
        if (punctuation)
            profile.CharSet.AddCharSet('p');
        if (brackets)
            profile.CharSet.AddCharSet('b');
        if (specialAscii)
            profile.CharSet.AddCharSet('s');

        profile.ExcludeLookAlike = excludeLookAlike;
        profile.Length = (uint)passwordLength;
        profile.NoRepeatingCharacters = true;

        KeePassLib.Cryptography.PasswordGenerator.PwGenerator.Generate(out ps, profile, null, _pool);

        return ps.ReadString();
    }

2
public static string GeneratePassword(int passLength) {
        var chars = "abcdefghijklmnopqrstuvwxyz@#$&ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
        var random = new Random();
        var result = new string(
            Enumerable.Repeat(chars, passLength)
                      .Select(s => s[random.Next(s.Length)])
                      .ToArray());
        return result;
    }

请解释您的答案
Linus

1
进行一些更改后,此功能可以很好地发挥作用。1.如果一次执行多个呼叫,则增加了熵。这意味着调用Sleep()或其他一些导致调用之间的函数的熵。2.增加源字符的数量和随机性。我创建了一系列包含500个char密码的keypass,但不包括一堆我不想处理的类似字符和其他可转义字符,并使用了产生的2000个字符串。即使只有100毫秒的熵,其随机性也非常好。
Wizengamot

2

我将在锅中添加另一个不明智的答案。

我有一个用例,在该情况下,我需要随机密码来进行机器与机器的通信,因此对人类可读性没有任何要求。我也没有访问Membership.GeneratePassword我的项目的权限,也不想添加依赖项。

我相当确定Membership.GeneratePassword正在做类似的事情,但是在这里您可以调整要提取的字符池。

public static class PasswordGenerator
{
    private readonly static Random _rand = new Random();

    public static string Generate(int length = 24)
    {
        const string lower = "abcdefghijklmnopqrstuvwxyz";
        const string upper = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        const string number = "1234567890";
        const string special = "!@#$%^&*_-=+";

        // Get cryptographically random sequence of bytes
        var bytes = new byte[length];
        new RNGCryptoServiceProvider().GetBytes(bytes);

        // Build up a string using random bytes and character classes
        var res = new StringBuilder();
        foreach(byte b in bytes)
        {
            // Randomly select a character class for each byte
            switch (_rand.Next(4))
            {
                // In each case use mod to project byte b to the correct range
                case 0:
                    res.Append(lower[b % lower.Count()]);
                    break;
                case 1:
                    res.Append(upper[b % upper.Count()]);
                    break;
                case 2:
                    res.Append(number[b % number.Count()]);
                    break;
                case 3:
                    res.Append(special[b % special.Count()]);
                    break;
            }
        }
        return res.ToString();
    }
}

和一些示例输出:

PasswordGenerator.Generate(12)
"pzY=64@-ChS$"
"BG0OsyLbYnI_"
"l9#5^2&adj_i"
"#++Ws9d$%O%X"
"IWhdIN-#&O^s"

为了避免有关以下方面的投诉Random:随机性的主要来源仍然是加密RNG。即使您可以确定性地预先确定要产生的序列Random(例如,它仅产生1),您仍然不会知道将要选择的下一个字符(尽管那样限制可能性的范围)。

一种简单的扩展是为不同的字符集增加权重,就像增加最大值并添加掉落情况以增加权重一样简单。

switch (_rand.Next(6))
{
    // Prefer letters 2:1
    case 0:
    case 1:
        res.Append(lower[b % lower.Count()]);
        break;
    case 2:
    case 3:
        res.Append(upper[b % upper.Count()]);
        break;
    case 4:
        res.Append(number[b % number.Count()]);
        break;
    case 5:
        res.Append(special[b % special.Count()]);
        break;
}

对于更人性化的随机密码生成器,我曾经使用EFF骰子列表实现了提示系统。


1

我喜欢查看生成密码,就像生成软件密钥一样。您应该从遵循良好做法的字符数组中进行选择。接受@ Radu094回答的内容,并对其进行修改以遵循良好实践。不要将每个字母都放在字符数组中。通过电话很难说出或理解一些字母。

您还应该考虑对生成的密码使用校验和,以确保它是由您生成的。实现此目的的一种好方法是使用LUHN算法


1

这是我快速整理的内容。

    public string GeneratePassword(int len)
    {
        string res = "";
        Random rnd = new Random();
        while (res.Length < len) res += (new Func<Random, string>((r) => {
            char c = (char)((r.Next(123) * DateTime.Now.Millisecond % 123)); 
            return (Char.IsLetterOrDigit(c)) ? c.ToString() : ""; 
        }))(rnd);
        return res;
    }

1

我使用此代码生成具有字母,数字和non_alpha_numeric字符的余额组成的密码。

public static string GeneratePassword(int Length, int NonAlphaNumericChars)
    {
        string allowedChars = "abcdefghijkmnopqrstuvwxyzABCDEFGHJKLMNOPQRSTUVWXYZ0123456789";
        string allowedNonAlphaNum = "!@#$%^&*()_-+=[{]};:<>|./?";
        string pass = "";
        Random rd = new Random(DateTime.Now.Millisecond);
        for (int i = 0; i < Length; i++)
        {
            if (rd.Next(1) > 0 && NonAlphaNumericChars > 0)
            {
                pass += allowedNonAlphaNum[rd.Next(allowedNonAlphaNum.Length)];
                NonAlphaNumericChars--;
            }
            else
            {
                pass += allowedChars[rd.Next(allowedChars.Length)];
            }
        }
        return pass;
    }

0

这很短,对我来说很棒。

public static string GenerateRandomCode(int length)
{
    Random rdm = new Random();
    StringBuilder sb = new StringBuilder();

    for(int i = 0; i < length; i++)
        sb.Append(Convert.ToChar(rdm.Next(101,132)));

    return sb.ToString();
}

它可能“起作用”,但是肯定不是安全的。密码必须安全。
CodesInChaos 2014年

我认为随机密码是临时密码。不知道为什么必须要保证它们的安全性,即使必须这样做也可以在范围内添加数字和特殊字符。
user1058637 2014年

1
如果使用可预测的PRNG生成数字和特殊字符,则不能提高安全性。如果您知道何时生成密码,可以将其范围缩小到仅几个候选者。
CodesInChaos

0

在我的网站上,我使用以下方法:

    //Symb array
    private const string _SymbolsAll = "~`!@#$%^&*()_+=-\\|[{]}'\";:/?.>,<";

    //Random symb
    public string GetSymbol(int Length)
    {
        Random Rand = new Random(DateTime.Now.Millisecond);
        StringBuilder result = new StringBuilder();
        for (int i = 0; i < Length; i++)
            result.Append(_SymbolsAll[Rand.Next(0, _SymbolsAll.Length)]);
        return result.ToString();
    }

编辑_SymbolsAll数组列表的字符串。


如编辑说明中所述。如果您在没有代码参考的情况下链接到您的网站,则纯粹是在宣传它会成为垃圾邮件,因此请从您的答案中删除该链接。
Bowdzone

0

在接受的答案中添加了一些补充代码。仅使用“随机”即可改善答案,并允许使用某些密码选项。我也喜欢KeePass答案中的一些选项,但不想在我的解决方案中包含可执行文件。

private string RandomPassword(int length, bool includeCharacters, bool includeNumbers, bool includeUppercase, bool includeNonAlphaNumericCharacters, bool includeLookAlikes)
{
    if (length < 8 || length > 128) throw new ArgumentOutOfRangeException("length");
    if (!includeCharacters && !includeNumbers && !includeNonAlphaNumericCharacters) throw new ArgumentException("RandomPassword-Key arguments all false, no values would be returned");

    string pw = "";
    do
    {
        pw += System.Web.Security.Membership.GeneratePassword(128, 25);
        pw = RemoveCharacters(pw, includeCharacters, includeNumbers, includeUppercase, includeNonAlphaNumericCharacters, includeLookAlikes);
    } while (pw.Length < length);

    return pw.Substring(0, length);
}

private string RemoveCharacters(string passwordString, bool includeCharacters, bool includeNumbers, bool includeUppercase, bool includeNonAlphaNumericCharacters, bool includeLookAlikes)
{
    if (!includeCharacters)
    {
        var remove = new string[] { "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z" };
        foreach (string r in remove)
        {
            passwordString = passwordString.Replace(r, string.Empty);
            passwordString = passwordString.Replace(r.ToUpper(), string.Empty);
        }
    }

    if (!includeNumbers)
    {
        var remove = new string[] { "0", "1", "2", "3", "4", "5", "6", "7", "8", "9" };
        foreach (string r in remove)
            passwordString = passwordString.Replace(r, string.Empty);
    }

    if (!includeUppercase)
        passwordString = passwordString.ToLower();

    if (!includeNonAlphaNumericCharacters)
    {
        var remove = new string[] { "!", "@", "#", "$", "%", "^", "&", "*", "(", ")", "-", "_", "+", "=", "{", "}", "[", "]", "|", "\\", ":", ";", "<", ">", "/", "?", "." };
        foreach (string r in remove)
            passwordString = passwordString.Replace(r, string.Empty);
    }

    if (!includeLookAlikes)
    {
        var remove = new string[] { "(", ")", "0", "O", "o", "1", "i", "I", "l", "|", "!", ":", ";" };
        foreach (string r in remove)
            passwordString = passwordString.Replace(r, string.Empty);
    }

    return passwordString;
}

这是我搜索生成随机密码时的第一个链接,以下内容不在当前问题的范围内,但可能需要考虑。

  • 基于这样的假设:System.Web.Security.Membership.GeneratePassword密码安全,至少20%的字符为非字母数字。
  • 不知道在这种情况下是否删除字符并附加字符串是否被认为是一种好习惯,并且是否提供了足够的熵。
  • 可能希望考虑以SecureString的某种方式实现将密码安全存储在内存中。

仅供参考,您无需包含KeyPass可执行文件,因为它是开源的(源代码可在此处下载)
Alex Klaus

0

validChars可以是任何构造,但是我决定根据ASCII代码范围选择删除控制字符。在此示例中,它是12个字符串。

string validChars = String.Join("", Enumerable.Range(33, (126 - 33)).Where(i => !(new int[] { 34, 38, 39, 44, 60, 62, 96 }).Contains(i)).Select(i => { return (char)i; }));
string.Join("", Enumerable.Range(1, 12).Select(i => { return validChars[(new Random(Guid.NewGuid().GetHashCode())).Next(0, validChars.Length - 1)]; }))

0
 Generate random password of specified length with 
  - Special characters   
  - Number
  - Lowecase
  - Uppercase

  public static string CreatePassword(int length = 12)
    {
        const string lower = "abcdefghijklmnopqrstuvwxyz";
        const string upper = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
        const string number = "1234567890";
        const string special = "!@#$%^&*";

        var middle = length / 2;
        StringBuilder res = new StringBuilder();
        Random rnd = new Random();
        while (0 < length--)
        {
            if (middle == length)
            {
                res.Append(number[rnd.Next(number.Length)]);
            }
            else if (middle - 1 == length)
            {
                res.Append(special[rnd.Next(special.Length)]);
            }
            else
            {
                if (length % 2 == 0)
                {
                    res.Append(lower[rnd.Next(lower.Length)]);
                }
                else
                {
                    res.Append(upper[rnd.Next(upper.Length)]);
                }
            }
        }
        return res.ToString();
    }

代码仅回答了一些令人鼓舞的问题,因为它们没有为未来的读者提供太多信息,请提供您所写内容的一些解释
WhatsThePoint

一遍又一遍地创建相同的密码。需要具有Randomas静态实例
Trailmax

请注意,如果连续多次调用此答案,将会生成相同的密码!!!如果您希望使用该功能使用代码一次将密码分配给多个记录,则在调用之间必须添加熵。虽然仍然有用....
Wizengamot


0

如果要使用System.Web.Security.Membership.GeneratePassword使用的加密安全的随机数生成,但又想将字符集限制为字母数字字符,则可以使用正则表达式过滤结果:

static string GeneratePassword(int characterCount)
{
    string password = String.Empty;
    while(password.Length < characterCount)
        password += Regex.Replace(System.Web.Security.Membership.GeneratePassword(128, 0), "[^a-zA-Z0-9]", string.Empty);
    return password.Substring(0, characterCount);
}

-3
public string Sifre_Uret(int boy, int noalfa)
{

    //  01.03.2016   
    // Genel amaçlı şifre üretme fonksiyonu


    //Fonskiyon 128 den büyük olmasına izin vermiyor.
    if (boy > 128 ) { boy = 128; }
    if (noalfa > 128) { noalfa = 128; }
    if (noalfa > boy) { noalfa = boy; }


    string passch = System.Web.Security.Membership.GeneratePassword(boy, noalfa);

    //URL encoding ve Url Pass + json sorunu yaratabilecekler pass ediliyor.
    //Microsoft Garanti etmiyor. Alfa Sayısallar Olabiliyorimiş . !@#$%^&*()_-+=[{]};:<>|./?.
    //https://msdn.microsoft.com/tr-tr/library/system.web.security.membership.generatepassword(v=vs.110).aspx


    //URL ve Json ajax lar için filtreleme
    passch = passch.Replace(":", "z");
    passch = passch.Replace(";", "W");
    passch = passch.Replace("'", "t");
    passch = passch.Replace("\"", "r");
    passch = passch.Replace("/", "+");
    passch = passch.Replace("\\", "e");

    passch = passch.Replace("?", "9");
    passch = passch.Replace("&", "8");
    passch = passch.Replace("#", "D");
    passch = passch.Replace("%", "u");
    passch = passch.Replace("=", "4");
    passch = passch.Replace("~", "1");

    passch = passch.Replace("[", "2");
    passch = passch.Replace("]", "3");
    passch = passch.Replace("{", "g");
    passch = passch.Replace("}", "J");


    //passch = passch.Replace("(", "6");
    //passch = passch.Replace(")", "0");
    //passch = passch.Replace("|", "p");
    //passch = passch.Replace("@", "4");
    //passch = passch.Replace("!", "u");
    //passch = passch.Replace("$", "Z");
    //passch = passch.Replace("*", "5");
    //passch = passch.Replace("_", "a");

    passch = passch.Replace(",", "V");
    passch = passch.Replace(".", "N");
    passch = passch.Replace("+", "w");
    passch = passch.Replace("-", "7");





    return passch;



}

您能否解释一下代码的作用,为什么?评论来自
Wai Ha Lee

仅代码无答案的答案可能会被删除。
Rook

-4

插入一个计时器:timer1,2个按钮:button1,button2,1个文本框:textBox1,和一个组合框:comboBox1。确保您声明:

int count = 0;

源代码:

 private void button1_Click(object sender, EventArgs e)
    {
    // This clears the textBox, resets the count, and starts the timer
        count = 0;
        textBox1.Clear();
        timer1.Start();
    }

    private void timer1_Tick(object sender, EventArgs e)
    {
    // This generates the password, and types it in the textBox
        count += 1;
            string possible = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890";
            string psw = "";
            Random rnd = new Random { };
            psw += possible[rnd.Next(possible.Length)];
            textBox1.Text += psw;
            if (count == (comboBox1.SelectedIndex + 1))
            {
                timer1.Stop();
            }
    }
    private void Form1_Load(object sender, EventArgs e)
    {
        // This adds password lengths to the comboBox to choose from.
        comboBox1.Items.Add("1");
        comboBox1.Items.Add("2");
        comboBox1.Items.Add("3");
        comboBox1.Items.Add("4");
        comboBox1.Items.Add("5");
        comboBox1.Items.Add("6");
        comboBox1.Items.Add("7");
        comboBox1.Items.Add("8");
        comboBox1.Items.Add("9");
        comboBox1.Items.Add("10");
        comboBox1.Items.Add("11");
        comboBox1.Items.Add("12");
    }
    private void button2_click(object sender, EventArgs e)
    {
        // This encrypts the password
        tochar = textBox1.Text;
        textBox1.Clear();
        char[] carray = tochar.ToCharArray();
        for (int i = 0; i < carray.Length; i++)
        {
            int num = Convert.ToInt32(carray[i]) + 10;
            string cvrt = Convert.ToChar(num).ToString();
            textBox1.Text += cvrt;
        }
    }

1
请勿System.Random用于安全性。
CodesInChaos
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.