几个问题(九)-------序列化的作用,应用场景,除了网路传输

关于更多java序列化的问题可以参见JAVA系列之对象的序列化与反序列化
      Java的序列化机制只序列化对象的属性值,而不会去序列化什么所谓的方法。其实这个问题简单思考一下就可以搞清楚,方法是不带状态的,就是一些指令,指令是不需要序列化的,只要你的JVM classloader可以load到这个类,那么类方法指令自然就可以获得。序列化真正需要保存的只是对象属性的值,和对象的类型。

我们可以做一个简单的小试验,来证实一下:

Java代码 < type="application/x-shockwave-flash" width="14" height="15" src="javascripts/syntaxhighlighter/clipboard_new.swf" pluginspage="http://www.macromedia.com/go/getflashplayer" allowscriptaccess="always" quality="high" flashvars="clipboard=package%20com.javaeye%3B%0A%0Aimport%20java.io.Serializable%3B%0A%0Apublic%20class%20DomainObject%20%20implements%20Serializable%20%7B%0A%0A%09private%20String%20name%3B%0A%09%0A%09private%20int%20age%20%3B%0A%0A%09public%20int%20getAge()%3B%20%7B%0A%09%09return%20age%3B%0A%09%7D%0A%0A%09public%20void%20setAge(int%20age)%3B%20%7B%0A%09%09this.age%20%3D%20age%3B%0A%09%7D%0A%0A%09public%20String%20getName()%3B%20%7B%0A%09%09return%20name%3B%0A%09%7D%0A%0A%09public%20void%20setName(String%20name)%3B%20%7B%0A%09%09this.name%20%3D%20name%3B%0A%09%7D%0A%09%0A%09%0A%7D">
package com.javaeye;

import java.io.Serializable;

public class DomainObject  implements Serializable {

 private String name;
 
 private int age ;

 public int getAge(); {
  return age;
 }

 public void setAge(int age); {
  this.age = age;
 }

 public String getName(); {
  return name;
 }

 public void setName(String name); {
  this.name = name;
 }
 
 
}view plaincopy to clipboardprint?
package com.javaeye;    
    
import java.io.Serializable;    
    
public class DomainObject  implements Serializable {    
    
    private String name;    
        
    private int age ;    
    
    public int getAge(); {    
        return age;    
    }    
    
    public void setAge(int age); {    
        this.age = age;    
    }    
    
    public String getName(); {    
        return name;    
    }    
    
    public void setName(String name); {    
        this.name = name;    
    }    
        
        
}   
 package com.javaeye; 
  
 import java.io.Serializable; 
  
 public class DomainObject  implements Serializable { 
  
     private String name; 
      
     private int age ; 
  
     public int getAge(); { 
         return age; 
     } 
  
     public void setAge(int age); { 
         this.age = age; 
     } 
  
     public String getName(); { 
         return name; 
     } 
  
     public void setName(String name); { 
         this.name = name; 
     } 
      
      
 }  
Java代码
package com.javaeye;

import java.io.FileOutputStream;
import java.io.ObjectOutputStream;

public class Main {

 public static void main(String[] args); throws Exception {
  DomainObject obj = new DomainObject();;
  obj.setAge(29);;
  obj.setName("fankai");;
  FileOutputStream fos = new FileOutputStream("DomainObject");;
  ObjectOutputStream oos = new ObjectOutputStream(fos);;
  oos.writeObject(obj);;
  oos.close();;
  fos.close();;
 }

}view plaincopy to clipboardprint?
package com.javaeye;    
    
import java.io.FileOutputStream;    
import java.io.ObjectOutputStream;    
    
public class Main {    
    
    public static void main(String[] args); throws Exception {    
        DomainObject obj = new DomainObject();;    
        obj.setAge(29);;    
        obj.setName("fankai");;    
        FileOutputStream fos = new FileOutputStream("DomainObject");;    
        ObjectOutputStream oos = new ObjectOutputStream(fos);;    
        oos.writeObject(obj);;    
        oos.close();;    
        fos.close();;    
    }    
    
}  
 package com.javaeye; 
  
 import java.io.FileOutputStream; 
 import java.io.ObjectOutputStream; 
  
 public class Main { 
  
     public static void main(String[] args); throws Exception { 
         DomainObject obj = new DomainObject();; 
         obj.setAge(29);; 
         obj.setName("fankai");; 
         FileOutputStream fos = new FileOutputStream("DomainObject");; 
         ObjectOutputStream oos = new ObjectOutputStream(fos);; 
         oos.writeObject(obj);; 
         oos.close();; 
         fos.close();; 
     } 
  
 } 
      DomainObject是我们准备序列化的类,在Main里面,我们new一个DomainObject的对象,然后赋值,最后把该对象序列化到一个硬盘文件中。然后使用一种支持二进制编辑器,例如UltraEdit打开这个文件,看看Java都对DomainObject序列化了哪些信息,你就什么都明白了。

