目次
はじめに
キャッシュを有効にする @EnableCaching
カスタム キャッシュ マネージャー
@Cacheable
@CachePut
@CacheConfig
Condition & Unless
清理全部缓存
SpEL表达式
最佳实践
ホームページ Java &#&チュートリアル SpringBootプロジェクトでキャッシュを使用する方法

SpringBootプロジェクトでキャッシュを使用する方法

May 16, 2023 pm 02:34 PM
cache springboot

はじめに

キャッシュは、頻繁にアクセスされるデータをメモリに保存し、データベースなどの基盤となるデータ ソースへの負担を軽減することで、システムのパフォーマンスと安定性を効果的に向上させることができます。誰もがプロジェクトで多かれ少なかれこれを使用したことがあると思いますが、私たちのプロジェクトも例外ではありませんでした。しかし、最近会社のコードをレビューしていたとき、その記述は非常に愚かで低レベルでした。大まかな記述は次のとおりです:

public User getById(String id) {
	User user = cache.getUser();
    if(user != null) {
        return user;
    }
    // 从数据库获取
    user = loadFromDB(id);
    cahce.put(id, user);
	return user;
}
ログイン後にコピー

実際、Spring Boot は、アプリケーションにキャッシュを簡単に追加できる強力なキャッシュ抽象化を提供します。この記事では、Spring が提供するさまざまなキャッシュ アノテーションを使用してキャッシュのベスト プラクティスを実装する方法について説明します。

キャッシュを有効にする @EnableCaching

現在、ほとんどのプロジェクトは SpringBoot プロジェクトなので、アノテーション @EnableCaching をスタートアップ クラスに追加して、キャッシュ機能を有効にすることができます。

@SpringBootApplication
@EnableCaching
public class SpringCacheApp {

    public static void main(String[] args) {
        SpringApplication.run(Cache.class, args);
    }
}
ログイン後にコピー

キャッシュを使用できるようにしたいので、キャッシュ マネージャー Bean が必要です。デフォルトでは、@EnableCachingConcurrentMapCacheManager Bean を登録します。いいえ個別の Bean 宣言。 ConcurrentMapCacheManager は、値を ConcurrentHashMap のインスタンスに保存します。これは、キャッシュ メカニズムの最も単純なスレッドセーフな実装です。

カスタム キャッシュ マネージャー

デフォルトのキャッシュ マネージャーは JVM メモリに保存されているためニーズを満たすことができません。では、Redis に保存するにはどうすればよいでしょうか?現時点では、カスタム キャッシュ マネージャーを追加する必要があります。

1. 依存関係を追加する

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
ログイン後にコピー

2. Redis キャッシュ マネージャーを構成する

@Configuration
@EnableCaching
public class CacheConfig {

    @Bean
    public RedisConnectionFactory redisConnectionFactory() {
        return new LettuceConnectionFactory();
    }

    @Bean
    public CacheManager cacheManager() {
        RedisCacheConfiguration redisCacheConfiguration = RedisCacheConfiguration.defaultCacheConfig()
            .disableCachingNullValues()
            .serializeValuesWith(SerializationPair.fromSerializer(new GenericJackson2JsonRedisSerializer()));

        RedisCacheManager redisCacheManager = RedisCacheManager.builder(redisConnectionFactory())
            .cacheDefaults(redisCacheConfiguration)
            .build();

        return redisCacheManager;
    }
}
ログイン後にコピー

キャッシュ マネージャーを用意したところで、ビジネス レベルでキャッシュを操作するにはどうすればよいでしょうか?

@Cacheable@CachePut、または @CacheEvict アノテーションを使用してキャッシュを操作できます。

@Cacheable

このアノテーションはメソッドの実行結果をキャッシュできます。キャッシュ制限時間内に再度メソッドが呼び出された場合、メソッド自体は呼び出されませんが、結果が取得されます。キャッシュから直接取得され、呼び出し元に返されます。

SpringBootプロジェクトでキャッシュを使用する方法

例 1: データベース クエリの結果をキャッシュします。

@Service
public class MyService {

    @Autowired
    private MyRepository repository;

