检查NameValueCollection中是否存在密钥

本文关键字:存在 密钥 是否 NameValueCollection 检查 | 更新日期: 2023-09-27 18:28:56

有没有一种快速简单的方法可以在不循环的情况下检查NameValueCollection中是否存在键?

正在查找Dictionary.ContainsKey()或类似的内容。

当然,有很多方法可以解决这个问题。只是想知道是否有人能帮我挠痒痒。

检查NameValueCollection中是否存在密钥

来自MSDN:

在以下情况下,此属性返回null:

1) 如果没有找到指定的密钥;

所以你可以:

NameValueCollection collection = ...
string value = collection[key];
if (value == null) // key doesn't exist

2) 如果找到指定的键并且其关联值为null。

CCD_ 1先调用base.Get(),然后调用base.FindEntry(),后者在内部使用性能为O(1)的Hashtable

当键在集合中且其关联值为null时,此方法会处理这种情况。

private static bool ContainsKey(this NameValueCollection collection, string key) =>
    collection.Get(key) is not null || collection.AllKeys.Contains(key);

从C#9开始,您可以使用is not null,否则使用!= null

是的,您可以使用Linq检查AllKeys属性:

using System.Linq;
...
collection.AllKeys.Contains(key);

然而,Dictionary<string, string[]>更适合这个目的,可能是通过一种扩展方法创建的:

public static void Dictionary<string, string[]> ToDictionary(this NameValueCollection collection) 
{
    return collection.Cast<string>().ToDictionary(key => key, key => collection.GetValues(key));
}
var dictionary = collection.ToDictionary();
if (dictionary.ContainsKey(key))
{
   ...
}

我认为这些答案中没有一个是完全正确/最优的。NameValueCollection不仅不区分空值和缺失值,而且对其键也不区分大小写。因此,我认为一个完整的解决方案是:

public static bool ContainsKey(this NameValueCollection @this, string key)
{
    return @this.Get(key) != null 
        // I'm using Keys instead of AllKeys because AllKeys, being a mutable array,
        // can get out-of-sync if mutated (it weirdly re-syncs when you modify the collection).
        // I'm also not 100% sure that OrdinalIgnoreCase is the right comparer to use here.
        // The MSDN docs only say that the "default" case-insensitive comparer is used
        // but it could be current culture or invariant culture
        || @this.Keys.Cast<string>().Contains(key, StringComparer.OrdinalIgnoreCase);
}
queryItems.AllKeys.Contains(key)

请注意,密钥可能不是唯一的,并且比较通常区分大小写。如果你只想得到第一个匹配密钥的值,而不想为大小写烦恼,那么就使用这个:

        public string GetQueryValue(string queryKey)
        {
            foreach (string key in QueryItems)
            {
                if(queryKey.Equals(key, StringComparison.OrdinalIgnoreCase))
                    return QueryItems.GetValues(key).First(); // There might be multiple keys of the same name, but just return the first match
            }
            return null;
        }

您可以使用Get方法并检查collection[key]0,因为如果NameValueCollection不包含指定的键,该方法将返回null

请参阅MSDN。

如果集合的大小很小,可以使用rich.okelly提供的解决方案。但是,大的集合意味着生成字典的速度可能明显慢于只搜索密钥集合。

此外,如果您的使用场景是在不同的时间点搜索关键字,其中NameValueCollection可能已被修改,那么每次生成字典的速度可能会比只搜索关键字集合慢。

这也可以是一个无需引入新方法的解决方案:

    item = collection["item"] != null ? collection["item"].ToString() : null;

正如您在引用源中看到的,NameValueCollection继承自NameObjectCollectionBase

因此,您获取基类型,通过反射获取私有哈希表,并检查它是否包含特定的密钥。

为了在Mono中也能工作,您需要查看Mono中哈希表的名称,这是您可以在这里看到的(m_ItemsContainer),如果初始FieldInfo为null(Mono运行时),则获取Mono字段。

像这个

public static class ParameterExtensions
{
    private static System.Reflection.FieldInfo InitFieldInfo()
    {
        System.Type t = typeof(System.Collections.Specialized.NameObjectCollectionBase);
        System.Reflection.FieldInfo fi = t.GetField("_entriesTable", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
        if(fi == null) // Mono
            fi = t.GetField("m_ItemsContainer", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
        return fi;
    }
    private static System.Reflection.FieldInfo m_fi = InitFieldInfo();

    public static bool Contains(this System.Collections.Specialized.NameValueCollection nvc, string key)
    {
        //System.Collections.Specialized.NameValueCollection nvc = new System.Collections.Specialized.NameValueCollection();
        //nvc.Add("hello", "world");
        //nvc.Add("test", "case");
        // The Hashtable is case-INsensitive
        System.Collections.Hashtable ent = (System.Collections.Hashtable)m_fi.GetValue(nvc);
        return ent.ContainsKey(key);
    }
}

对于超纯的非反射.NET 2.0代码,您可以在键上循环,而不是使用哈希表,但这很慢。

private static bool ContainsKey(System.Collections.Specialized.NameValueCollection nvc, string key)
{
    foreach (string str in nvc.AllKeys)
    {
        if (System.StringComparer.InvariantCultureIgnoreCase.Equals(str, key))
            return true;
    }
    return false;
}

在VB中,它是:

if not MyNameValueCollection(Key) is Nothing then
.......
end if

在C#中应该只是:

if (MyNameValueCollection(Key) != null) { }

不确定它应该是null还是"",但这应该会有所帮助。

当我在小元素集合中工作时,我正在使用这个集合。

元素很多的地方,我觉得需要用"字典"。我的代码:

NameValueCollection ProdIdes;
string prodId = _cfg.ProdIdes[key];
if (string.IsNullOrEmpty(prodId))
{
    ......
}

或者可以使用这个:

 string prodId = _cfg.ProdIdes[key] !=null ? "found" : "not found";
NameValueCollection n = Request.QueryString;
if (n.HasKeys())
   {
       //something
   }

返回值类型:System.Boolean如果NameValueCollection包含不为null的键,则为true;否则为false。链接