为了更方便观察,我使用Linux下面的strings去提取文本信息,输出为:
robbin@linux:~> strings DomainObject
com.javaeye.DomainObject
ageL
namet
Ljava/lang/String;xp
fankai
这些信息很直观的告诉我们序列化都保存了些什么内容:
1)对象的类型
2)对象属性的类型
3)对象属性的值

并没有什么方法签名的信息,更不要说什么序列化方法了。
然后我们再做一个试验,给DomainObject增加两个方法:
Java代码 < type="application/x-shockwave-flash" width="14" height="15" src="javascripts/syntaxhighlighter/clipboard_new.swf" pluginspage="http://www.macromedia.com/go/getflashplayer" allowscriptaccess="always" quality="high" flashvars="clipboard=package%20com.javaeye%3B%0A%0Aimport%20java.io.Serializable%3B%0A%0Apublic%20class%20DomainObject%20%20implements%20Serializable%20%7B%0A%0A%09private%20String%20name%3B%0A%09%0A%09private%20int%20age%20%3B%0A%0A%09public%20int%20getAge()%3B%20%7B%0A%09%09return%20age%3B%0A%09%7D%0A%0A%09public%20void%20setAge(int%20age)%3B%20%7B%0A%09%09this.age%20%3D%20age%3B%0A%09%7D%0A%0A%09public%20String%20getName()%3B%20%7B%0A%09%09return%20name%3B%0A%09%7D%0A%0A%09public%20void%20setName(String%20name)%3B%20%7B%0A%09%09this.name%20%3D%20name%3B%0A%09%7D%0A%09%0A%09public%20String%20toString()%3B%20%7B%0A%09%09return%20%22This%20is%20a%20serializable%20test!%22%3B%0A%09%7D%0A%09%0A%09public%20void%20doSomeWork()%3B%20%7B%0A%09%09System.out.println(%22hello%22)%3B%3B%0A%09%7D%0A%7D">
package com.javaeye;

import java.io.Serializable;

public class DomainObject  implements Serializable {

 private String name;
 
 private int age ;

 public int getAge(); {
  return age;
 }

 public void setAge(int age); {
  this.age = age;
 }

 public String getName(); {
  return name;
 }

 public void setName(String name); {
  this.name = name;
 }
 
 public String toString(); {
  return "This is a serializable test!";
 }
 
 public void doSomeWork(); {
  System.out.println("hello");;
 }
}view plaincopy to clipboardprint?
package com.javaeye;    
    
import java.io.Serializable;    
    
public class DomainObject  implements Serializable {    
    
    private String name;    
        
    private int age ;    
    
    public int getAge(); {    
        return age;    
    }    
    
    public void setAge(int age); {    
        this.age = age;    
    }    
    
    public String getName(); {    
        return name;    
    }    
    
    public void setName(String name); {    
        this.name = name;    
    }    
        
    public String toString(); {    
        return "This is a serializable test!";    
    }    
        
    public void doSomeWork(); {    
        System.out.println("hello");;    
    }    
}    
 package com.javaeye; 
  
 import java.io.Serializable; 
  
 public class DomainObject  implements Serializable { 
  
     private String name; 
      
     private int age ; 
  
     public int getAge(); { 
         return age; 
     } 
  
     public void setAge(int age); { 
         this.age = age; 
     } 
  
     public String getName(); { 
         return name; 
     } 
  
     public void setName(String name); { 
         this.name = name; 
     } 
      
     public String toString(); { 
         return "This is a serializable test!"; 
     } 
      
     public void doSomeWork(); { 
         System.out.println("hello");; 
     } 
 }   
      我们增加了toString方法和doSomeWork方法,按照你的理论,如果序列化方法的话,产生的文件体积必然增大。记录一下文件体积,92Byte,好了,删除,运行程序,生成了新的文件,看一下体积,还是92Byte!

拿到Linux下面再提取一下字符串:
robbin@linux:~> strings DomainObject
com.javaeye.DomainObject
ageL
namet
Ljava/lang/String;xp
fankai
完全一模一样!

然后我们再做第三个试验,这次把DomainObject的两个属性以及相关方法删除掉:
Java代码 < type="application/x-shockwave-flash" width="14" height="15" src="javascripts/syntaxhighlighter/clipboard_new.swf" pluginspage="http://www.macromedia.com/go/getflashplayer" allowscriptaccess="always" quality="high" flashvars="clipboard=package%20com.javaeye%3B%0A%0Aimport%20java.io.Serializable%3B%0A%0Apublic%20class%20DomainObject%20%20implements%20Serializable%20%7B%0A%0A%09public%20String%20toString()%3B%20%7B%0A%09%09return%20%22This%20is%20a%20serializable%20test!%22%3B%0A%09%7D%0A%09%0A%09public%20void%20doSomeWork()%3B%20%7B%0A%09%09System.out.println(%22hello%22)%3B%3B%0A%09%7D%0A%7D">
package com.javaeye;

import java.io.Serializable;

public class DomainObject  implements Serializable {

 public String toString(); {
  return "This is a serializable test!";
 }
 
 public void doSomeWork(); {
  System.out.println("hello");;
 }
}view plaincopy to clipboardprint?
package com.javaeye;    
    
import java.io.Serializable;    
    
