不变性(Immutability)模式

1. 不变性(Immutability)模式

1.1. 不变性模式的概念

定义:对象一旦被创建,其内部状态就不再发生变化,也即“只读无写”,不会出现并发写的问题,自然线程安全。

适用场景:只读共享数据场景,是并发编程中最简单直接的线程安全保障方案。

不变性模式的作用:

不变性模式的实际作用不是为了“能不能安全读”,而是为了简化并发程序设计。如果对象不可变:

  • 不需要加锁
  • 不需要 volatile
  • 不用担心可见性问题
  • 不用写复杂的同步控制逻辑

只需保证创建时一次写入,之后多线程随便读即可。

不变性模式不是为了实现线程安全,而是为了“让线程安全成为默认”,并从语言/设计层面强制杜绝修改,从而简化并发程序的设计。

1.2. 实现不变性模式

1. Java 实现方式

public final class Person {private final String name;private final int age;public Person(String name, int age) {this.name = name;this.age = age;}public Person withAge(int newAge) {return new Person(this.name, newAge);}public String getName() { return name; }public int getAge() { return age; }
}

2. Python 实现方式

from dataclasses import dataclass@dataclass(frozen=True)
class Person:name: strage: intp1 = Person("Alice", 30)
p2 = Person(p1.name, 31)  # 通过创建新对象实现“修改”

3. Go 实现方式(没有语言级别支持,只能靠规范)

type Person struct {name stringage  int
}// 提供只读 getter
func (p Person) Name() string {return p.name
}// 修改返回新对象
func (p Person) WithAge(newAge int) Person {return Person{name: p.name, age: newAge}
}

Immutability 模式注意事项

1. final 属性 ≠ 不可变

如果类的属性是对象,即使属性用 final 修饰,属性本身不变,属性引用的对象内容仍然可能被更改。

正确方式:确保引用对象也不可变。

class Foo {int age;
}final class Bar {final Foo foo;  // Foo 是可变的
}

2. 不可变对象也需要正确发布

  • 不可变对象虽然线程安全,但持有不可变对象的引用不一定是线程安全的
  • 若引用在多个线程间共享,需使用 volatile 或原子类保证可见性/原子性。

原子类示例:


AtomicReference<WMRange> rf = new AtomicReference<>(new WMRange(0, 0));

1.3. 享元模式(Flyweight Pattern)

概念

  • 定义:通过复用相同对象,避免重复创建,降低内存消耗。
  • 本质:对象池(缓存)。
  • 关键点
    • 对象不可变(否则复用会有副作用)
    • 对象的创建要通过工厂/缓存控制

1. Java 中的应用

Java 的 Long.valueOf() 使用享元模式缓存了 [-128,127] 范围的数值:

static final Long cache[] = new Long[256];static {for (int i = 0; i < cache.length; i++) {cache[i] = new Long(i - 128);}
}Long.valueOf(long l) {if (l >= -128 && l <= 127)return LongCache.cache[(int) l + 128];return new Long(l);
}

锁问题示例:

Long al = Long.valueOf(1);
Long bl = Long.valueOf(1);
// al == bl,锁是同一把 → 不安全!

2. Python 中的享元模式

Python 的 intstr 对象也使用了享元机制。

a = 100
b = 100
print(a is b)  # True,因为缓存了 [-5, 256] 的整数

自定义享元:

class FlyweightFactory:_cache = {}@staticmethoddef get(value):if value not in FlyweightFactory._cache:FlyweightFactory._cache[value] = HeavyObject(value)return FlyweightFactory._cache[value]

3. Go 中的享元模式

Go 中没有自动享元机制,但可手动实现对象池:

var pool = make(map[string]*User)func GetUser(name string) *User {if u, ok := pool[name]; ok {return u}u := &User{name: name}pool[name] = ureturn u
}
  • Go 标准库中也提供了对象池工具:sync.Pool(并非享元,但类似概念)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/pingmian/84355.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

探秘鸿蒙 HarmonyOS NEXT:鸿蒙定时器,简单倒计时的场景应用

在鸿蒙 ArkTS 开发中&#xff0c;定时器是实现动态效果和定时任务的重要工具。基于鸿蒙 API 12 及以上版本&#xff0c;ArkTS 提供了功能丰富的定时器 API&#xff0c;本文将带你全面了解定时器的使用方法。 一、定时器常用 API 介绍 ArkTS 中的定时器主要分为一次性定时器&a…

安卓基础(语义树)

进化1 package com.example.demotest.unread;import android.accessibilityservice.AccessibilityService; import android.content.res.Resources; import android.graphics.Rect; import android.util.DisplayMetrics; import android.util.Log; import android.view.access…

Linux基础开发工具——vim工具

文章目录 vim工具什么是vimvim的多模式和使用vim的基础模式vim的三种基础模式三种模式的初步了解 常用模式的详细讲解插入模式命令模式模式转化光标的移动文本的编辑 底行模式替换模式视图模式总结 使用vim的小技巧vim的配置(了解) vim工具 本文章仍然是继续讲解Linux系统下的…

C++_核心编程_多态案例二-制作饮品

#include <iostream> #include <string> using namespace std;/*制作饮品的大致流程为&#xff1a;煮水 - 冲泡 - 倒入杯中 - 加入辅料 利用多态技术实现本案例&#xff0c;提供抽象制作饮品基类&#xff0c;提供子类制作咖啡和茶叶*//*基类*/ class AbstractDr…

AcWing--数据结构1

