Nonplussed

Serializer

Dec 11th, 2020 (edited)
65
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Java 4.79 KB | None | 0 0
  1. package com.alandude.serialization;
  2.  
  3. import java.util.ArrayList;
  4. import java.util.List;
  5. import java.io.BufferedOutputStream;
  6. import java.io.File;
  7. import java.io.FileOutputStream;
  8. import java.io.IOException;
  9. import java.io.OutputStream;
  10. import java.lang.reflect.Field;
  11. import java.lang.reflect.Modifier;
  12. import java.nio.file.Path;
  13. import java.nio.file.Paths;
  14.  
  15. public final class Serializer {
  16.     private Serializer() { }
  17.  
  18.     /** Serializes an object into a given output file.
  19.      * Object data always start with a @, and ending with a % */
  20.     private static void serialize(Object o, OutputStream out, List<Object> alreadySerialized) throws IOException {
  21.        
  22.         try {
  23.            
  24.             /* Store the name of the object's class
  25.              *
  26.              * Format:
  27.              *   @[CLASS_NAME]^
  28.              */
  29.             out.write('@');
  30.             for (char c : o.getClass().getName().toCharArray())
  31.                 out.write(c);
  32.             out.write('^');
  33.            
  34.             /* Store the name & value of every field in the object
  35.              * The format:
  36.              *
  37.              *    [NAME]:[TYPE_TAG][VALUE]
  38.              *    [NAME]:[TYPE_TAG][VALUE]
  39.              *     ...
  40.              */
  41.             for (Field field : o.getClass().getDeclaredFields()) {
  42.                 field.setAccessible(true);
  43.                
  44.                 // If this field is transient, don't even write this to the file
  45.                 // So when we deserialize it, we'll leave the field be, Java will
  46.                 // set it to a default value.
  47.                 if (Modifier.isTransient(field.getModifiers()))
  48.                     continue;
  49.                
  50.                 /* Write the field name */
  51.                 for (char c : field.getName().toCharArray())
  52.                     out.write(c);
  53.                 out.write(':');
  54.                
  55.                
  56.                 /* Store the field's value */
  57.                 final Class<?> type = field.getType();
  58.                 final Object val = field.get(o);
  59.                
  60.                 /* This field is not transient, therefore we can save its state */
  61.                 if (type.isPrimitive()) {
  62.                    
  63.                     // Every primitive has a beginning tag:
  64.                     // boolean  0
  65.                     // byte     1
  66.                     // char     2
  67.                     // double   3
  68.                     // float    4
  69.                     // int      5
  70.                     // long     6
  71.                     // short    7
  72.                    
  73.                     // *For objects, they are tagged with 8
  74.                    
  75.                     if (type == Boolean.TYPE) {
  76.                         out.write(0);
  77.                         // 1 = true, 0 = false
  78.                         if ((boolean)val == true)
  79.                             out.write(1);
  80.                         else
  81.                             out.write(0);
  82.                     } else if (type == Byte.TYPE) {
  83.                         out.write(1);
  84.                         out.write((byte) val);
  85.                     } else if (type == Character.TYPE) {
  86.                         out.write(2);
  87.                         out.write((char) val);
  88.                     } else if (type == Double.TYPE) {
  89.                         out.write(3);
  90.                         // TODO: Double
  91.                     } else if (type == Float.TYPE) {
  92.                         out.write(4);
  93.                         // TODO: Float
  94.                     } else if (type == Integer.TYPE) {
  95.                         out.write(5);
  96.                        
  97.                         int i = (int) val;
  98.                         // Apply a mask to contain certain values, shift them so we can
  99.                         // fit them in a byte
  100.                         out.write((i & 0B0000_0000_0000_0000_0000_0000_1111_1111));
  101.                         out.write((i & 0B0000_0000_0000_0000_1111_1111_0000_0000) >> 8);
  102.                         out.write((i & 0B0000_0000_1111_1111_0000_0000_0000_0000) >> 16);
  103.                         out.write((i & 0B1111_1111_0000_0000_0000_0000_0000_0000) >> 24);
  104.                     } else if (type == Long.TYPE) {
  105.                         out.write(6);
  106.                         // TODO: Long
  107.                     } else {
  108.                         /* Shorts */
  109.                         out.write(7);
  110.                         short s = (short) val;
  111.                         // Use bitmasking to get two halves and store them into separate bytes
  112.                         out.write((s & 0B0000_0000_1111_1111));
  113.                         out.write((s & 0B1111_1111_0000_0000) >> 8);
  114.                     }
  115.                 } else {
  116.                     /* Objects */
  117.                     if (val != null) {
  118.                         out.write(8);
  119.                        
  120.                         // TODO: Use alreadySerialized, goddamnit
  121.                        
  122.                         // Whenever an object contains references to other objects, we must serialize them as well.
  123.                         // The exact same process takes place for that inner object, except it is written within
  124.                         // the outer class.
  125.                        
  126.                         // This is also the reason why the data format of classes start with @ and end with %.
  127.                         // We can analyze the structure of the data, and figure out what bytes are part of which class.
  128.                         alreadySerialized.add(val);
  129.                         serialize(val, out, alreadySerialized);
  130.                     }
  131.                 }
  132.             }
  133.            
  134.             out.write('%'); // signify that the the data format is finished
  135.            
  136.         } catch (IllegalArgumentException|IllegalAccessException e) {
  137.             System.out.println("\nAn exception occured when serializing object: " + o);
  138.             System.out.println("---------------------------------------------------------");
  139.             e.printStackTrace();
  140.             System.exit(1);
  141.         }
  142.     }
  143.    
  144.     public static void serialize(Object o, File file) throws IOException {
  145.         try (OutputStream out = new BufferedOutputStream(new FileOutputStream(file))) {
  146.             serialize(o, out, new ArrayList<Object>());
  147.         }
  148.     }
  149.     public static void serialize(Object o, Path path) throws IOException {
  150.         serialize(o, new File(path.toString()));
  151.     }
  152.     public static void serialize(Object o, String path) throws IOException {
  153.         serialize(o, Paths.get(path));
  154.     }
  155. }
Add Comment
Please, Sign In to add comment