public class DomainObject  implements Serializable {    
    
    public String toString(); {    
        return "This is a serializable test!";    
    }    
        
    public void doSomeWork(); {    
        System.out.println("hello");;    
    }    
}   
 package com.javaeye; 
  
 import java.io.Serializable; 
  
 public class DomainObject  implements Serializable { 
  
     public String toString(); { 
         return "This is a serializable test!"; 
     } 
      
     public void doSomeWork(); { 
         System.out.println("hello");; 
     } 
 }  
修改Main类如下:
Java代码 < type="application/x-shockwave-flash" width="14" height="15" src="javascripts/syntaxhighlighter/clipboard_new.swf" pluginspage="http://www.macromedia.com/go/getflashplayer" allowscriptaccess="always" quality="high" flashvars="clipboard=package%20com.javaeye%3B%0A%0Aimport%20java.io.FileOutputStream%3B%0Aimport%20java.io.ObjectOutputStream%3B%0A%0Apublic%20class%20Main%20%7B%0A%0A%09public%20static%20void%20main(String%5B%5D%20args)%3B%20throws%20Exception%20%7B%0A%09%09DomainObject%20obj%20%3D%20new%20DomainObject()%3B%3B%0A%0A%09%09FileOutputStream%20fos%20%3D%20new%20FileOutputStream(%22DomainObject%22)%3B%3B%0A%09%09ObjectOutputStream%20oos%20%3D%20new%20ObjectOutputStream(fos)%3B%3B%0A%09%09oos.writeObject(obj)%3B%3B%0A%09%09oos.close()%3B%3B%0A%09%09fos.close()%3B%3B%0A%09%7D%0A%0A%7D">
package com.javaeye;

import java.io.FileOutputStream;
import java.io.ObjectOutputStream;

public class Main {

 public static void main(String[] args); throws Exception {
  DomainObject obj = new DomainObject();;

  FileOutputStream fos = new FileOutputStream("DomainObject");;
  ObjectOutputStream oos = new ObjectOutputStream(fos);;
  oos.writeObject(obj);;
  oos.close();;
  fos.close();;
 }

}view plaincopy to clipboardprint?
package com.javaeye;    
    
import java.io.FileOutputStream;    
import java.io.ObjectOutputStream;    
    
public class Main {    
    
    public static void main(String[] args); throws Exception {    
        DomainObject obj = new DomainObject();;    
    
        FileOutputStream fos = new FileOutputStream("DomainObject");;    
        ObjectOutputStream oos = new ObjectOutputStream(fos);;    
        oos.writeObject(obj);;    
        oos.close();;    
        fos.close();;    
    }    
    
}   
 package com.javaeye; 
  
 import java.io.FileOutputStream; 
 import java.io.ObjectOutputStream; 
  
 public class Main { 
  
     public static void main(String[] args); throws Exception { 
         DomainObject obj = new DomainObject();; 
  
         FileOutputStream fos = new FileOutputStream("DomainObject");; 
         ObjectOutputStream oos = new ObjectOutputStream(fos);; 
         oos.writeObject(obj);; 
         oos.close();; 
         fos.close();; 
     } 
  
 }  
      按照你的理论,如果序列化方法的话,我们必然应该在文件里面发现方法的签名信息,甚至方法里面包含的字符串,好了,再运行一遍,然后打开看一下吧!文件现在体积变成了45Byte,拿到Linux下面提取一下信息:
robbin@linux:~> strings DomainObject
com.javaeye.DomainObject
只有对象的类型信息,再无其它东西了!

      请记住序列化机制只保存对象的类型信息,属性的类型信息和属性值,和方法没有什么关系,你就是给这个类增加10000个方法,序列化内容也不会增加任何东西,不要想当然的臆测自己不了解的知识,动手去做!


本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/ilibaba/archive/2009/03/10/3975680.aspx

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java序列和反序列有以下几个应用场景: 1. 网络传输:将Java对象序列后通过网络传输到另一台机器上,实现分布式系统中的远程调用。 2. 缓存:将Java对象序列后存储到缓存中,提高系统性能。 3. 消息队列:将Java对象序列后发送到消息队列中,实现异步处理。 4. 持久存储:将Java对象序列后存储到磁盘中,实现数据持久。 以下是Java序列和反序列的示例代码: ```java import java.io.*; public class SerializationDemo { public static void main(String[] args) { // 序列对象 try { // 创建一个Person对象 Person person = new Person("Tom", 18); // 创建一个ObjectOutputStream对象 ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("person.txt")); // 将person对象写入到文件中 oos.writeObject(person); // 关闭流 oos.close(); } catch (IOException e) { e.printStackTrace(); } // 反序列对象 try { // 创建一个ObjectInputStream对象 ObjectInputStream ois = new ObjectInputStream(new FileInputStream("person.txt")); // 从文件中读取Person对象 Person person = (Person) ois.readObject(); // 输出Person对象的信息 System.out.println(person.getName() + " " + person.getAge()); // 关闭流 ois.close(); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } } } // Person类需要实现Serializable接口才能被序列 class Person implements Serializable { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值