連絡先データと共に、Exchange WebサービスマネージドAPIを使用しています。私は次のコードを持っています。これは 機能的 ですが、理想的ではありません。
foreach (Contact c in contactList)
{
string openItemUrl = "https://" + service.Url.Host + "/owa/" + c.WebClientReadFormQueryString;
row = table.NewRow();
row["FileAs"] = c.FileAs;
row["GivenName"] = c.GivenName;
row["Surname"] = c.Surname;
row["CompanyName"] = c.CompanyName;
row["Link"] = openItemUrl;
//home address
try { row["HomeStreet"] = c.PhysicalAddresses[PhysicalAddressKey.Home].Street.ToString(); }
catch (Exception e) { }
try { row["HomeCity"] = c.PhysicalAddresses[PhysicalAddressKey.Home].City.ToString(); }
catch (Exception e) { }
try { row["HomeState"] = c.PhysicalAddresses[PhysicalAddressKey.Home].State.ToString(); }
catch (Exception e) { }
try { row["HomeZip"] = c.PhysicalAddresses[PhysicalAddressKey.Home].PostalCode.ToString(); }
catch (Exception e) { }
try { row["HomeCountry"] = c.PhysicalAddresses[PhysicalAddressKey.Home].CountryOrRegion.ToString(); }
catch (Exception e) { }
//and so on for all kinds of other contact-related fields...
}
私が言ったように、このコード は動作します 。今、私はそれを吸うようにしたいです 少し少ない 、可能であれば。
アクセスしようとする前に辞書内のキーの存在を確認することができるメソッドが見つからず、(.ToString()
を使って)それを読み込もうとしても存在しない場合は例外がスローされます。
500
与えられたキーが辞書に存在しませんでした。
(まだ機能している間に)このコードを吸い込まないようにリファクタリングするにはどうすればよいですか。
ContainsKey
を使用できます。
if (dict.ContainsKey(key)) { ... }
または TryGetValue
:
dict.TryGetValue(key, out value);
更新 :コメントによると、ここの実際のクラスはIDictionary
ではなく PhysicalAddressDictionary
なので、メソッドは Contains
と TryGetValue
ですが同じように機能します。
使用例
PhysicalAddressEntry entry;
PhysicalAddressKey key = c.PhysicalAddresses[PhysicalAddressKey.Home].Street;
if (c.PhysicalAddresses.TryGetValue(key, out entry))
{
row["HomeStreet"] = entry;
}
更新2: これは作業コードです(質問者によって編集されたもの)
PhysicalAddressEntry entry;
PhysicalAddressKey key = PhysicalAddressKey.Home;
if (c.PhysicalAddresses.TryGetValue(key, out entry))
{
if (entry.Street != null)
{
row["HomeStreet"] = entry.Street.ToString();
}
}
...必要なキーごとに必要に応じて内側の条件式を繰り返します。 TryGetValueはPhysicalAddressKey(Home、Workなど)ごとに1回だけ行われます。
c.PhysicalAddresses
の種類は何ですか? Dictionary<TKey,TValue>
の場合は、 ContainsKey
メソッドを使用できます。
PhysicalAddressDictionary.TryGetValue
public bool TryGetValue (
PhysicalAddressKey key,
out PhysicalAddressEntry physicalAddress
)
私はDictionaryを使用していますが、反復性とキーの欠落が考えられるため、簡単な方法ですぐにパッチを当てました。
private static string GetKey(IReadOnlyDictionary<string, string> dictValues, string keyValue)
{
return dictValues.ContainsKey(keyValue) ? dictValues[keyValue] : "";
}
それを呼ぶ:
var entry = GetKey(dictList,"KeyValue1");
完了したジョブを取得します。
これが私が今日作り上げたちょっとしたことです。私のために働くようだ。基本的に、チェックをするためにあなたのベース名前空間のAddメソッドをオーバーライドし、それから実際にそれを追加するためにベースのAddメソッドを呼び出します。これがあなたのために働くことを願っています
using System;
using System.Collections.Generic;
using System.Collections;
namespace Main
{
internal partial class Dictionary<TKey, TValue> : System.Collections.Generic.Dictionary<TKey, TValue>
{
internal new virtual void Add(TKey key, TValue value)
{
if (!base.ContainsKey(key))
{
base.Add(key, value);
}
}
}
internal partial class List<T> : System.Collections.Generic.List<T>
{
internal new virtual void Add(T item)
{
if (!base.Contains(item))
{
base.Add(item);
}
}
}
public class Program
{
public static void Main()
{
Dictionary<int, string> dic = new Dictionary<int, string>();
dic.Add(1,"b");
dic.Add(1,"a");
dic.Add(2,"c");
dic.Add(1, "b");
dic.Add(1, "a");
dic.Add(2, "c");
string val = "";
dic.TryGetValue(1, out val);
Console.WriteLine(val);
Console.WriteLine(dic.Count.ToString());
List<string> lst = new List<string>();
lst.Add("b");
lst.Add("a");
lst.Add("c");
lst.Add("b");
lst.Add("a");
lst.Add("c");
Console.WriteLine(lst[2]);
Console.WriteLine(lst.Count.ToString());
}
}
}