0

0

如何使用C#处理Oracle的复合数据类型_Udt与Object Type在ODP.NET中的映射

P粉602998670

P粉602998670

发布时间:2026-03-11 13:34:03

|

529人浏览过

|

来源于php中文网

原创

oracle udt 映射失败主因是c#类未标记[oraclecustomtypemapping]、未实现ioraclecustomtype/ioraclecustomtypefactory,或udttypename大小写/全限定名不匹配;嵌套udt需逐层映射,避免混用oracleobject与自定义类,且须确保schema权限正确。

oracle udt 映射失败:oracleudt 类没实现或类型名不匹配

odp.net 要把 oracle 的 object type(比如 person_type)映射成 c# 类,必须让该类显式标记为 [oraclecustomtypemapping],且实现 ioraclecustomtypeioraclecustomtypefactory。只加个 public class person 是没用的,odp.net 完全不认识。

常见错误现象:System.InvalidOperationException: Unable to cast object of type 'Oracle.ManagedDataAccess.Types.OracleObject' to type 'YourNamespace.Person'

  • [OracleCustomTypeMapping("SCHEMA.PERSON_TYPE")] 中的 schema 名必须全大写、与数据库中一致(哪怕你建的时候用了双引号小写,这里也得按实际存储名来)
  • 工厂类(实现 IOracleCustomTypeFactory)的 CreateObject 方法返回的实例,必须是你要映射的 C# 类型,不能是基类或接口
  • 如果 UDT 在 public 同义词下,schema 名仍要写真实 owner,比如 "HR.PERSON_TYPE",而不是 "PUBLIC.PERSON_TYPE"

OracleCommand 获取 UDT 返回值时,OracleDbType.Object 必须配对指定 UdtTypeName

调用带 OUT 参数的存储过程,或查询返回 OBJECT 列时,不能只设 OracleDbType.Object,否则 ODP.NET 不知道该转成哪个 .NET 类——它会退化成 OracleObject,你拿不到强类型对象。

使用场景:执行 SELECT person_col FROM people WHERE id = 1,其中 person_colPERSON_TYPE

  • 参数/列必须显式设置 UdtTypeName = "SCHEMA.PERSON_TYPE",例如:param.UdtTypeName = "HR.PERSON_TYPE"
  • 如果是查询结果集,DataReader 读取时也要先确认列的 OracleDbTypeObject,再用 GetOracleObject(i),然后手动调用 ToCustomObject<person>()</person>(需引用 Oracle.ManagedDataAccess.Types
  • 注意:UdtTypeName 区分大小写,且不能带引号;若类型在非默认 schema,必须写全限定名

嵌套 UDT 或集合(VARRAY/TABLE OF)导致反序列化崩溃

Oracle 的 OBJECT TYPE 字段里如果包含另一个 OBJECT TYPE,或定义了 MEMBER FUNCTION,ODP.NET 默认不会递归解析——它卡在第一层,抛出 NotSupportedException 或空引用。

蛙蛙写作——超级AI智能写作助手
蛙蛙写作——超级AI智能写作助手

蛙蛙写作辅助AI写文,帮助获取创意灵感,提供拆书、小说转剧本、视频生成等功能,是一款功能全面的AI智能写作工具。

下载

性能影响:每层嵌套都会触发一次额外的元数据查询,深度 >3 层时延迟明显上升

  • 嵌套对象字段必须也声明为对应 C# 类型,并同样实现 IOracleCustomType 映射(不能用 objectOracleObject
  • VARRAYTABLE OF 需额外注册集合映射:用 [OracleCustomTypeMapping("SCHEMA.PERSON_LIST")] 标记一个实现 IOracleCollection 的类,内部用 List<person></person> 存储
  • 避免在 UDT 中定义 MEMBER FUNCTIONMAP MEMBER FUNCTION,ODP.NET 不调用它们,但可能干扰类型发现逻辑

使用 Oracle.ManagedDataAccess 时,OracleObject 与自定义类混用引发内存泄漏

开发中常误以为“先用 GetOracleObject() 拿到原生对象,再手工赋值给 C# 类”更安全,结果每次调用都创建未释放的 OracleObject 实例,长期运行后连接池耗尽、GC 压力陡增。

兼容性影响:.NET Core 3.1+ 上,OracleObject 的 finalizer 行为不稳定,某些版本会跳过清理

  • 全程坚持用强类型映射路径:参数设 UdtTypeName → 绑定自定义类实例 → 执行 → 直接接收返回的 C# 对象
  • 绝不调用 GetOracleObject() 后又丢弃它;如真需底层访问,用完立刻调用 Dispose()
  • 检查项目是否同时引用了 Oracle.DataAccess(旧版)和 Oracle.ManagedDataAccess,二者混用会导致类型系统冲突,映射直接失效

最麻烦的其实是 schema 权限和同义词链——UDT 映射失败十次里有七次不是代码问题,而是当前连接用户没被授予 EXECUTE 权限,或者 CREATE TYPE 时用了双引号导致名字大小写敏感,而 C# 里写的 UdtTypeName 没对上。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据类型有哪几种
数据类型有哪几种

数据类型有整型、浮点型、字符型、字符串型、布尔型、数组、结构体和枚举等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

336

2023.10.31

php数据类型
php数据类型

本专题整合了php数据类型相关内容,阅读专题下面的文章了解更多详细内容。

224

2025.10.31

c语言 数据类型
c语言 数据类型

本专题整合了c语言数据类型相关内容,阅读专题下面的文章了解更多详细内容。

138

2026.02.12

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1902

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

656

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2387

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

47

2026.01.19

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

850

2024.01.03

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号