Oracle异常汇总

ORA-00000 to ORA-00877

ORA-00000: 正常的成功的完成(操作)

ORA-00000: normal, successful completion

原因1: 正常执行完成。【部分验证】

Normal exit.

分析: 此异常多数为程序没有执行SQL语句或者说成功执行完SQL语句,但人为或因逻辑有误,非要使用相关方法程序去获取Oracle的错误信息,得到此异常,实质是Oracle告知没有异常产生,猜测是异常信息的默认值为这个。目前发现以下两种情况:

存储过程、PL/SQL块等,使用sqlerrm获取异常,如下例所示。【已验证】

declare

v_sqlcodenumber;

v_sqlerrmvarchar2(4000);

begin

/*

      ……

      相关执行代码

      ……

  */

v_sqlcode:=sqlcode;

v_sqlerrm:=sqlerrm;

dbms_output.put_line('本次的异常code:'||v_sqlcode||chr(10)||'本次的异常信息:'||v_sqlerrm);

exception

whenothersthen

rollback;

v_sqlcode:=sqlcode;

v_sqlerrm:=sqlerrm;

dbms_output.put_line('本次的异常code:'||v_sqlcode||chr(10)||'本次的异常信息:'||v_sqlerrm);

end;

/

使用OCI的C程序中,用erhms()函数(OCIErrorGet())获得Oracle错误信息。【未验证,网络汇总】

措施: 无。【如果是人为需要获取该异常,则不用做任何操作;如果是逻辑有误,那么需要调整不去此异常或者在遇到此异常时将其屏蔽去掉。】

None

原因2: hosts文件配置错误。【未验证,网络汇总】

分析: 这种错误通常由于数据库是复制过来的,hosts文件中的ip对应的host name和当前的主机名不一致导致甚至hosts文件丢失,都会导致数据库startup时报此错。

措施: 校验hosts文件是否有错或缺失,进行修改或补充。

hosts文件在不同系统中所处的目录:

Windows XP/2000/Vista/7/8/8.1/10==> C:\windows\system32\drivers\etc\

  Linux及其他类Unix操作系统 ==> /etc/

ORA-00001: 违反唯一约束条件 (string.string=>[拥有者].[约束名])

ORA-00001: unique constraint (string.string) violated

原因1: UPDATE或INSERT语句试图插入重复的键。对于在DBMS MAC模式下配置的Trusted Oracle,如果在不同级别存在重复条目,您可能会看到此信息。【已验证】

An UPDATE or INSERT statement attempted to insert a duplicate key. For Trusted Oracle configured in DBMS MAC mode, you may see this message if a duplicate entry exists at a different level.

分析: 如下例所示,此异常一般为违反作用于表上的唯一约束或者主键约束导致,它们限制了表的一列或多列值的唯一性,不能插入重复数据。

-- 创建测试表

createtableora_00001_1(

achar(24)/*primary key*/,-- 亦可加注释内信息实现添加主键约束

bnumber/*unique*/,-- 亦可加注释内信息实现添加唯一约束

-- 增加主键约束

constraintora_00001_1_aprimarykey(a)

-- 亦可加注释内信息实现添加唯一约束

/*, constraint ora_00001_1_b unique (b)*/

);

-- 亦可加注释内信息实现添加主键约束

/*alter table ora_00001_1 add constraint ora_00001_1_a primary key (a);*/

-- 增加唯一约束

altertableora_00001_1addconstraintora_00001_1_bunique(b);

-- 插入测试数据

insertintoora_00001_1(a,b)values('1',1);

insertintoora_00001_1(a,b)values('2',2);

commit;