    @Cacheable(value = "myCache", key = "#id")
    public MyEntity getEntityById(Long id) {
        return repository.findById(id).orElse(null);
    }
}
ログイン後にコピー

この例では、@Cacheable アノテーションを使用して、ID に基づいて getEntityById() メソッドの結果をキャッシュします。データベースから MyEntity オブジェクトを取得します。

しかし、データを更新したらどうなるでしょうか?古いデータがまだキャッシュにありますか?

@CachePut

その後、@CachePut が出てきました。@Cacheable アノテーションとの違いは、@CachePut## を使用することです。 # annotation annotation. このメソッドは、実行前にキャッシュ内に以前に実行された結果があるかどうかを確認しません。代わりに、メソッドは毎回実行され、実行結果はキーと値のペアの形式で指定されたキャッシュに書き込まれます。 @CachePut アノテーションは通常、キャッシュデータを更新するために使用されます。これは、書き込みモードでの二重書き込みモードを使用するキャッシュに相当します。

@Service
public class MyService {

    @Autowired
    private MyRepository repository;

    @CachePut(value = "myCache", key = "#entity.id")
    public void saveEntity(MyEntity entity) {
        repository.save(entity);
    }
}
ログイン後にコピー

@CacheEvict

@CacheEvict とマークされたメソッドは、呼び出されたときに、保存されているデータをキャッシュから削除します。 @CacheEvict アノテーションは通常、キャッシュされたデータを削除するために使用されます。これは、書き込みモードで障害モードを使用するキャッシュと同等です。

SpringBootプロジェクトでキャッシュを使用する方法

@Service
public class MyService {

    @Autowired
    private MyRepository repository;

     @CacheEvict(value = "myCache", key = "#id")
    public void deleteEntityById(Long id) {
        repository.deleteById(id);
    }
}
ログイン後にコピー

@Caching

@Caching アノテーションは、メソッドまたはクラスで複数の Spring Cache 関連のものを同時に指定するために使用されます。時間の注釈。

SpringBootプロジェクトでキャッシュを使用する方法

例 1:

@Caching アノテーションで指定された evict 属性は、メソッド saveEntity## が実行されると無効になります。 # は 2 つのキャッシュと呼ばれます。 <div class="code" style="position:relative; padding:0px; margin:0px;"><pre class='brush:php;toolbar:false;'>@Service public class MyService { @Autowired private MyRepository repository; @Cacheable(value = &quot;myCache&quot;, key = &quot;#id&quot;) public MyEntity getEntityById(Long id) { return repository.findById(id).orElse(null); } @Caching(evict = { @CacheEvict(value = &quot;myCache&quot;, key = &quot;#entity.id&quot;), @CacheEvict(value = &quot;otherCache&quot;, key = &quot;#entity.id&quot;) }) public void saveEntity(MyEntity entity) { repository.save(entity); } }</pre><div class="contentsignin">ログイン後にコピー</div></div>例 2:

getEntityById

