- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我正在从事的项目需要一些简单的审计日志记录,以记录用户何时更改他们的电子邮件、帐单地址等。我们正在处理的对象来自不同的来源,一个是 WCF 服务,另一个是 Web服务。
我已经使用反射实现了以下方法来查找两个不同对象的属性更改。这将生成具有差异的属性列表以及它们的旧值和新值。
public static IList GenerateAuditLogMessages(T originalObject, T changedObject)
{
IList list = new List();
string className = string.Concat("[", originalObject.GetType().Name, "] ");
foreach (PropertyInfo property in originalObject.GetType().GetProperties())
{
Type comparable =
property.PropertyType.GetInterface("System.IComparable");
if (comparable != null)
{
string originalPropertyValue =
property.GetValue(originalObject, null) as string;
string newPropertyValue =
property.GetValue(changedObject, null) as string;
if (originalPropertyValue != newPropertyValue)
{
list.Add(string.Concat(className, property.Name,
" changed from '", originalPropertyValue,
"' to '", newPropertyValue, "'"));
}
}
}
return list;
}
我正在寻找 System.IComparable,因为“所有数值类型(例如 Int32 和 Double)都实现 IComparable,String、Char 和 DateTime 也是如此。”这似乎是查找非自定义类的任何属性的最佳方式。
利用 WCF 或 Web 服务代理代码生成的 PropertyChanged 事件听起来不错,但没有为我的审计日志(旧值和新值)提供足够的信息。
正在寻找关于是否有更好的方法来执行此操作的意见,谢谢!
@Aaronaught,这里是一些示例代码,它基于执行 object.Equals 生成正匹配:
Address address1 = new Address();
address1.StateProvince = new StateProvince();
Address address2 = new Address();
address2.StateProvince = new StateProvince();
IList list = Utility.GenerateAuditLogMessages(address1, address2);
"[Address] StateProvince changed from 'MyAccountService.StateProvince' to 'MyAccountService.StateProvince'"
它是 StateProvince 类的两个不同实例,但属性值相同(在本例中均为 null)。我们没有覆盖 equals 方法。
最佳答案
IComparable
用于排序比较。要么使用 IEquatable
相反,或者只使用静态 System.Object.Equals
方法。如果对象不是原始类型但仍通过覆盖 Equals
定义自己的相等比较,后者的好处是也可以工作。 .
object originalValue = property.GetValue(originalObject, null);
object newValue = property.GetValue(changedObject, null);
if (!object.Equals(originalValue, newValue))
{
string originalText = (originalValue != null) ?
originalValue.ToString() : "[NULL]";
string newText = (newText != null) ?
newValue.ToString() : "[NULL]";
// etc.
}
这显然不是完美的,但如果您只对您控制的类执行此操作,那么您可以确保它始终满足您的特定需求。
还有其他方法可以比较对象(例如校验和、序列化等),但如果类没有始终如一地实现 IPropertyChanged
,这可能是最可靠的方法。并且您想真正了解差异。
新示例代码的更新:
Address address1 = new Address();
address1.StateProvince = new StateProvince();
Address address2 = new Address();
address2.StateProvince = new StateProvince();
IList list = Utility.GenerateAuditLogMessages(address1, address2);
使用 object.Equals
的原因在您的审核方法中导致“命中”是因为实例实际上不相等!
当然,StateProvince
在这两种情况下都可能为空,但是 address1
和 address2
StateProvince
仍有非空值属性和每个实例都不同。因此,address1
和 address2
具有不同的属性。
我们反过来看,以这段代码为例:
Address address1 = new Address("35 Elm St");
address1.StateProvince = new StateProvince("TX");
Address address2 = new Address("35 Elm St");
address2.StateProvince = new StateProvince("AZ");
这些应该被认为是相等的吗?好吧,他们会用你的方法,因为StateProvince
不执行 IComparable
.这是您的方法报告两个对象在原始情况下相同的唯一原因。自 StateProvince
类没有实现 IComparable
,跟踪器完全跳过该属性。但是这两个地址显然不相等!
这就是我最初建议使用 object.Equals
的原因,因为这样你就可以在 StateProvince
中覆盖它获得更好结果的方法:
public class StateProvince
{
public string Code { get; set; }
public override bool Equals(object obj)
{
if (obj == null)
return false;
StateProvince sp = obj as StateProvince;
if (object.ReferenceEquals(sp, null))
return false;
return (sp.Code == Code);
}
public bool Equals(StateProvince sp)
{
if (object.ReferenceEquals(sp, null))
return false;
return (sp.Code == Code);
}
public override int GetHashCode()
{
return Code.GetHashCode();
}
public override string ToString()
{
return string.Format("Code: [{0}]", Code);
}
}
完成此操作后,object.Equals
代码将完美运行。而不是天真地检查是否 address1
和 address2
字面上有相同的StateProvince
引用,它实际上会检查语义是否相等。
另一种方法是扩展跟踪代码以实际下降到子对象中。换句话说,对于每个属性,检查 Type.IsClass
和可选的 Type.IsInterface
属性(property),如果true
,然后递归调用属性本身的更改跟踪方法,在递归返回的任何审计结果前加上属性名称。所以你最终会得到 StateProvinceCode
的变化.
我有时也会使用上面的方法,但是直接覆盖 Equals
会更容易在要比较语义相等性(即审计)的对象上并提供适当的 ToString
override 清楚地表明发生了什么变化。它不能针对深度嵌套进行扩展,但我认为以这种方式进行审计是不寻常的。
最后一个技巧是定义你自己的接口(interface),比如说IAuditable<T>
,它采用相同类型的第二个实例作为参数,并实际返回所有差异的列表(或可枚举的)。它类似于我们重写的 object.Equals
上面的方法,但会返回更多信息。当对象图非常复杂并且您知道不能依赖反射或 Equals
时,这很有用。 .您可以将其与上述方法结合使用;实际上,您所要做的就是替代IComparable
为你的IAuditable
并调用 Audit
方法(如果它实现了该接口(interface))。
关于c# - 查找两个 C# 对象之间的属性差异,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2387946/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!