forked from kangjianwei/LearningJDK
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathObjectOutputStream.java
2888 lines (2522 loc) · 117 KB
/
ObjectOutputStream.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/*
* Copyright (c) 1996, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package java.io;
import java.io.ObjectStreamClass.WeakClassKey;
import java.lang.ref.ReferenceQueue;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.StringJoiner;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import sun.reflect.misc.ReflectUtil;
import sun.security.action.GetBooleanAction;
/**
* An ObjectOutputStream writes primitive data types and graphs of Java objects
* to an OutputStream. The objects can be read (reconstituted) using an
* ObjectInputStream. Persistent storage of objects can be accomplished by
* using a file for the stream. If the stream is a network socket stream, the
* objects can be reconstituted on another host or in another process.
*
* <p>Only objects that support the java.io.Serializable interface can be
* written to streams. The class of each serializable object is encoded
* including the class name and signature of the class, the values of the
* object's fields and arrays, and the closure of any other objects referenced
* from the initial objects.
*
* <p>The method writeObject is used to write an object to the stream. Any
* object, including Strings and arrays, is written with writeObject. Multiple
* objects or primitives can be written to the stream. The objects must be
* read back from the corresponding ObjectInputstream with the same types and
* in the same order as they were written.
*
* <p>Primitive data types can also be written to the stream using the
* appropriate methods from DataOutput. Strings can also be written using the
* writeUTF method.
*
* <p>The default serialization mechanism for an object writes the class of the
* object, the class signature, and the values of all non-transient and
* non-static fields. References to other objects (except in transient or
* static fields) cause those objects to be written also. Multiple references
* to a single object are encoded using a reference sharing mechanism so that
* graphs of objects can be restored to the same shape as when the original was
* written.
*
* <p>For example to write an object that can be read by the example in
* ObjectInputStream:
* <br>
* <pre>
* FileOutputStream fos = new FileOutputStream("t.tmp");
* ObjectOutputStream oos = new ObjectOutputStream(fos);
*
* oos.writeInt(12345);
* oos.writeObject("Today");
* oos.writeObject(new Date());
*
* oos.close();
* </pre>
*
* <p>Classes that require special handling during the serialization and
* deserialization process must implement special methods with these exact
* signatures:
* <br>
* <pre>
* private void readObject(java.io.ObjectInputStream stream)
* throws IOException, ClassNotFoundException;
* private void writeObject(java.io.ObjectOutputStream stream)
* throws IOException
* private void readObjectNoData()
* throws ObjectStreamException;
* </pre>
*
* <p>The writeObject method is responsible for writing the state of the object
* for its particular class so that the corresponding readObject method can
* restore it. The method does not need to concern itself with the state
* belonging to the object's superclasses or subclasses. State is saved by
* writing the individual fields to the ObjectOutputStream using the
* writeObject method or by using the methods for primitive data types
* supported by DataOutput.
*
* <p>Serialization does not write out the fields of any object that does not
* implement the java.io.Serializable interface. Subclasses of Objects that
* are not serializable can be serializable. In this case the non-serializable
* class must have a no-arg constructor to allow its fields to be initialized.
* In this case it is the responsibility of the subclass to save and restore
* the state of the non-serializable class. It is frequently the case that the
* fields of that class are accessible (public, package, or protected) or that
* there are get and set methods that can be used to restore the state.
*
* <p>Serialization of an object can be prevented by implementing writeObject
* and readObject methods that throw the NotSerializableException. The
* exception will be caught by the ObjectOutputStream and abort the
* serialization process.
*
* <p>Implementing the Externalizable interface allows the object to assume
* complete control over the contents and format of the object's serialized
* form. The methods of the Externalizable interface, writeExternal and
* readExternal, are called to save and restore the objects state. When
* implemented by a class they can write and read their own state using all of
* the methods of ObjectOutput and ObjectInput. It is the responsibility of
* the objects to handle any versioning that occurs.
*
* <p>Enum constants are serialized differently than ordinary serializable or
* externalizable objects. The serialized form of an enum constant consists
* solely of its name; field values of the constant are not transmitted. To
* serialize an enum constant, ObjectOutputStream writes the string returned by
* the constant's name method. Like other serializable or externalizable
* objects, enum constants can function as the targets of back references
* appearing subsequently in the serialization stream. The process by which
* enum constants are serialized cannot be customized; any class-specific
* writeObject and writeReplace methods defined by enum types are ignored
* during serialization. Similarly, any serialPersistentFields or
* serialVersionUID field declarations are also ignored--all enum types have a
* fixed serialVersionUID of 0L.
*
* <p>Primitive data, excluding serializable fields and externalizable data, is
* written to the ObjectOutputStream in block-data records. A block data record
* is composed of a header and data. The block data header consists of a marker
* and the number of bytes to follow the header. Consecutive primitive data
* writes are merged into one block-data record. The blocking factor used for
* a block-data record will be 1024 bytes. Each block-data record will be
* filled up to 1024 bytes, or be written whenever there is a termination of
* block-data mode. Calls to the ObjectOutputStream methods writeObject,
* defaultWriteObject and writeFields initially terminate any existing
* block-data record.
*
* @author Mike Warres
* @author Roger Riggs
* @see java.io.DataOutput
* @see java.io.ObjectInputStream
* @see java.io.Serializable
* @see java.io.Externalizable
* @see <a href="{@docRoot}/../specs/serialization/output.html">Object Serialization Specification, Section 2, Object Output Classes</a>
* @since 1.1
*/
/*
* 对象输出流,参与序列化过程
*
* 对于实现了Externalizable接口的类,需要自行实现序列化逻辑
*/
public class ObjectOutputStream extends OutputStream implements ObjectOutput, ObjectStreamConstants {
/** stream protocol version */
private int protocol = PROTOCOL_VERSION_2; // 序列化版本
/** recursion depth */
private int depth; // 序列化嵌套深度
/** filter stream for handling block data conversion */
private final BlockDataOutputStream bout; // 块数据输出流
/** obj -> wire handle map */
private final HandleTable handles; // 共享对象哈希表
/** obj -> replacement obj map */
private final ReplaceTable subs; // 替换对象哈希表
/** buffer for writing primitive field values */
private byte[] primVals; // 待序列化的基本类型字段的值
/** if true, invoke writeObjectOverride() instead of writeObject() */
private final boolean enableOverride; // 是否使用子类实现的writeObjectOverride()方法,而不是使用默认的序列化逻辑
/** if true, invoke replaceObject() */
private boolean enableReplace; // 是否需要调用replaceObject()方法,默认为false
/* values below valid only during upcalls to writeObject()/writeExternal() */
/**
* Context during upcalls to class-defined writeObject methods; holds
* object currently being serialized and descriptor for current class.
* Null when not during writeObject upcall.
*/
private SerialCallbackContext curContext; // 序列化上下文
/** current PutField object */
private PutFieldImpl curPut; // 当前待序列化的字段集
/** custom storage for debug trace info */
private final DebugTraceInfoStack debugInfoStack;
/**
* value of "sun.io.serialization.extendedDebugInfo" property,
* as true or false for extended information about exception's place
*/
private static final boolean extendedDebugInfo = AccessController.doPrivileged(
new GetBooleanAction("sun.io.serialization.extendedDebugInfo")
);
/*▼ 构造器 ████████████████████████████████████████████████████████████████████████████████┓ */
/**
* Creates an ObjectOutputStream that writes to the specified OutputStream.
* This constructor writes the serialization stream header to the
* underlying stream; callers may wish to flush the stream immediately to
* ensure that constructors for receiving ObjectInputStreams will not block
* when reading the header.
*
* <p>If a security manager is installed, this constructor will check for
* the "enableSubclassImplementation" SerializablePermission when invoked
* directly or indirectly by the constructor of a subclass which overrides
* the ObjectOutputStream.putFields or ObjectOutputStream.writeUnshared
* methods.
*
* @param out output stream to write to
* @throws IOException if an I/O error occurs while writing stream header
* @throws SecurityException if untrusted subclass illegally overrides
* security-sensitive methods
* @throws NullPointerException if <code>out</code> is <code>null</code>
* @since 1.4
* @see ObjectOutputStream#ObjectOutputStream()
* @see ObjectOutputStream#putFields()
* @see ObjectInputStream#ObjectInputStream(InputStream)
*/
public ObjectOutputStream(OutputStream out) throws IOException {
verifySubclass();
bout = new BlockDataOutputStream(out);
handles = new HandleTable(10, (float) 3.00);
subs = new ReplaceTable(10, (float) 3.00);
enableOverride = false;
// 写入序列化头(包含一个魔数和一个版本号)
writeStreamHeader();
// 设置待写数据处于块模式下
bout.setBlockDataMode(true);
if (extendedDebugInfo) {
debugInfoStack = new DebugTraceInfoStack();
} else {
debugInfoStack = null;
}
}
/**
* Provide a way for subclasses that are completely reimplementing
* ObjectOutputStream to not have to allocate private data just used by
* this implementation of ObjectOutputStream.
*
* <p>If there is a security manager installed, this method first calls the
* security manager's <code>checkPermission</code> method with a
* <code>SerializablePermission("enableSubclassImplementation")</code>
* permission to ensure it's ok to enable subclassing.
*
* @throws SecurityException if a security manager exists and its
* <code>checkPermission</code> method denies enabling
* subclassing.
* @throws IOException if an I/O error occurs while creating this stream
* @see SecurityManager#checkPermission
* @see java.io.SerializablePermission
*/
protected ObjectOutputStream() throws IOException, SecurityException {
SecurityManager sm = System.getSecurityManager();
if (sm != null) {
sm.checkPermission(SUBCLASS_IMPLEMENTATION_PERMISSION);
}
bout = null;
handles = null;
subs = null;
enableOverride = true;
debugInfoStack = null;
}
/*▲ 构造器 ████████████████████████████████████████████████████████████████████████████████┛ */
/*▼ 写 ████████████████████████████████████████████████████████████████████████████████┓ */
/**
* Writes a boolean.
*
* @param val the boolean to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入boolean
public void writeBoolean(boolean val) throws IOException {
bout.writeBoolean(val);
}
/**
* Writes a 16 bit char.
*
* @param val the char value to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入char
public void writeChar(int val) throws IOException {
bout.writeChar(val);
}
/**
* Writes an 8 bit byte.
*
* @param val the byte value to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入byte
public void writeByte(int val) throws IOException {
bout.writeByte(val);
}
/**
* Writes a 16 bit short.
*
* @param val the short value to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入short
public void writeShort(int val) throws IOException {
bout.writeShort(val);
}
/**
* Writes a 32 bit int.
*
* @param val the integer value to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入int
public void writeInt(int val) throws IOException {
bout.writeInt(val);
}
/**
* Writes a 64 bit long.
*
* @param val the long value to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入long
public void writeLong(long val) throws IOException {
bout.writeLong(val);
}
/**
* Writes a 32 bit float.
*
* @param val the float value to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入float
public void writeFloat(float val) throws IOException {
bout.writeFloat(val);
}
/**
* Writes a 64 bit double.
*
* @param val the double value to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入double
public void writeDouble(double val) throws IOException {
bout.writeDouble(val);
}
/**
* Writes a String as a sequence of bytes.
*
* @param str the String of bytes to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入字符串中的byte信息(只取char的低8位)
public void writeBytes(String str) throws IOException {
bout.writeBytes(str);
}
/**
* Writes a String as a sequence of chars.
*
* @param str the String of chars to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入字符串中的char信息
public void writeChars(String str) throws IOException {
bout.writeChars(str);
}
/**
* Writes a byte. This method will block until the byte is actually written.
*
* @param val the byte to be written to the stream
*
* @throws IOException If an I/O error has occurred.
*/
// 向块数据输出流写入byte信息
public void write(int val) throws IOException {
bout.write(val);
}
/**
* Writes an array of bytes. This method will block until the bytes are
* actually written.
*
* @param buf the data to be written
*
* @throws IOException If an I/O error has occurred.
*/
// 向块数据输出流写入字节数组b中的数据
public void write(byte[] buf) throws IOException {
bout.write(buf, 0, buf.length, false);
}
/**
* Writes a sub array of bytes.
*
* @param buf the data to be written
* @param off the start offset in the data
* @param len the number of bytes that are written
*
* @throws IOException If an I/O error has occurred.
*/
// 向块数据输出流写入字节数组b中指定范围的的数据
public void write(byte[] buf, int off, int len) throws IOException {
if(buf == null) {
throw new NullPointerException();
}
int endoff = off + len;
if(off<0 || len<0 || endoff>buf.length || endoff<0) {
throw new IndexOutOfBoundsException();
}
bout.write(buf, off, len, false);
}
/**
* Primitive data write of this String in
* <a href="DataInput.html#modified-utf-8">modified UTF-8</a>
* format. Note that there is a
* significant difference between writing a String into the stream as
* primitive data or as an Object. A String instance written by writeObject
* is written into the stream as a String initially. Future writeObject()
* calls write references to the string into the stream.
*
* @param str the String to be written
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
*/
// 向最终输出流写入字符串的utf8形式
public void writeUTF(String str) throws IOException {
bout.writeUTF(str);
}
/*▲ 写 ████████████████████████████████████████████████████████████████████████████████┛ */
/*▼ 序列化 ████████████████████████████████████████████████████████████████████████████████┓ */
/**
* Write the specified object to the ObjectOutputStream. The class of the
* object, the signature of the class, and the values of the non-transient
* and non-static fields of the class and all of its supertypes are
* written. Default serialization for a class can be overridden using the
* writeObject and the readObject methods. Objects referenced by this
* object are written transitively so that a complete equivalent graph of
* objects can be reconstructed by an ObjectInputStream.
*
* <p>Exceptions are thrown for problems with the OutputStream and for
* classes that should not be serialized. All exceptions are fatal to the
* OutputStream, which is left in an indeterminate state, and it is up to
* the caller to ignore or recover the stream state.
*
* @throws InvalidClassException Something is wrong with a class used by
* serialization.
* @throws NotSerializableException Some object to be serialized does not
* implement the java.io.Serializable interface.
* @throws IOException Any exception thrown by the underlying
* OutputStream.
*/
// 将指定的共享对象obj写入输出流
public final void writeObject(Object obj) throws IOException {
if(enableOverride) {
// 使用子类实现的序列化逻辑
writeObjectOverride(obj);
return;
}
try {
// 使用默认的序列化逻辑:对共享对象obj进行序列化
writeObject0(obj, false);
} catch(IOException ex) {
if(depth == 0) {
writeFatalException(ex);
}
throw ex;
}
}
/**
* Writes an "unshared" object to the ObjectOutputStream. This method is
* identical to writeObject, except that it always writes the given object
* as a new, unique object in the stream (as opposed to a back-reference
* pointing to a previously serialized instance). Specifically:
* <ul>
* <li>An object written via writeUnshared is always serialized in the
* same manner as a newly appearing object (an object that has not
* been written to the stream yet), regardless of whether or not the
* object has been written previously.
*
* <li>If writeObject is used to write an object that has been previously
* written with writeUnshared, the previous writeUnshared operation
* is treated as if it were a write of a separate object. In other
* words, ObjectOutputStream will never generate back-references to
* object data written by calls to writeUnshared.
* </ul>
* While writing an object via writeUnshared does not in itself guarantee a
* unique reference to the object when it is deserialized, it allows a
* single object to be defined multiple times in a stream, so that multiple
* calls to readUnshared by the receiver will not conflict. Note that the
* rules described above only apply to the base-level object written with
* writeUnshared, and not to any transitively referenced sub-objects in the
* object graph to be serialized.
*
* <p>ObjectOutputStream subclasses which override this method can only be
* constructed in security contexts possessing the
* "enableSubclassImplementation" SerializablePermission; any attempt to
* instantiate such a subclass without this permission will cause a
* SecurityException to be thrown.
*
* @param obj object to write to stream
*
* @throws NotSerializableException if an object in the graph to be
* serialized does not implement the Serializable interface
* @throws InvalidClassException if a problem exists with the class of an
* object to be serialized
* @throws IOException if an I/O error occurs during serialization
* @since 1.4
*/
// 将指定的非共享对象obj写入输出流
public void writeUnshared(Object obj) throws IOException {
try {
writeObject0(obj, true);
} catch(IOException ex) {
if(depth == 0) {
writeFatalException(ex);
}
throw ex;
}
}
/**
* Write the non-static and non-transient fields of the current class to this stream.
* This may only be called from the writeObject method of the class being serialized.
* It will throw the NotActiveException if it is called otherwise.
*
* @throws IOException if I/O errors occur while writing to the underlying <code>OutputStream</code>
*/
/*
* 默认序列化
*
* 将当前类的非static和非transient字段序列化到输出流
* 只能从待序列化的类的writeObject()方法中调用此方法,否则会抛异常。
*/
public void defaultWriteObject() throws IOException {
SerialCallbackContext ctx = curContext;
if(ctx == null) {
throw new NotActiveException("not in call to writeObject");
}
Object curObj = ctx.getObj();
// 获取待序列化对象的序列化描述符
ObjectStreamClass curDesc = ctx.getDesc();
bout.setBlockDataMode(false);
// 对指定的对象curObj执行默认的序列化过程
defaultWriteFields(curObj, curDesc);
bout.setBlockDataMode(true);
}
/**
* Method used by subclasses to override the default writeObject method.
* This method is called by trusted subclasses of ObjectInputStream that
* constructed ObjectInputStream using the protected no-arg constructor.
* The subclass is expected to provide an override method with the modifier
* "final".
*
* @param obj object to be written to the underlying stream
*
* @throws IOException if there are I/O errors while writing to the
* underlying stream
* @see #ObjectOutputStream()
* @see #writeObject(Object)
* @since 1.2
*/
// [子类覆盖]如果开启了enableOverride,则会调用该方法来替代默认的序列化逻辑
protected void writeObjectOverride(Object obj) throws IOException {
}
/**
* Subclasses may implement this method to allow class data to be stored in the stream.
* By default this method does nothing.
* The corresponding method in ObjectInputStream is resolveClass.
* This method is called exactly once for each unique class in the stream.
* The class name and signature will have already been written to the stream.
* This method may make free use of the ObjectOutputStream to save any representation of
* the class it deems suitable (for example, the bytes of the class file).
* The resolveClass method in the corresponding subclass of
* ObjectInputStream must read and use any data or objects written by annotateClass.
*
* @param cl the class to annotate custom data for
*
* @throws IOException Any exception thrown by the underlying OutputStream.
* @see ObjectInputStream#resolveClass(ObjectStreamClass)
*/
// [子类覆盖]由子类重写的回调方法,在写入非代理类序列化描述符之后被调用
protected void annotateClass(Class<?> cl) throws IOException {
}
/**
* Subclasses may implement this method to store custom data in the stream
* along with descriptors for dynamic proxy classes.
*
* <p>This method is called exactly once for each unique proxy class
* descriptor in the stream. The default implementation of this method in
* <code>ObjectOutputStream</code> does nothing.
*
* <p>The corresponding method in <code>ObjectInputStream</code> is <code>resolveProxyClass</code>.
* For a given subclass of <code>ObjectOutputStream</code> that overrides this method,
* the <code>resolveProxyClass</code> method in the corresponding subclass of
* <code>ObjectInputStream</code> must read any data or objects written by <code>annotateProxyClass</code>.
*
* @param cl the proxy class to annotate custom data for
*
* @throws IOException any exception thrown by the underlying <code>OutputStream</code>
* @see ObjectInputStream#resolveProxyClass(String[])
* @since 1.3
*/
// [子类覆盖]由子类重写的回调方法,在写入代理类序列化描述符之后被调用
protected void annotateProxyClass(Class<?> cl) throws IOException {
}
/**
* This method will allow trusted subclasses of ObjectOutputStream to
* substitute one object for another during serialization. Replacing
* objects is disabled until enableReplaceObject is called. The
* enableReplaceObject method checks that the stream requesting to do
* replacement can be trusted. The first occurrence of each object written
* into the serialization stream is passed to replaceObject. Subsequent
* references to the object are replaced by the object returned by the
* original call to replaceObject. To ensure that the private state of
* objects is not unintentionally exposed, only trusted streams may use
* replaceObject.
*
* <p>The ObjectOutputStream.writeObject method takes a parameter of type
* Object (as opposed to type Serializable) to allow for cases where
* non-serializable objects are replaced by serializable ones.
*
* <p>When a subclass is replacing objects it must insure that either a
* complementary substitution must be made during deserialization or that
* the substituted object is compatible with every field where the
* reference will be stored. Objects whose type is not a subclass of the
* type of the field or array element abort the serialization by raising an
* exception and the object is not be stored.
*
* <p>This method is called only once when each object is first
* encountered. All subsequent references to the object will be redirected
* to the new object. This method should return the object to be
* substituted or the original object.
*
* <p>Null can be returned as the object to be substituted, but may cause
* NullReferenceException in classes that contain references to the
* original object since they may be expecting an object instead of
* null.
*
* @param obj the object to be replaced
*
* @return the alternate object that replaced the specified one
*
* @throws IOException Any exception thrown by the underlying
* OutputStream.
*/
// [子类覆盖]如果写入的是普通对象,且开启了enableReplace,则在序列化前会调用该方法进行对象替换
protected Object replaceObject(Object obj) throws IOException {
return obj;
}
/**
* The writeStreamHeader method is provided so subclasses can append or prepend their own header to the stream.
* It writes the magic number and version to the stream.
*
* @throws IOException if I/O errors occur while writing to the underlying stream
*/
// [子类覆盖]写入序列化头(包含一个魔数和一个版本号)
protected void writeStreamHeader() throws IOException {
bout.writeShort(STREAM_MAGIC);
bout.writeShort(STREAM_VERSION);
}
/**
* Underlying writeObject/writeUnshared implementation.
*/
// 默认的序列化逻辑:将指定的对象obj写入输出流,unshared指示该对象是否为非共享
private void writeObject0(Object obj, boolean unshared) throws IOException {
// 设置块数据模式为false,并返回旧的模式
boolean oldMode = bout.setBlockDataMode(false);
depth++;
try {
// handle previously written and non-replaceable objects
int h;
// 获取obj的替换对象,如果不存在,返回其自身
obj = subs.lookup(obj);
if(obj == null) {
writeNull(); // 向最终输出流写入null
return;
// 如果obj是共享对象,且obj已存在于对象顺序表中
} else if(!unshared && (h = handles.lookup(obj)) != -1) {
writeHandle(h); // 向最终输出流写入共享对象在对象顺序表中的位置
return;
// 如果obj是类对象
} else if(obj instanceof Class) {
writeClass((Class) obj, unshared); // 向最终输出流写入类对象obj的序列化描述符信息
return;
} else if(obj instanceof ObjectStreamClass) {
writeClassDesc((ObjectStreamClass) obj, unshared);
return;
}
/* check for replacement object */
// 记录待序列化对象的原始形式
Object orig = obj;
// 待序列化对象的类型
Class<?> cl = obj.getClass();
// 待序列化类的序列化描述符
ObjectStreamClass desc;
for(; ; ) {
// REMIND: skip this check for strings/arrays?
Class<?> repCl;
// 获取类对象cl的序列化描述符,返回之前会先去缓存中查找
desc = ObjectStreamClass.lookup(cl, true);
if(!desc.hasWriteReplaceMethod() // 是否包含writeReplace方法
|| (obj = desc.invokeWriteReplace(obj)) == null // 调用obj对象的writeReplace方法,生成的对象是否为null
|| (repCl = obj.getClass()) == cl) { // 被替换之后的对象的类型是否维持原样
break;
}
cl = repCl;
}
// 如果需要调用replaceObject()方法(默认为false)
if(enableReplace) {
// 替换对象
Object rep = replaceObject(obj);
// 如果对象发生了变化
if(rep != obj && rep != null) {
cl = rep.getClass();
// 获取类对象cl的序列化描述符,返回之前会先去缓存中查找。
desc = ObjectStreamClass.lookup(cl, true);
}
obj = rep;
}
/* if object replaced, run through original checks a second time */
// 如果待序列化对象已被替换,需要再次检查对象类型
if(obj != orig) {
// 将orig插入到HandleTable的对象顺序表,并将obj插入到ReplaceTable的替换对象顺序表中
subs.assign(orig, obj);
if(obj == null) {
writeNull();
return;
} else if(!unshared && (h = handles.lookup(obj)) != -1) {
writeHandle(h);
return;
} else if(obj instanceof Class) {
writeClass((Class) obj, unshared);
return;
} else if(obj instanceof ObjectStreamClass) {
writeClassDesc((ObjectStreamClass) obj, unshared);
return;
}
}
/* remaining cases */
// 对String类型的对象进行序列化
if(obj instanceof String) {
writeString((String) obj, unshared);
// 对数组类型的对象进行序列化
} else if(cl.isArray()) {
writeArray(obj, desc, unshared);
// 对枚举类型的对象进行序列化
} else if(obj instanceof Enum) {
writeEnum((Enum<?>) obj, desc, unshared);
// 对Serializable类型(包含Externalizable类型)的对象进行序列化
} else if(obj instanceof Serializable) {
writeOrdinaryObject(obj, desc, unshared);
} else {
if(extendedDebugInfo) {
throw new NotSerializableException(cl.getName() + "\n" + debugInfoStack.toString());
} else {
throw new NotSerializableException(cl.getName());
}
}
} finally {
depth--;
bout.setBlockDataMode(oldMode);
}
}
/**
* Fetches and writes values of serializable fields of given object to
* stream. The given class descriptor specifies which field values to
* write, and in which order they should be written.
*/
// 对指定的对象obj执行默认的序列化过程
private void defaultWriteFields(Object obj, ObjectStreamClass desc) throws IOException {
Class<?> cl = desc.forClass();
if(cl != null && obj != null && !cl.isInstance(obj)) {
throw new ClassCastException();
}
// 确保序列化描述符已初始化,且其中不包含无效字段
desc.checkDefaultSerialize();
// 获取fields中基本类型字段所占字节数
int primDataSize = desc.getPrimDataSize();
if(primDataSize>0) {
if(primVals == null || primVals.length<primDataSize) {
primVals = new byte[primDataSize];
}
// 获取obj中所有待序列化的基本类型字段的值
desc.getPrimFieldValues(obj, primVals);
// 向最终输出流(包括块数据缓冲区)写入字节数组b中的数据(此处会直接将数据写入最终输出流)
bout.write(primVals, 0, primDataSize, false);
}
// 获取fields中引用类型字段数量
int numObjFields = desc.getNumObjFields();
if(numObjFields>0) {
// 获取待序列化字段
ObjectStreamField[] fields = desc.getFields(false);
// 存储引用类型字段值
Object[] objVals = new Object[numObjFields];
// 计算基本类型字段数量
int numPrimFields = fields.length - objVals.length;
// 获取待序列化的引用类型字段的值
desc.getObjFieldValues(obj, objVals);
// 遍历引用类型字段
for(int i = 0; i<objVals.length; i++) {
if(extendedDebugInfo) {
debugInfoStack.push("field (class \"" + desc.getName() + "\", name: \"" + fields[numPrimFields + i].getName() + "\", type: \"" + fields[numPrimFields + i].getType() + "\")");
}
try {
// 获取i处的对象是否非共享
boolean unshared = fields[numPrimFields + i].isUnshared();
// 默认的序列化逻辑:将指定的对象obj写入输出流,unshared指示该对象是否为非共享
writeObject0(objVals[i], unshared);
} finally {
if(extendedDebugInfo) {
debugInfoStack.pop();
}
}
}
}
}
/**
* Retrieve the object used to buffer persistent fields to be written to the stream.
* The fields will be written to the stream when writeFields method is called.
*
* @return an instance of the class Putfield that holds the serializable fields
*
* @throws IOException if I/O errors occur
* @since 1.2
*/
/*
* 返回一个PutField对象以存放待序列化字段
*
* 注:该方法需要在待序列化对象自行实现的writeObject()方法内被调用
*/
public ObjectOutputStream.PutField putFields() throws IOException {
if(curPut == null) {
SerialCallbackContext ctx = curContext;
// 只有待序列化的对象自己实现了writeObject()方法,这里的ctx才非空(参见writeSerialData)
if(ctx == null) {
throw new NotActiveException("not in call to writeObject");
}
ctx.checkAndSetUsed();
// 获取待序列化对象的序列化描述符
ObjectStreamClass curDesc = ctx.getDesc();
// 构造一个PutField对象以存放待序列化字段
curPut = new PutFieldImpl(curDesc);
}
return curPut;
}
/**
* Write the buffered fields to the stream.
*
* @throws IOException if I/O errors occur while writing to the underlying
* stream
* @throws NotActiveException Called when a classes writeObject method was
* not called to write the state of the object.
* @since 1.2
*/
// 向当前输出流写入待序列化的字段
public void writeFields() throws IOException {
if(curPut == null) {
throw new NotActiveException("no current PutField object");
}
bout.setBlockDataMode(false);
curPut.writeFields(); // 向当前输出流写入待序列化的字段
bout.setBlockDataMode(true);
}
/*▲ 序列化 ████████████████████████████████████████████████████████████████████████████████┛ */