用数组来模拟链表。这种实现链表的方式也叫静态链表。 1.单链表 写邻接表&#xff1a;存储图和树 我们定义&#xff1a;e[N]用来表示某个点的值是多少&#xff1b;ne[N]用来表示某个点的next指针是多少 e和ne是用下标关联起来的 如&#xff1a;head->3->5->7->…

云启出海,智联未来|阿里云网络「企业出海」系列客户沙龙上海站圆满落地

借阿里云中企出海大会的东风&#xff0c;以**「云启出海&#xff0c;智联未来&#xff5c;打造安全可靠的出海云网络引擎」为主题的阿里云企业出海客户沙龙云网络&安全专场于5.28日下午在上海顺利举办&#xff0c;现场吸引了来自携程、小红书、米哈游、哔哩哔哩、波克城市、…

多模态分类案例实现

以下是基于飞桨平台实现的多模态分类详细案例&#xff0c;结合图像和文本信息进行分类任务。案例包含数据处理、模型构建、训练和评估的完整流程&#xff0c;并提供详细注释&#xff1a; 一、多模态分类案例实现 import os import json import numpy as np from PIL import I…

Express框架:Node.js的轻量级Web应用利器

Hi,我是布兰妮甜 !在当今快速发展的Web开发领域,Node.js已成为构建高性能、可扩展网络应用的重要基石。而在这片肥沃的生态系统中,Express框架犹如一座经久不衰的灯塔,指引着无数开发者高效构建Web应用的方向。本文章在为读者提供一份全面而深入的Express框架指南。无论您…

K-Means颜色变卦和渐变色

一、理论深度提升&#xff1a;补充算法细节与数学基础 1. K-Means 算法核心公式&#xff08;增强专业性&#xff09; 在 “原理步骤” 中加入数学表达式&#xff0c;说明聚类目标&#xff1a; K-Means 的目标是最小化簇内平方和&#xff08;Within-Cluster Sum of Squares, W…

深入解析C#表达式求值:优先级、结合性与括号的魔法

—— 为什么2/6*4不等于1/12&#xff1f; &#x1f50d; 一、表达式求值顺序为何重要&#xff1f; 表达式如精密仪器&#xff0c;子表达式求值顺序直接决定结果。例如&#xff1a; int result 3 * 5 2;若先算乘法&#xff1a;(3*5)2 17 ✅若先算加法&#xff1a;3*(52)21…

Docker 离线安装指南

参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性&#xff0c;不同版本的Docker对内核版本有不同要求。例如&#xff0c;Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本&#xff0c;Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…

Spring——Spring相关类原理与实战

摘要 本文深入探讨了 Spring 框架中 InitializingBean 接口的原理与实战应用&#xff0c;该接口是 Spring 提供的一个生命周期接口&#xff0c;用于在 Bean 属性注入完成后执行初始化逻辑。文章详细介绍了接口定义、作用、典型使用场景&#xff0c;并与其他相关概念如 PostCon…

Angular微前端架构:Module Federation + ngx-build-plus (Webpack)

以下是一个完整的 Angular 微前端示例&#xff0c;其中使用的是 Module Federation 和 npx-build-plus 实现了主应用&#xff08;Shell&#xff09;与子应用&#xff08;Remote&#xff09;的集成。 &#x1f6e0;️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放

简介 前面两期文章我们介绍了I2S的读取和写入&#xff0c;一个是通过INMP441麦克风模块采集音频&#xff0c;一个是通过PCM5102A模块播放音频&#xff0c;那如果我们将两者结合起来&#xff0c;将麦克风采集到的音频通过PCM5102A播放&#xff0c;是不是就可以做一个扩音器了呢…

冯诺依曼架构是什么?

冯诺依曼架构是什么&#xff1f; 冯诺依曼架构&#xff08;Von Neumann Architecture&#xff09;是现代计算机的基础设计框架&#xff0c;由数学家约翰冯诺依曼&#xff08;John von Neumann&#xff09;及其团队在1945年提出。其核心思想是通过统一存储程序与数据&#xff0…

【持续更新】linux网络编程试题

问题1 请简要说明TCP/IP协议栈的四层结构&#xff0c;并分别举出每一层出现的典型协议或应用。 答案 应用层&#xff1a;ping,telnet,dns 传输层&#xff1a;tcp,udp 网络层&#xff1a;ip,icmp 数据链路层&#xff1a;arp,rarp 问题2 下列协议或应用分别属于TCP/IP协议…

椭圆曲线密码学(ECC)

一、ECC算法概述 椭圆曲线密码学&#xff08;Elliptic Curve Cryptography&#xff09;是基于椭圆曲线数学理论的公钥密码系统&#xff0c;由Neal Koblitz和Victor Miller在1985年独立提出。相比RSA&#xff0c;ECC在相同安全强度下密钥更短&#xff08;256位ECC ≈ 3072位RSA…

【JVM】- 内存结构

引言 JVM&#xff1a;Java Virtual Machine 定义&#xff1a;Java虚拟机&#xff0c;Java二进制字节码的运行环境好处&#xff1a; 一次编写&#xff0c;到处运行自动内存管理&#xff0c;垃圾回收的功能数组下标越界检查&#xff08;会抛异常&#xff0c;不会覆盖到其他代码…

React 基础入门笔记

一、JSX语法规则 1. 定义虚拟DOM时&#xff0c;不要写引号 2.标签中混入JS表达式时要用 {} &#xff08;1&#xff09;.JS表达式与JS语句&#xff08;代码&#xff09;的区别 &#xff08;2&#xff09;.使用案例 3.样式的类名指定不要用class&#xff0c;要用className 4.内…

Linux链表操作全解析

Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表&#xff1f;1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…