メソッドを呼び出すと、Spring はまず結果が myCache キャッシュにキャッシュされているかどうかを確認します。その場合、Spring はメソッドを実行する代わりに、キャッシュされた結果を返します。結果がまだキャッシュされていない場合、Spring はメソッドを実行し、結果を myCache キャッシュにキャッシュします。メソッドの実行後、Spring は @CacheEvict アノテーションに基づいて、キャッシュされた結果を otherCache キャッシュから削除します。 <div class="code" style="position:relative; padding:0px; margin:0px;"><pre class='brush:php;toolbar:false;'>@Service public class MyService { @Caching( cacheable = { @Cacheable(value = &quot;myCache&quot;, key = &quot;#id&quot;) }, evict = { @CacheEvict(value = &quot;otherCache&quot;, key = &quot;#id&quot;) } ) public MyEntity getEntityById(Long id) { return repository.findById(id).orElse(null); } }</pre><div class="contentsignin">ログイン後にコピー</div></div>例 3:

saveData

メソッドを呼び出すと、Spring はまず @CacheEvict アノテーションに基づいて otherCache キャッシュからデータを削除します。 Spring はメソッドを実行し、結果をデータベースまたは外部 API に保存します。 メソッドが実行されると、Spring は

@ に基づいて結果を

myCachemyOtherCache、および myThirdCache キャッシュに追加します。 CachePut アノテーション。中央。 Spring は、@Cacheable アノテーションに基づいて、結果が myFourthCache および myFifthCache キャッシュにキャッシュされているかどうかもチェックします。結果がまだキャッシュされていない場合、Spring は結果を適切なキャッシュにキャッシュします。結果がすでにキャッシュされている場合、Spring はメソッドを再度実行する代わりに、キャッシュされた結果を返します。 <div class="code" style="position:relative; padding:0px; margin:0px;"><pre class='brush:php;toolbar:false;'>@Service public class MyService { @Caching( put = { @CachePut(value = &quot;myCache&quot;, key = &quot;#result.id&quot;), @CachePut(value = &quot;myOtherCache&quot;, key = &quot;#result.id&quot;), @CachePut(value = &quot;myThirdCache&quot;, key = &quot;#result.name&quot;) }, evict = { @CacheEvict(value = &quot;otherCache&quot;, key = &quot;#id&quot;) }, cacheable = { @Cacheable(value = &quot;myFourthCache&quot;, key = &quot;#id&quot;), @Cacheable(value = &quot;myFifthCache&quot;, key = &quot;#result.id&quot;) } ) public MyEntity saveData(Long id, String name) { // Code to save data to a database or external API MyEntity entity = new MyEntity(id, name); return entity; } }</pre><div class="contentsignin">ログイン後にコピー</div></div><h3 id="CacheConfig">@CacheConfig</h3><p>通过<code>@CacheConfig 注解,我们可以将一些缓存配置简化到类级别的一个地方,这样我们就不必多次声明相关值:

@CacheConfig(cacheNames={"myCache"})
@Service
public class MyService {

    @Autowired
    private MyRepository repository;

    @Cacheable(key = "#id")
    public MyEntity getEntityById(Long id) {
        return repository.findById(id).orElse(null);
    }

    @CachePut(key = "#entity.id")
    public void saveEntity(MyEntity entity) {
        repository.save(entity);
    }

    @CacheEvict(key = "#id")
    public void deleteEntityById(Long id) {
        repository.deleteById(id);
    }
}
ログイン後にコピー

Condition & Unless

  • condition作用:指定缓存的条件(满足什么条件才缓存),可用 SpEL 表达式(如 #id>0,表示当入参 id 大于 0 时才缓存)

  • unless作用 : 否定缓存,即满足 unless 指定的条件时,方法的结果不进行缓存,使用 unless 时可以在调用的方法获取到结果之后再进行判断(如 #result == null,表示如果结果为 null 时不缓存)

//when id >10, the @CachePut works. 
@CachePut(key = "#entity.id", condition="#entity.id > 10")
public void saveEntity(MyEntity entity) {
	repository.save(entity);
}


//when result != null, the @CachePut works.
@CachePut(key = "#id", condition="#result == null")
public void saveEntity1(MyEntity entity) {
	repository.save(entity);
}
ログイン後にコピー

清理全部缓存

通过allEntriesbeforeInvocation属性可以来清除全部缓存数据,不过allEntries是方法调用后清理,beforeInvocation是方法调用前清理。

//方法调用完成之后,清理所有缓存
@CacheEvict(value="myCache",allEntries=true)
public void delectAll() {
    repository.deleteAll();
}

//方法调用之前,清除所有缓存
@CacheEvict(value="myCache",beforeInvocation=true)
public void delectAll() {
    repository.deleteAll();
}
ログイン後にコピー

SpEL表达式

Spring Cache注解中频繁用到SpEL表达式,那么具体如何使用呢?

SpEL 表达式的语法

SpringBootプロジェクトでキャッシュを使用する方法

Spring Cache可用的变量

SpringBootプロジェクトでキャッシュを使用する方法

最佳实践

通过Spring缓存注解可以快速优雅地在我们项目中实现缓存的操作,但是在双写模式或者失效模式下,可能会出现缓存数据一致性问题(读取到脏数据),Spring Cache 暂时没办法解决。最后我们再总结下Spring Cache使用的一些最佳实践。

  • 只缓存经常读取的数据:缓存可以显着提高性能,但只缓存经常访问的数据很重要。很少或从不访问的缓存数据会占用宝贵的内存资源,从而导致性能问题。

  • 根据应用程序的特定需求选择合适的缓存提供程序和策略。SpringBoot 支持多种缓存提供程序,包括 EhcacheHazelcast 和 Redis

  • 使用缓存时请注意潜在的线程安全问题。对缓存的并发访问可能会导致数据不一致或不正确,因此选择线程安全的缓存提供程序并在必要时使用适当的同步机制非常重要。

  • 避免过度缓存。缓存对于提高性能很有用,但过多的缓存实际上会消耗宝贵的内存资源,从而损害性能。在缓存频繁使用的数据和允许垃圾收集不常用的数据之间取得平衡很重要。

  • 使用适当的缓存逐出策略。使用缓存时,重要的是定义适当的缓存逐出策略以确保在必要时从缓存中删除旧的或陈旧的数据。

  • 使用适当的缓存键设计。缓存键对于每个数据项都应该是唯一的,并且应该考虑可能影响缓存数据的任何相关参数,例如用户 ID、时间或位置。

  • 常规数据(读多写少、即时性与一致性要求不高的数据)完全可以使用 Spring Cache,至于写模式下缓存数据一致性问题的解决,只要缓存数据有设置过期时间就足够了。

  • 特殊数据(读多写多、即时性与一致性要求非常高的数据),不能使用 Spring Cache,建议考虑特殊的设计(例如使用 Cancal 中间件等)。

以上がSpringBootプロジェクトでキャッシュを使用する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

Springboot が Jasypt を統合して構成ファイルの暗号化を実装する方法 Springboot が Jasypt を統合して構成ファイルの暗号化を実装する方法 Jun 01, 2023 am 08:55 AM

Jasypt の概要 Jasypt は、開発者が最小限の労力で基本的な暗号化機能を自分のプロジェクトに追加できる Java ライブラリであり、暗号化の仕組みを深く理解する必要はありません。一方向および双方向暗号化の高いセキュリティ。標準ベースの暗号化テクノロジー。パスワード、テキスト、数値、バイナリを暗号化します... Spring ベースのアプリケーション、オープン API への統合、JCE プロバイダーでの使用に適しています... 次の依存関係を追加します: com.github.ulisesbocchiojasypt-spring-boot-starter2. 1.1 Jasypt の特典はシステムのセキュリティを保護し、コードが漏洩した場合でもデータ ソースは保証されます。

Redis を使用して SpringBoot に分散ロックを実装する方法 Redis を使用して SpringBoot に分散ロックを実装する方法 Jun 03, 2023 am 08:16 AM

1. Redis は分散ロックの原則を実装しており、分散ロックが必要な理由 分散ロックについて話す前に、分散ロックが必要な理由を説明する必要があります。分散ロックの反対はスタンドアロン ロックです。マルチスレッド プログラムを作成するとき、共有変数を同時に操作することによって引き起こされるデータの問題を回避します。通常、ロックを使用して共有変数を相互に除外し、データの正確性を確保します。共有変数の使用範囲は同じプロセス内です。共有リソースを同時に操作する必要があるプロセスが複数ある場合、どうすれば相互排他的になるのでしょうか?今日のビジネス アプリケーションは通常マイクロサービス アーキテクチャであり、これは 1 つのアプリケーションが複数のプロセスをデプロイすることも意味します。複数のプロセスが MySQL の同じレコード行を変更する必要がある場合、順序の乱れた操作によって引き起こされるダーティ データを避けるために、分散が必要です。今回導入するスタイルはロックされています。ポイントを獲得したい

SpringBoot が Redisson を統合して遅延キューを実装する方法 SpringBoot が Redisson を統合して遅延キューを実装する方法 May 30, 2023 pm 02:40 PM

使用シナリオ 1. 注文は正常に行われましたが、支払いが 30 分以内に行われませんでした。支払いがタイムアウトになり、注文が自動的にキャンセルされました 2. 注文に署名があり、署名後 7 日間評価が行われませんでした。注文がタイムアウトして評価されない場合、システムはデフォルトでプラスの評価を設定します 3. 注文は正常に行われます。販売者が 5 分間注文を受け取らない場合、注文はキャンセルされます。 4. 配送がタイムアウトします。 SMS リマインダーをプッシュします... 遅延が長く、リアルタイム パフォーマンスが低いシナリオでは、タスク スケジュールを使用して定期的なポーリング処理を実行できます。例: xxl-job 今日は選択します

Springbootがjarパッケージにファイルを読み込んだ後にファイルにアクセスできない問題を解決する方法 Springbootがjarパッケージにファイルを読み込んだ後にファイルにアクセスできない問題を解決する方法 Jun 03, 2023 pm 04:38 PM

Springboot はファイルを読み取りますが、jar パッケージにパッケージ化した後、最新の開発にアクセスできません。jar パッケージにパッケージ化した後、Springboot がファイルを読み取れない状況があります。その理由は、パッケージ化後、ファイルの仮想パスが変更されるためです。は無効であり、ストリーム経由でのみアクセスできます。読み取ります。ファイルはリソースの下にあります publicvoidtest(){Listnames=newArrayList();InputStreamReaderread=null;try{ClassPathResourceresource=newClassPathResource("name.txt");Input

SpringBootとSpringMVCの比較と差異分析 SpringBootとSpringMVCの比較と差異分析 Dec 29, 2023 am 11:02 AM

SpringBoot と SpringMVC はどちらも Java 開発で一般的に使用されるフレームワークですが、それらの間には明らかな違いがいくつかあります。この記事では、これら 2 つのフレームワークの機能と使用法を調べ、その違いを比較します。まず、SpringBoot について学びましょう。 SpringBoot は、Spring フレームワークに基づいたアプリケーションの作成と展開を簡素化するために、Pivo​​tal チームによって開発されました。スタンドアロンの実行可能ファイルを構築するための高速かつ軽量な方法を提供します。

入社してキャッシュとは何かを理解した 入社してキャッシュとは何かを理解した Jul 31, 2023 pm 04:03 PM

実際は次のようなものです。当時、私のリーダーは私に perf ハードウェア パフォーマンス監視タスクを与えました。perf を使用するプロセス中に、コマンド perf list を入力すると、次の情報が表示されました。私のタスクは、これらのキャッシュ イベントを有効にすることです。しかし重要なのは、これらのミスやロードが何を意味するのか全く分からないということです。

SpringBoot が Redis をカスタマイズしてキャッシュのシリアル化を実装する方法 SpringBoot が Redis をカスタマイズしてキャッシュのシリアル化を実装する方法 Jun 03, 2023 am 11:32 AM

1. RedisAPI のデフォルトのシリアル化メカニズムである RedisTemplate1.1 をカスタマイズします。API ベースの Redis キャッシュ実装では、データ キャッシュ操作に RedisTemplate テンプレートを使用します。ここで、RedisTemplate クラスを開いて、クラスのソース コード情報を表示します。publicclassRedisTemplateextendsRedisAccessorimplementsRedisOperations、BeanClassLoaderAware{//キーを宣言、値の各種シリアル化メソッド、初期値は空 @NullableprivateRedisSe

SQL ステートメントを使用せずに Springboot+Mybatis-plus を実装して複数のテーブルを追加する方法 SQL ステートメントを使用せずに Springboot+Mybatis-plus を実装して複数のテーブルを追加する方法 Jun 02, 2023 am 11:07 AM

Springboot+Mybatis-plus が SQL ステートメントを使用して複数テーブルの追加操作を実行しない場合、私が遭遇した問題は、テスト環境で思考をシミュレートすることによって分解されます: パラメーターを含む BrandDTO オブジェクトを作成し、パラメーターをバックグラウンドに渡すことをシミュレートします。 Mybatis-plus で複数テーブルの操作を実行するのは非常に難しいことを理解してください。Mybatis-plus-join などのツールを使用しない場合は、対応する Mapper.xml ファイルを設定し、臭くて長い ResultMap を設定するだけです。対応する SQL ステートメントを記述します。この方法は面倒に見えますが、柔軟性が高く、次のことが可能です。

See all articles