-- ORA-00001: 违反唯一约束条件 (C##LY.ORA_00001_1_A);

insertintoora_00001_1(a,b)values('1',3);

-- ORA-00001: 违反唯一约束条件 (C##LY.ORA_00001_1_B)

insertintoora_00001_1(a,b)values('3',2);

-- ORA-00001: 违反唯一约束条件 (C##LY.ORA_00001_1_A)

updateora_00001_1seta='1'wherea='2';

-- ORA-00001: 违反唯一约束条件 (C##LY.ORA_00001_1_B)

updateora_00001_1setb='2'whereb='1';

措施: 删除唯一约束限制或不插入重复值。

Either remove the unique restriction or do not insert the key.

如果分析确定此处唯一约束或主键约束不需要,那么则可使用下面语句删除约束

-- 查询约束与索引信息

selecta.owner约束所有者,

a.constraint_name约束名,

casea.constraint_type

when'P'then

'Primary key'

when'U'then

'Unique key'

when'C'then

' Check constraint on a table'

when'R'then

'Referential integrity'

when'V'then

'With check option, on a view'

when'O'then

'With read only, on a view'

when'H'then

'Hash expression'

when'F'then

'Constraint that involves a REF column'

when'S'then

'Supplemental logging'

else

'unkown'

end约束类型,

b.table_name表名,

b.column_name列名,

c.index_name索引名,

c.uniqueness是否唯一索引/*,

        d.table_name 表名,

        d.column_name 列名*/

fromuser_constraintsa,user_cons_columnsb,user_indexesc/*, user_ind_columns d*/

wherea.constraint_name=b.constraint_name

anda.index_name=c.index_name

/*and c.index_name = d.index_name*/

anda.owner=b.owner

anda.owner=c.table_owner

anda.owner=&"[拥有者]"

anda.constraint_name=&"[约束名]";

-- 由于如果约束对应的唯一索引若是事先手工创建的,那么在删除约束时索引不会被删除,Oracle之后自动删除自己隐式创建的索引。

-- 因此加上drop index,可确保一定将索引删除。

altertable[表名]dropconstraint[约束名]dropindex;

-- 如果是主键约束,有可能遇到有用作外键的情况,那么在删除时仍会报=>ORA-02273: 此唯一/主键已被某些外键引用

-- 报错后了解是否有问题,是否需去除此主键和外键,然后可考虑用下面语句删除主键约束,会同时删除外键约束

altertable[表名]dropconstraint[约束名]cascadedropindex;

altertable[表名]dropprimarykeycascadedropindex;

如果分析确定是值重复,那么需排查表数据与预执行的SQL语句的重复值冲突、同一个事务内执行的SQL语句之间的重复值冲突,去掉重复值的插入或更新。

备注:

唯一约束与主键约束的同:

都通过唯一索引来限制约束列的唯一性,确保任何使表中约束的列在行与行之间存在重复值的操作失败

若无事先创建好唯一索引,都会在创建唯一约束或主键约束时隐式创建同名的唯一约束

有约束必定有索引(无法在保持约束存在的情况下删除索引=>ORA-02429: 无法删除用于强制唯一/主键的索引)

有索引不一定有约束(只删除约束但不删除索引则仍然会限制索引列的值的唯一性)

唯一约束与主键约束的异:

唯一约束允许在该列或多列上存在NULL值,但主键约束不能存在NULL值

一个表只能创建一个主键约束,但可创建多个唯一约束

主键可扩展作为外键,唯一约束不可

ORA-00017: 会话被要求设置跟踪事件

ORA-00017: session requested to set trace event

原因1: 当前会话被要求通过另一个会话设置一个跟踪事件【ora12_ERRMG】

The current session was requested to set a trace event by another session.

措施: 内部使用;无需操作。

This is used internally; no action is required.

ORA-00018: 超出最大会话数

ORA-00018: maximum number of sessions exceeded

原因1: 所有会话状态对象都在使用中。【部分验证】

All session state objects are in use.

分析: 很明显,系统中所有会话数目已经达到设置的SESSIONS值,因此准备要创建的会话无法成功创建,而这个会话包括有用户建立连接至数据库是产生的会话、后台进程产生的会话以及各类涉及到硬解析处理数据字典基表的DML、DDL语句产生的递归会话。

措施: 增加SESSIONS初始化参数值。【是否需要增加SESSIONS值还需进行判断,是否是由于此值过小而现实场景需要更大的值?】

Increase the value of the SESSIONS initialization parameter.

如果判断确定是由于SESSIONS值过小导致,则需修改增大此参数值:

/*

      alter system set 参数名=值  [scope=应用范围];

      scope需知:

          scope=both,表示修改会立即生效且会修改spfile文件以确保数据库在重启后也会生效如果(以spfile启动此项为缺省值);

          scope=memory,表示修改会立即生效但不会修改spfile文件,因此重启后失效(以pfile启动此项为缺省值,且只可设置这个值);

          scope=spfile,表示只修改spfile文件,在重启数据库后才生效(对应静态参数则只可设置此项值,设置其它值会报错:

                                          ORA-02095: specified initialization parameter cannot be modified)。

  */

-- 查看"是否可用ALTER SYSTEM修改"列值,根据结果进行修改

selectname参数名,

casetype

when1then

'Boolean'

when2then

'String'

when3then

'Integer'

when4then

'Parameter file'

when5then

'Reserved'

when6then

'Big integer'

else

'unknown'

end参数类型,

value"会话级(若可修改)或实例级参数值",

display_value展示值,

isses_modifiable"是否可用ALTER SESSION修改",

caseissys_modifiable

when'IMMEDIATE'then

'无论pfile还是spfile启动,都可用"alter system set '||name||'=&'||

'新的参数值;"更改参数并立即生效。'

when'DEFERRED'then

'无论pfile还是spfile启动,都可用"alter system set '||name||'=&'||

'新的参数值;"更改参数并将在之后的会话中生效。'

when'FALSE'then

casea.is_spfile

when0then

'使用pfile启动,需手动修改pfile文件中对应参数值再重启。'

else

'使用spfile启动,可用"alter system set '||name||'=&'||name||

' scope=spfile;"更改参数。更改将在后续的实例中生效(当前数据库需重启)。'

end

else

'?'

end"是否可用ALTER SYSTEM修改",

isinstance_modifiable"是否不同实例间值可不同"

fromv$parameter,

(selectcount(1)is_spfilefromv$parametertwheret.name='spfile')a

wherename='sessions';

如果判断确定SESSIONS值合理,则需分析确定产生大量会话的原因,是否相关程序代码建立了连接未释放?或者其它原因等。【待完善】

备注: 参数SESSIONS:

属性描述

参数类型Integer

默认值派生公式: (1.1 * PROCESSES) + 5)[11gR1,11gR2](1.5 * PROCESSES) + 22

可修改(不用重启及时生效)否[11gR2,12cR1]可用ALTER SYSTEM修改

取值范围1~2^31[11gR1,11gR2]1~2^16(1~65536)

基础参数是

SESSIONS指定可以在系统中创建的最大会话数。因为每次登录都需要一个会话,所以这个参数有效地确定了系统中最大并发用户数。您应该始终将此参数显式设置等于最大并发用户数的估计值+后台进程数+递归会话数(大约占总数的10%)。

Oracle使用此参数的默认值作为其最小值。 将SESSIONS值设置成[1~默认值)不会触发错误,因为Oracle会忽略此值直接使用默认值。

ENQUEUE_RESOURCES和TRANSACTIONS参数的默认值派生自SESSIONS。因此,如果增加SESSIONS的值,则应考虑是否也调整ENQUEUE_RESOURCES和TRANSACTIONS的值。 (请注意,从Oracle Database 10g release 2(10.2)起,ENQUEUE_RESOURCES已被废弃。)

在共享服务器环境中,PROCESSES的值可能相当小。因此,Oracle建议您将SESSIONS的值调整为大约1.1 *总连接数。

ORA-00019: 超出最大许可会话数

ORA-00019: maximum number of session licenses exceeded

原因1: 所有许可会话都在使用中。【部分验证】

All licenses are in use.

分析: 很明显,系统中并发用户会话已经达到设置的LICENSE_MAX_SESSIONS值,因此准备要创建的用户会话无法创建。

措施: 增大LICENSE_MAX_SESSIONS初始化参数的值。【是否需要增加LICENSE_MAX_SESSIONS值还需进行判断,是否是由于此值过小而现实场景需要更大的值?】

Increase the value of the LICENSE MAX SESSIONS initialization parameter.

如果判断确定是由于LICENSE_MAX_SESSIONS值过小导致,则需修改增大此参数值,【详情参见ORA-00018=>原因1=>措施,将SQL语句中name = 'sessions'修改为name = 'license_max_sessions'即可】

如果判断确定LICENSE_MAX_SESSIONS值合理,则需分析确定产生大量会话的原因,是否相关程序代码建立了连接未释放?或者其它原因等。【待完善】

备注: 参数LICENSE_MAX_SESSIONS:

属性描述

参数类型Integer

默认值0

可修改(不用重启及时生效)可用ALTER SYSTEM修改

取值范围0~许可会话数

基础参数否

Oracle实时应用集群多个实例可以具有不同的值,但是安装数据库的所有实例的总和应小于或等于该数据库许可的会话总数。

LICENSE_MAX_SESSIONS指定允许的并发用户会话的最大数量。达到此限制后,只有具有RESTRICTED SESSION权限的用户才能连接到数据库。无法连接的用户收到表示系统达到最大容量的警告消息。

零值表示不强制执行并发使用(会话)许可。如果将此参数设置为非零数字,则可能还需要设置LICENSE_SESSIONS_WARNING(请参阅“LICENSE_SESSIONS_WARNING”)。

不要同时启用并发使用许可和用户许可,即LICENSE_MAX_SESSIONS与LICENSE_MAX_USERS两参数值至少一个要设置为零。

ORA-00020: 超出最大进程数(string=>[最大进程数])

ORA-00020: maximum number of processes (string) exceeded

原因1: 所有进程状态对象都在使用中。【部分验证】

All process state objects are in use.

分析: 很明显,系统中进程数已经达到设置的PROCESSES值,因此准备要创建的用户会话无法创建。

措施: 增加PROCESSES初始化参数的值。【是否需要增加PROCESSES值还需进行判断,是否是由于此值过小而现实场景需要更大的值?】

Increase the value of the PROCESSES initialization parameter.

如果判断确定是由于PROCESSES值过小导致,则需修改增大此参数值,【详情参见ORA-00018=>原因1=>措施,将SQL语句中name = 'sessions'修改为name = 'processes'即可】

如果判断确定PROCESSES值合理,则需分析确定产生大量进程的原因,是否相关程序代码建立了连接未释放?或者其它原因等。【待完善】

备注: 参数PROCESSES:

属性描述

参数类型Integer

默认值40~操作系统依赖数[10gR2,11gR1]100[11gR2,12cR1]该值是派生的,它通常取决于警报日志中报告的核心数。

可修改(不用重启及时生效)否

取值范围6~操作系统依赖数

基础参数是

Oracle实时应用集群多个实例可以具有不同的值。

PROCESSES指定可以同时连接到Oracle的最大操作系统用户进程数。它的值应允许所有后台进程运行,如锁,作业队列进程和并行执行进程。

该参数派生了SESSIONS和TRANSACTIONS参数的默认值。因此,如果更改PROCESSES的值,则应评估是否要调整这些派生参数的值。

ORA-01500 to ORA-02098

ORA-01722: 无效数字

ORA-01722: invalid number

原因1: 指定的数字无效

The specified number was invalid.

分析: 究其根本在于,要执行的语句在人为显式转换或Oracle判断决定隐式转换,故意或非故意得使一个非数值类型且无法转换为数值类型的值转换为数值类型失败导致的。如下几例:

显式转换:

-- to_number、to_binary_float、to_binary_double转换字符串为数值

selectto_number('2017年')fromdual;

selectto_binary_float('8.935M')fromdual;

selectto_binary_double('0.001s')fromdual;

隐式转换:

---------涉及到数值类型列的赋值或函数需要函数数值类型结果的转换---------

-- insert/update/merge等语句操作列值

-- 测试表

createtableora_01722_1(

anumber,

bchar(24),

cvarchar2(300)

);

-- 插入赋值,无法隐式转换成数值,报错

insertintoora_01722_1(a)values('111测试字符串');

-- 插入赋值,可以隐式转换成数值,不报错

insertintoora_01722_1(a,b)values('111','bbb');

commit;

-- 更新赋值,无法隐式转换成数值,报错

updateora_01722_1seta='222测试字符串';

-- decode由第三列的数值1确定了改函数输出结果为数值类型

-- 当匹配到'a'返回1,结果为数值,不报错

selectdecode('a','a',1,'b',2,'c','three',0)test1fromdual;

-- 当匹配到'c'返回'three',结果不为数值且无法转换为数值,报错

selectdecode('c','a',1,'b',2,'c','three',0)test2fromdual;

-- nvl由第一列输入的数值确定了它在此时输出的结果也会为数值

-- 当发现第一个值不为空时,就尝试输出后边的值作为本次nvl函数操作后的数值结果,发现无法转换为数值,报错

selectnvl(&请输入数值,'测试')fromdual;

-- nvl2由第二列的2确定了它在此时输出的结果也会为数值

-- 当不管结果是否为2,它都会讲后边的值进行测试转换……所以无论是否输入null都报错

selectnvl2(&无论是否null,2,'测试')fromdual;

---------涉及到与数值类型的比较或其它运算---------

-- 未比较到'三',不报错

selectdecode(1,1,'one',2,'two','三','three','zero')fromdual;

-- 比较到'三',报错

selectdecode(3,1,'one',2,'two','三','three','zero')fromdual;

-- 与数值进行算术运算,无法隐式转换为数值,报错

select'a'+2fromdual;

-- 与ora_01722_1的数值类型列a进行比较,将右边转换为数值失败,报错

select*fromora_01722_1twheret.a='a';

-- ora_01722_1列b与数值类型进行比较,将左边转换为数值失败,报错

select*fromora_01722_1twheret.b=2;

隐式转换:特别注意时灵时不灵的查询,都是重新类似于上面的情况,但由于之前查询的是部分表数据或部分视图数据等,但是报错是涉及到另一部分甚至是全表全视图的数据,而由于未注意隐式转换导致“脏数据”无法转换成数值类型而报错。其实不是它脏,是你不注意字段类型的区分,请不要依赖于Oracle的隐式转换!

措施: 指定一个有效的数字。

Specify a valid number.

根据分析的情况,明确是否需要主动转换,是否数据存在问题,该是数值就传数值,该是字符串就添加左右英文单引号包裹成字符串。

备注: 当需要的时候,Oracle数据库会自动将值从一种数据类型转换为另一种(如CHAR、VARCHAR2、NCHAR、NVARCHAR2、BINARY_FLOAT、BINARY_DOUBLE就有可能隐式转换为NUMBER)。由于以下原因,Oracle建议您指定显式转换,而不是依赖于隐式自动转换:

当您使用显式数据类型转换函数时,SQL语句更容易理解。

隐式数据类型转换可能会对性能产生负面影响,特别是如果列值的数据类型被隐式转换为基本常量的数据类型,而不是主动转为其它类型。

隐式转换要根据需要转换时的上下文来决定,而且在每类场景中不一定产生一样的效果。例如,从datetime值到varchar2值的隐式转换可能会根据NLS_DATE_FORMAT参数的值返回意外的一年。

隐式转换的算法可能会随着软件版本和Oracle产品之间的变化而变化。显式转换的变化则更有预见性。

如果在索引表达式中产生了隐式数据类型转换,那么Oracle数据库可能不会使用该索引,因为它是为转换前数据类型定义的。这可能会对性能产生负面影响。

ORA-01747: user.table.column, table.column 或列说明无效【已验证】

ORA-01747: invalid user.table.column, table.column, or column specification

原因1: 列名为关键字。

分析: 一般为在SQL语句或存储过程、函数等中使用到的此字段为oracle的保留关键字,且保留方式标识了此关键字在某些情况下,例如在DML中是否不允许作为标识符的。如下列情况:

-- 查询能做属性但不能作为标识符或某些场景(如DML操作)下不能作为标识符的关键字

selectt.*

fromv$reserved_wordst

where(t.res_semi='Y'ort.reserved='Y')

andt.res_attr='N';

-- 根据上面关键字建表,为测试需要,实际使用时请避免将Oracle保留关键字作为表的字段!

createtableora_01747_1(

"TRIGGER"number,"WHERE"number,"REVOKE"number,"INCREMENT"number,"THEN"number,

"FILE"number,"PRIOR"number,"CONNECT"number,"COMMENT"number,"SYSDATE"number,

"ONLINE"number,"DECIMAL"number,"SESSION"number,"MODIFY"number,"IN"number,

"@"number,","number,"GRANT"number,"INTO"number,"VALIDATE"number,"."number,

"ADD"number,"ORDER"number,"HAVING"number,"TO"number,"NULL"number,"RENAME"number,

"LEVEL"number,"USER"number,"ANY"number,/*"ROWID" number, --不可作建表属性*/

"SHARE"number,"MODE"number,"UNION"number,"/"number,"SET"number,"INDEX"number,

"MAXEXTENTS"number,"VALUES"number,"|"number,"VIEW"number,"["number,"WITH"number,

"EXCLUSIVE"number,"ALTER"number,"FROM"number,"SELECT"number,"BY"number,"-"number,

"MLSLABEL"number,"AND"number,"+"number,"ROWS"number,"CHECK"number,":"number,

"VARCHAR2"number,"IMMEDIATE"number,"CURRENT"number,"AS"number,"*"number,"TABLE"number,

"LONG"number,"SYNONYM"number,"ASC"number,"UNIQUE"number,"LIKE"number,"DESC"number,

"VARCHAR"number,"INITIAL"number,"CHAR"number,"="number,"DROP"number

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容