java基础加强五(序列化、properties、简单设计模式)

一、java的序列化

序列化:可以将一些数据存储到磁盘上,或者通过网络传输到另一台计算器

反序列化:将一个已被序列化的对象,从硬盘或者网络读取到程序中。

Java的序列化流使用的是ObjectOutputStream,反序列化使用的是ObejctInputStream。其实就是用流的方式将java对象转换为二进制数据存储或传输。

例子:把一个对象存储到磁盘的文件中。

public class User {
    private String name;

    public void setName(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
}
public static void main(String[] args) throws IOException {
    //实例化一个对象
    User u =new User();
    u.setName("小明");
    //构造一个序列化流(输出)
    ObjectOutputStream out =new ObjectOutputStream(new FileOutputStream("D:/Demo/小明.txt"));
    out.writeObject(u);
    out.close();
}

运行main方法,报了一错误,就是NotSerializableException: User,这个原因是如果在java中实例化一个对象,必须标记对象,使其实现Serializable接口。我们看一下api

 

所以让user实现接口后,再次运行,成功。

文件成功生成,存储到了D盘。这就是序列化,如果在编程中,可以用这种方法将对象存储到一个文件中,以便于后续的调用。

反序列化:

public static void main(String[] args) throws IOException, ClassNotFoundException {
    //构造一个反序列化流(输入)
    ObjectInputStream in =new ObjectInputStream(new FileInputStream("D:/Demo/小明.txt"));
    User u = (User) in.readObject();//读取并用user接收,在内存中开辟空间
    System.out.println("看看是不是小明:"+u.getName());
    in.close();
}

打印结果:小明。这样我们就完成了存储和读取。但是目前还有一个问题,就是如果我在user中添加一个属性年龄,再次读取会怎么样呢?当我们再次读取时,会报InvalidClassException错误,我们看一下api

也就是说之前我们存储的user有一个默认的版本号,但是user改变了,运行的时候又换了一个版本号,流中的版本号和以前的不一样。导致了错误。所以我们需要给类加上序列号,用作存储的标识。

加上序列号:1001

public class User implements Serializable {
    private static  final long serialVersionUID=1001L;
    private String name;
    private int age;
    public void setName(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
}

我们再次序列化到文件中,这时我们添加一个属性性别sex,再次反序列化,没有报错,说明序列化起到了作用。由此可见,虽然序列号不是必须添加的,但是为了程序的健壮性,还是建议加上一个序列号。

如果想要使某个属性不参加序列化,比如银行密码,我们可以在属性前面用transient关键字修饰,这样该属性变量就会忽略掉不参加序列化。

二、properties类

properties类是map的一个特殊类,本身也是双列集合。我们看一下api

 

我们看到,Properties 类是可以使用put方法的,但是我们应该更多使用setProperty方法,而且properties可以再流中使用,所以properties的主要作用就是读取和设置配置文件。

用一个设置和读取的示例代码展示使用方法

写入;

public static void main(String[] args) throws IOException {
        Properties p =new Properties();
        p.setProperty("主机地址","114.233.6.9");
        p.setProperty("账号","root");
        p.setProperty("密码","123456");
        //和IO流配合,写文件
        FileWriter out =new FileWriter("my.properties");
        p.store(out,"");//使用store方法向文件中写入配置
        out.close();
    }

读取:

public static void main(String[] args) throws IOException {
        Properties p =new Properties();
        FileReader in =new FileReader("my.properties");
        p.load(in);
        in.close();
        Set<String> keys = p.stringPropertyNames();//可以得到key的集合
        for (String key : keys) {
            System.out.println(key+"====="+p.getProperty(key));
        }
    }

如果只是读取properties文件的化还可以使用ResourceBundle类,它是java.util包的,不仅可以读取配置文件,还可以根据系统自带的国际化语言包翻译成不同的语言。

三、简单设计模式

请看简单的设计模式

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
序列化和反序列化可以破坏单例设计模式的安全性。当一个单例类被序列化后,然后再进行反序列化,会创建出一个新的实例,从而破坏了单例的特性。这是因为序列化和反序列化过程中会创建一个新的对象,并不会调用类的构造函数来初始化新对象。因此,即使单例类被序列化和反序列化,也不能保证只有一个实例存在。 为了解决这个问题,可以在单例类中添加一个readResolve方法,并在该方法中返回单例实例。这样,在反序列化时,就可以通过readResolve方法返回已存在的单例实例,而不是创建一个新的实例。通过这种方式,可以确保单例模式的安全性,避免了序列化和反序列化破坏单例的问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [深入浅出单例模式与反射与序列化对单例的破坏](https://blog.csdn.net/weixin_43975523/article/details/103140654)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [设计模式序列化、反序列化对单例的破坏、原因分析、解决方案及解析](https://blog.csdn.net/leo187/article/details/104332138)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值