热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

应该使用ToEntity<T>代替演员?

如何解决《应该使用ToEntity<T>代替演员?》经验,为你挑选了2个好方法。

所述XRM SDK定义一个ToEntity方法.我一直用它来从CRM中获取早期绑定的实体.今天我正在审查一些代码并看到实体刚刚被投射:

var cOntact= service.Retrieve("contact", id, new ColumnSet()) as Contact;

我甚至都不知道这是可能的.是否甚至需要ToEntity电话?



1> James Wood..:

除此之外,这不是一个特定的演员,它是一个转换,并且转换的行为与直接演员的行为略有不同.

您可以使用as运算符在兼容的引用类型或可空类型之间执行某些类型的转换... as运算符就像一个转换操作.但是,如果无法进行转换,则返回null而不是引发异常.

我假设你Contact是CrmSvcUtil例如,创建一个类public partial class Contact : Microsoft.Xrm.Sdk.Entity,并且service.Retrieve是IOrganizationService.Retrieve具有的返回类型Entity.

Contact是基类的派生类Entity.您不能将基类强制转换为更具体的派生类(请参阅是否可以使用C#中的显式类型转换将基类对象分配给派生类引用?).如果你试图从一个做投EntityContact,你会得到一个异常,并转换将返回一个空的对象.

包含来自CrmSvcUtil的GeneratedCode的示例,但没有与CRM的实际连接.

var entity = new Entity();

Console.WriteLine($"Type of local entity: {entity.GetType()}");

Console.WriteLine($"Local entity as Contact is null? {entity as COntact== null}");

输出:

Type of local entity: Microsoft.Xrm.Sdk.Entity
Local entity as Contact is null? True

所以给定Retrieve返回一个Entity,不能转换为Contact,你的代码行(var cOntact= service.Retrieve("contact", id, new ColumnSet()) as Contact;)甚至如何工作?

嗯,这很神奇.显然,如果在应用程序中包含CrmSvcUtil的GeneratedCode,该Retrieve函数将返回特定的派生类而不是泛型Entity.

包含CrmSvcUtil的GeneratedCode的示例包括:

CrmServiceClient service = new CrmServiceClient(ConfigurationManager.ConnectionStrings["Crm"].ConnectionString);

Contact c = new Contact()
{
    LastName = "Test"
};

Guid cOntactId= service.Create(c);

var respOnse= service.Retrieve("contact", contactId, new ColumnSet());

Console.WriteLine($"Type of response from CRM: {response.GetType()}");

Console.WriteLine($"Response from CRM as contact is null? {response as COntact== null}");

输出:

Type of response from CRM: Contact
Response from CRM as contact is null? False

没有生成代码的示例:

CrmServiceClient service = new CrmServiceClient(ConfigurationManager.ConnectionStrings["Crm"].ConnectionString);

Entity c = new Entity("contact");
c["lastname"] = "Test";

Guid cOntactId= service.Create(c);

var respOnse= service.Retrieve("contact", contactId, new ColumnSet());

Console.WriteLine($"Type of response: {response.GetType()}");

输出:

Type of response: Microsoft.Xrm.Sdk.Entity

回到你的问题.如果你在项目中包含生成的代码,Retrieve那么Contact无论如何返回一个你只需做一个简单的(Contact)service.Retrieve(...)转换(例如)或转换(as).就什么ToEntity做而言,它实际上并没有进行演员表或转换.它创建一个新对象并在其他一些东西中执行浅拷贝.因此,如果符合您的需要,请使用它,但如果没有它,您可能会离开.

解密代码:

public T ToEntity() where T : Entity
{
    if (typeof(T) == typeof(Entity))
    {
        Entity entity = new Entity();
        this.ShallowCopyTo(entity);
        return entity as T;
    }
    if (string.IsNullOrWhiteSpace(this._logicalName))
    {
        throw new NotSupportedException("LogicalName must be set before calling ToEntity()");
    }
    string text = null;
    object[] customAttributes = typeof(T).GetCustomAttributes(typeof(EntityLogicalNameAttribute), true);
    if (customAttributes != null)
    {
        object[] array = customAttributes;
        int num = 0;
        if (num 



2> Pawel Gradec..:

它总是那样工作,从这里看一下CRM 2011示例代码

ColumnSet cols = new ColumnSet(new String[] { "name", "address1_postalcode", "lastusedincampaign", "versionnumber" });
Account retrievedAccount = (Account)_serviceProxy.Retrieve("account", _accountId, cols);
Console.Write("retrieved ");

这就是你必须EnableProxyTypes();在你的IOrganizationService 上做的原因.基本上如果你这样做,所有调用都将返回早期绑定类型,而不是Entity对象(当然早期绑定是从Entity继承的,但你知道我的意思).这只是一个从CRM获取数据的功能.

这与ToEntity <>()无关,因为你仍然无法做到这样的事情:

var account = new Entity("account");
var earlyBoundAccount = account as Account; //this will result in NULL

因此,如果您有实体(例如在插件目标或PostImage中),您仍然必须使用ToEntity将其转换为早期绑定.

更新:我深入挖掘并检查了EnableProxyTypes的作用 - 它只是使用DataContractSerializerOperationBehavior类来注入它自己IDataContractSurrogate来处理响应的序列化/反序列化(例如,如何使用它可以在这里找到).通过查看反序列化的CRM来源,您可以亲眼看到如何实现反序列化:

object IDataContractSurrogate.GetDeserializedObject(object obj, Type targetType)
{
    bool supportIndividualAssemblies = this._proxyTypesAssembly != null;
    OrganizationResponse organizatiOnResponse= obj as OrganizationResponse;
    if (organizationResponse != null)
    {
        Type typeForName = KnownProxyTypesProvider.GetInstance(supportIndividualAssemblies).GetTypeForName(organizationResponse.ResponseName, this._proxyTypesAssembly);
        if (typeForName == null)
        {
            return obj;
        }
        OrganizationResponse organizationResponse2 = (OrganizationResponse)Activator.CreateInstance(typeForName);
        organizationResponse2.RespOnseName= organizationResponse.ResponseName;
        organizationResponse2.Results = organizationResponse.Results;
        return organizationResponse2;
    }
    else
    {
        Entity entity = obj as Entity;
        if (entity == null)
        {
            return obj;
        }
        Type typeForName2 = KnownProxyTypesProvider.GetInstance(supportIndividualAssemblies).GetTypeForName(entity.LogicalName, this._proxyTypesAssembly);
        if (typeForName2 == null)
        {
            return obj;
        }
        Entity entity2 = (Entity)Activator.CreateInstance(typeForName2);
        entity.ShallowCopyTo(entity2);
        return entity2;
    }
}

因此,基本上通过实体逻辑名称获取KnownProxyTypes的类型并使用实例化Activator.再次 - 这仅适用于您为其启用代理类型的IOrganizationService(并且据我记得,如果代理在同一个程序集中实例化IOrganizationService,则默认情况下启用此功能,即使您没有明确调用它,但是这个我不是百分百肯定的)


推荐阅读
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 基于PgpoolII的PostgreSQL集群安装与配置教程
    本文介绍了基于PgpoolII的PostgreSQL集群的安装与配置教程。Pgpool-II是一个位于PostgreSQL服务器和PostgreSQL数据库客户端之间的中间件,提供了连接池、复制、负载均衡、缓存、看门狗、限制链接等功能,可以用于搭建高可用的PostgreSQL集群。文章详细介绍了通过yum安装Pgpool-II的步骤,并提供了相关的官方参考地址。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • Centos7.6安装Gitlab教程及注意事项
    本文介绍了在Centos7.6系统下安装Gitlab的详细教程,并提供了一些注意事项。教程包括查看系统版本、安装必要的软件包、配置防火墙等步骤。同时,还强调了使用阿里云服务器时的特殊配置需求,以及建议至少4GB的可用RAM来运行GitLab。 ... [详细]
  • Oracle Database 10g许可授予信息及高级功能详解
    本文介绍了Oracle Database 10g许可授予信息及其中的高级功能,包括数据库优化数据包、SQL访问指导、SQL优化指导、SQL优化集和重组对象。同时提供了详细说明,指导用户在Oracle Database 10g中如何使用这些功能。 ... [详细]
  • ZSI.generate.Wsdl2PythonError: unsupported local simpleType restriction ... [详细]
  • 本文介绍了RPC框架Thrift的安装环境变量配置与第一个实例,讲解了RPC的概念以及如何解决跨语言、c++客户端、web服务端、远程调用等需求。Thrift开发方便上手快,性能和稳定性也不错,适合初学者学习和使用。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • Webmin远程命令执行漏洞复现及防护方法
    本文介绍了Webmin远程命令执行漏洞CVE-2019-15107的漏洞详情和复现方法,同时提供了防护方法。漏洞存在于Webmin的找回密码页面中,攻击者无需权限即可注入命令并执行任意系统命令。文章还提供了相关参考链接和搭建靶场的步骤。此外,还指出了参考链接中的数据包不准确的问题,并解释了漏洞触发的条件。最后,给出了防护方法以避免受到该漏洞的攻击。 ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
  • 本文介绍了如何使用C#制作Java+Mysql+Tomcat环境安装程序,实现一键式安装。通过将JDK、Mysql、Tomcat三者制作成一个安装包,解决了客户在安装软件时的复杂配置和繁琐问题,便于管理软件版本和系统集成。具体步骤包括配置JDK环境变量和安装Mysql服务,其中使用了MySQL Server 5.5社区版和my.ini文件。安装方法为通过命令行将目录转到mysql的bin目录下,执行mysqld --install MySQL5命令。 ... [详细]
  • 解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法
    本文介绍了解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法,包括检查location配置是否正确、pass_proxy是否需要加“/”等。同时,还介绍了修改nginx的error.log日志级别为debug,以便查看详细日志信息。 ... [详细]
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • 本文介绍了RxJava在Android开发中的广泛应用以及其在事件总线(Event Bus)实现中的使用方法。RxJava是一种基于观察者模式的异步java库,可以提高开发效率、降低维护成本。通过RxJava,开发者可以实现事件的异步处理和链式操作。对于已经具备RxJava基础的开发者来说,本文将详细介绍如何利用RxJava实现事件总线,并提供了使用建议。 ... [详细]
author-avatar
seaknkoo_776
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有