- jedis 与 redission 实现分布式锁
不知言愁
项目实战分布式redisjava
本文为博主原创,未经允许不得转载:1.Jedis实现分布式锁2.Redission实现分布式锁为了确保分布式锁可用,至少要保证锁的实现同时满足以下几个条件互斥性:在任意时刻只有一个客户端能持有锁不会死锁:即使有一个客户端在持有锁的期间发生崩溃而没有主动解锁,也能保证后续其它客户端能加锁容错性:只要大部分的Redis节点正常运行,客户端就可以加锁和解锁解铃还须系铃人:加锁和解锁必须是同一个客户端,客
- 注解方式优雅实现Redission
Zhaozz!
redis
1、背景实际开发过程中,一些高并发场景需要保证接口执行的一致性,通常采用加锁的方式,本地锁Reentrantlock和Synchnorized虽然可以实现但是不适用于分布式部署模式,而redis的setnx锁无法保证原子性,故而采用redission锁,它不仅适用于分布式服务还能保证原子性。2、Redisson分布式锁常规使用2.1LockgetLock获取锁,lock.lock进行加锁,会出现的
- 实现订单到期关闭
雾里有果橙
rabbitmq分布式
目录一、被动关闭二、定时任务三、JDK自带的DelayQueue四、Netty的时间轮五、Kafka的时间轮六、RocketMQ延迟消息七、RabbitMQ死信队列八、RabbitMQ插件九、Redis过期监听十、Redis的zset十一、Redission+Redis在电商、支付等系统中,一般都是先创建订单(支付单),再给用户一定的时间进行支付,如果没有按时支付的话,就需要把之前的订单(支付单)
- spring-data-redis自定义实现看门狗机制
皮卡冲撞
springredisjava看门狗
文章目录前言redission分布式锁看门狗机制简单流程图spring-data-redis实现看门狗机制指南开始引入依赖配置redis连接以及基础配置实现redis分布式锁工具类直接失败和锁重试机制实现效果图展示前言项目中使用redis分布式锁解决了点赞和楼层排序得问题,所以这里就对这个redis得分布式锁进行了学习,一般使用得是redission提供得分布式锁解决得这个问题,但是知其然更要知其
- SpringBoot整合redisson实现分布式锁
242030
springbootspringboot
SpringBoot整合redisson实现分布式锁本文主要通过SpringBoot整合redisson来实现分布式锁,并结合demo测试结果。1、pom依赖4.0.0org.springframework.bootspring-boot-starter-parent2.5.4com.examplespringboot-redission0.0.1-SNAPSHOTspringboot-redis
- 使用 sorted set 实现令牌桶限流
呦,又写BUG呢
Java分布式
业务场景为限制消息发送,要求每天不超过一次,每七天不超过三次。Redission的RRateLimiter虽然功能完备且支持自定义限流配置,但是每个限流器都需要维护三个key,并且lua脚本中的判断逻辑较为复杂。见:Redisson分布式限流器RRateLimiter的使用及原理此外,本业务场景每次固定只需要获取一个令牌,且时间等限流参数固定,因此完全可以通过一个sortedset实现令牌桶限流。
- Redis实现分布式锁
sighting_info
redis分布式数据库
注解方式实现1、使用redission实现加锁和解锁逻辑publicinterfaceDistributedLocker{RLocklock(StringlockKey);RLocklock(StringlockKey,inttimeout);RLocklock(StringlockKey,TimeUnitunit,inttimeout);booleantryLock(StringlockKey,
- Redis实战之-分布式锁-redission
Maiko Star
redisredis分布式数据库
一、分布式锁-redission功能介绍基于setnx实现的分布式锁存在下面的问题:重入问题:重入问题是指获得锁的线程可以再次进入到相同的锁的代码块中,可重入锁的意义在于防止死锁,比如HashTable这样的代码中,他的方法都是使用synchronized修饰的,假如他在一个方法内,调用另一个方法,那么此时如果是不可重入的,不就死锁了吗?所以可重入锁他的主要意义是防止死锁,我们的synchroni
- springboot+redission分布式锁注解
dn小小
javaspringjava分布式锁
packagecom.kjhy.common.core.aop;importcom.kjhy.common.core.config.RedissonConfig;importcom.kjhy.common.core.exception.Exceptions;importlombok.extern.slf4j.Slf4j;importorg.apache.commons.lang.StringUti
- spring schedule 实时更新 cron 表达式,并且立即生效。(单机,非分布式调度,无需quartz)
w1047667241
定时任务源码分析javaspring定时任务
本文的讨论,仅限于单机下的调度,不是分布式调度的管理。分布式请参考xxl-job,redission分布式锁等框架主要解决3个问题:1)@Scheduled(cron="0/5****?")注解写死后,不能更新cron 表达式;2)即使能更新,也不能立刻生效;3)事务管理失效。总共3个目标:1》quartz有点重,所以不考虑用quartz实现2》实现实时的更新cron,立刻生效;接口调用方式3》实
- Redis重点总结补充
今天的代码敲了吗
Java面试八股文redis数据库缓存
Redis重点总结1.redis分布式锁2.redission实现分布式锁注意:加锁、设置过期时间等操作都是基于lua脚本完成.redisson分布式锁,实现可重入(前提是同一个线程下3.redis主从集群实现主从复制(Master-slaveReplication)的工作原理:slave从节点服务启动并连接到Master之后,它将主动发送一个SYNC命令Maser服务主节点收到同步命令后将启动后
- Redission 分布式锁原理
花开不识君
分布式技术数据库javaredis
Redission分布式锁原理Redission是一redis客户端和jedis、lettuce一样,但他提供诸多如分布式锁这些方便的工具加锁过程lua脚本"if(redis.call('exists',KEYS[1])==0)thenredis.call('hset',KEYS[1],ARGV[2],1);redis.call('pexpire',KEYS[1],ARGV[1]);returnn
- [Redis实战]分布式锁-redission
Ja kar ta
Redisredis分布式
五、分布式锁-redission5.1分布式锁-redission功能介绍基于setnx实现的分布式锁存在下面的问题:重入问题:重入问题就是指获得锁的线程可以再次进入到相同的锁的代码中,可重入锁的意义在于防止死锁。比如HashTable这样的代码中,他的方法都是使用synchronized修饰的,假如他在一个方法内,调用另一个方法,那么此时如果是不可重入的,不就死锁了吗?所以可重入锁他的主要意义是
- springboot2.x版本集成redis说明(lettuce、redisson)
DreamBoy_W.W.Y
redisredis
总纲一、前言二、详细一、前言目前主流springboot2.x版本集成redis,主要使用lettuce和redisson。lettuce作为新式的redis客户端,基于netty采用异步非阻塞式IO,是线程安全的,优点是提供了很多redis高级功能,例如集群、哨兵、管道等,缺点是api抽象,学习成本高。lettuce好是好,但是jedis比他生得早。redission作为redis的分布式客户端
- 分布式锁Lock4J 使用总结
在奋斗的大道
Java架构专栏springBoot学习笔记深蓝计划分布式
Lok4j简介lock4j是一个分布式锁组件,其提供了多种不同的支持以满足不同性能和环境的需求。立志打造一个简单但富有内涵的分布式锁组件。特点简单易用,功能强大,扩展性强。支持redission,redisTemplate,zookeeper。可混用,支持扩展Docker安装Redis请参考文章:Docker安装RedisDocker安装ZooKeeper请参考文章:Docker安装Zookeep
- Redis关键知识点总结
半城风花半城雨
Redis面试redis数据库缓存
Reference:http://redis.cn用处缓存数据库分布式锁(Redission的redlock,自定义的lock等)过滤器(布隆过滤器/增强的带计数的布隆过滤器/布谷鸟过滤器等)大规模的计算辅助(bitmap)消息订阅/监听-->例如分布式的websocket发送消息时可用Redis消息订阅/监听将消息发到所有实例上进行推送延时队列-->例如email发送服务中同一个emailcli
- redission
小白要变大牛
笔记java
redission是什么?Redisson是一个在Redis的基础上实现的Java驻内存数据网格(In-MemoryDataGrid)。它不仅提供了一系列的分布式的Java常用对象,还提供了许多分布式服务Redisson提供了使用Redis的最简单和最便捷的方法。Redisson的宗旨是促进使用者对Redis的关注分离(SeparationofConcern),从而让使用者能够将精力更集中地放在处
- 扒开Redisson的小棉袄,Debug深入剖析分布式锁之可重入锁No.1
逆熵架构
上次搭建好三主三从的rediscluster集群之后,也试了试redission的加锁解锁的API,那么redission是怎么实现分布式锁的呢?我们就开始对这几行关键的代码进行分析,打好断点,debug调试,是分析源码,学习源码的一个好的方法,OK,让我们开始一场浪漫绚丽的源码探索之旅,redisson的源码写的很漂亮哦。首先,我们主要去关注加锁的过程,那么对于redisson创建,通信等一些涉
- 一个功能强大的分布式锁框架:Lock4j
String、code
分布式
一个功能强大的分布式锁框架:Lock4j1简介Lock4j是一个分布式锁组件,它提供了多种不同的支持以满足不同性能和环境的需求,基于SpringAOP的声明式和编程式分布式锁,支持RedisTemplate、Redisson、Zookeeper。2特性简单易用,功能强大,扩展性强。支持redission,redisTemplate,zookeeper,可混用,支持扩展。**Gitee地址:http
- Redis(十) - Redission原理与实践
Super_Leng
Redisredisjava数据库
文章目录一、Redission分布式锁原理1.Redission介绍2.Redission基本使用(1)引入依赖(2)配置Redisson客户端(3)使用Redission的分布式锁3.Redission可重入锁原理(1)可重入锁流程分析(2)演示可重入锁(3)Redisson源码分析4.Redisson可重试原理5.Redisson的WatchDog机制6.小结:Redisson分布式锁原理二、
- 大佬浅谈分布式锁
阿斌在路上
redis实现redis分布锁一、redis实现分布式锁(可重入锁)redission实现分布式锁1、对锁要有时间超时处理2、锁可重入锁(重入设置时间)redis.setnx(key,value);实现redis.expire(key,timeout);二、ZooKeeper实现锁(是一个分布式的,开放源码的分布式应用程序协调服务)三、数据库实现分布式锁下面是分布式锁逻辑图image.png
- Spring-Redis
littlepants718
springredisbootstrap
什么是RedisRedis是一种基于键值对(key-value)的NoSQL数据库什么是Spring-redis为了方便在spring工程中使用redis,创建的依赖包配置Redis连接为了操作Redis,我们需要使用Java的Redis客户端,下面是Redis官方推荐的三种客户端IO方式线程安全Jedis阻塞式否Lettuce非阻塞是Redission非阻塞是Redis各个客户端的操作差别很大,
- Redission分布式锁原理初探
zhangyifang_009
技术学习分布式javaredis
什么是分布式锁,为什么需要分布式锁在多线程并发请求当中,为了保证我们的资源同一时刻只有一个线程进行操作(如商品超卖问题、购票系统等),我们通常要添加锁机制,如ReentrantLock,也就是可重入的互斥锁,与synchronized功能类似,因为比较灵活,所以经常使用。这在单机情况下是没有问题的,但在多节点的情况下,也就意味着有多个进程,ReentrantLock锁机制可能就会不起作用,所以我们
- 基于redisson实现发布订阅(多服务间用避坑)
肥仔哥哥1930
高效开发分享业务设计思路第三方对接redisson信息发布订阅redisson信息消费不到redisson信息监听不到信息多服务间监听不到
前言今天要分享的是基于Redisson实现信息发布与订阅(以前分享过直接基于redis的实现),如果你是在多服务间基于redisson做信息传递,并且有服务压根就收不到信息,那你一定要看完。今天其实重点是避坑,真正的集成使用就几步。一、redission介绍介绍的文字我都懒得写,其实要我写详细,我也是google,下面直接贴图吧介绍的挺详细的吧,下面还有代码示例哦,不得不说这个GPT插件挺好用的。
- Redis分布式锁(三种实现方式:setnx+expire,set,Redission)
levitgu
redis分布式数据库微服务云原生缓存后端
Redis分布式锁分布式锁的由来在传统单机部署的情况下,可以使用Java并发处理相关的API(如ReentrantLcok或synchronized)进行互斥控制。但是在分布式系统后,由于分布式系统多线程、多进程并且分布在不同机器上,这将使原单机并发控制锁策略失效,为了解决这个问题就需要一种跨JVM的互斥机制来控制共享资源的访问,这就是分布式锁的由来。当多个进程不在同一个系统中,就需要用分布式锁控
- redis cluster配置之read-mode
yangyuscript
redisredis缓存分布式
背景生产部署了redis集群,三台机器(三主三从,主从不在同一台机器上),redission连接使用。当有一个master节点挂掉时,redis整个集群不可用。解决过程运维登上机器上,执行clusterinfo发现集群OK状态执行clusternodes发现挂掉的master节点fail状态,对应的slave已经变成了master节点(failover成功)手动连接上集群,redis-cli-h-
- Redission从入门到入门
来自宇宙的曹先生
Redission分布式锁
1.Redisson简介Redisson是一个在Java环境中使用的Redis客户端库。它提供了丰富的功能,使得在Java应用中与Redis交互变得更加简单和高效。Redisson不仅提供了基本的Redis操作,还提供了许多高级功能,使其成为在Java项目中实现分布式和可扩展解决方案的强大工具。以下是Redisson的一些主要特点和功能:1.易于使用的Java对象接口Redisson提供了一系列的
- 分布式
猛浩
java开发语言
建议使用Redission@ResourceprivateRedissonClientredissonClient;RLocklock=redissonClient.getLock(lockKey);try{if(!lock.tryLock()){log.warn(".....{}",lockKey);returnfalse;}...}catch(Exceptione){...}finally{f
- redission源码解读
wppwpp1
javaRedisjava
可以参考这篇文章,讲的比较详细https://www.cnblogs.com/throwable/p/14264804.htmlredlock算法的介绍Redis分布式锁之Redlock算法,那些你可能不知道的秘密!-掘金jedis和redission的区别?Jedis和Redisson都是Java语言中常用的Redis客户端库,它们之间的区别如下:线程模型不同:Jedis是单线程模型,而Redi
- 黑马点评笔记 分布式锁
过去日记
项目回顾笔记分布式redisjava后端
文章目录分布式锁基本原理和实现方式对比Redis分布式锁的实现核心思路实现分布式锁版本一Redis分布式锁误删情况说明解决Redis分布式锁误删问题分布式锁的原子性问题分布式锁-Redission分布式锁-redission可重入锁原理分布式锁-redission锁重试和WatchDog机制分布式锁-redission锁的MutiLock原理分布式锁基本原理和实现方式对比分布式锁:满足分布式系统或
- 基本数据类型和引用类型的初始值
3213213333332132
java基础
package com.array;
/**
* @Description 测试初始值
* @author FuJianyong
* 2015-1-22上午10:31:53
*/
public class ArrayTest {
ArrayTest at;
String str;
byte bt;
short s;
int i;
long
- 摘抄笔记--《编写高质量代码:改善Java程序的151个建议》
白糖_
高质量代码
记得3年前刚到公司,同桌同事见我无事可做就借我看《编写高质量代码:改善Java程序的151个建议》这本书,当时看了几页没上心就没研究了。到上个月在公司偶然看到,于是乎又找来看看,我的天,真是非常多的干货,对于我这种静不下心的人真是帮助莫大呀。
看完整本书,也记了不少笔记
- 【备忘】Django 常用命令及最佳实践
dongwei_6688
django
注意:本文基于 Django 1.8.2 版本
生成数据库迁移脚本(python 脚本)
python manage.py makemigrations polls
说明:polls 是你的应用名字,运行该命令时需要根据你的应用名字进行调整
查看该次迁移需要执行的 SQL 语句(只查看语句,并不应用到数据库上):
python manage.p
- 阶乘算法之一N! 末尾有多少个零
周凡杨
java算法阶乘面试效率
&n
- spring注入servlet
g21121
Spring注入
传统的配置方法是无法将bean或属性直接注入到servlet中的,配置代理servlet亦比较麻烦,这里其实有比较简单的方法,其实就是在servlet的init()方法中加入要注入的内容:
ServletContext application = getServletContext();
WebApplicationContext wac = WebApplicationContextUtil
- Jenkins 命令行操作说明文档
510888780
centos
假设Jenkins的URL为http://22.11.140.38:9080/jenkins/
基本的格式为
java
基本的格式为
java -jar jenkins-cli.jar [-s JENKINS_URL] command [options][args]
下面具体介绍各个命令的作用及基本使用方法
1. &nb
- UnicodeBlock检测中文用法
布衣凌宇
UnicodeBlock
/** * 判断输入的是汉字 */ public static boolean isChinese(char c) { Character.UnicodeBlock ub = Character.UnicodeBlock.of(c);
- java下实现调用oracle的存储过程和函数
aijuans
javaorale
1.创建表:STOCK_PRICES
2.插入测试数据:
3.建立一个返回游标:
PKG_PUB_UTILS
4.创建和存储过程:P_GET_PRICE
5.创建函数:
6.JAVA调用存储过程返回结果集
JDBCoracle10G_INVO
- Velocity Toolbox
antlove
模板toolboxvelocity
velocity.VelocityUtil
package velocity;
import org.apache.velocity.Template;
import org.apache.velocity.app.Velocity;
import org.apache.velocity.app.VelocityEngine;
import org.apache.velocity.c
- JAVA正则表达式匹配基础
百合不是茶
java正则表达式的匹配
正则表达式;提高程序的性能,简化代码,提高代码的可读性,简化对字符串的操作
正则表达式的用途;
字符串的匹配
字符串的分割
字符串的查找
字符串的替换
正则表达式的验证语法
[a] //[]表示这个字符只出现一次 ,[a] 表示a只出现一
- 是否使用EL表达式的配置
bijian1013
jspweb.xmlELEasyTemplate
今天在开发过程中发现一个细节问题,由于前端采用EasyTemplate模板方法实现数据展示,但老是不能正常显示出来。后来发现竟是EL将我的EasyTemplate的${...}解释执行了,导致我的模板不能正常展示后台数据。
网
- 精通Oracle10编程SQL(1-3)PLSQL基础
bijian1013
oracle数据库plsql
--只包含执行部分的PL/SQL块
--set serveroutput off
begin
dbms_output.put_line('Hello,everyone!');
end;
select * from emp;
--包含定义部分和执行部分的PL/SQL块
declare
v_ename varchar2(5);
begin
select
- 【Nginx三】Nginx作为反向代理服务器
bit1129
nginx
Nginx一个常用的功能是作为代理服务器。代理服务器通常完成如下的功能:
接受客户端请求
将请求转发给被代理的服务器
从被代理的服务器获得响应结果
把响应结果返回给客户端
实例
本文把Nginx配置成一个简单的代理服务器
对于静态的html和图片,直接从Nginx获取
对于动态的页面,例如JSP或者Servlet,Nginx则将请求转发给Res
- Plugin execution not covered by lifecycle configuration: org.apache.maven.plugin
blackproof
maven报错
转:http://stackoverflow.com/questions/6352208/how-to-solve-plugin-execution-not-covered-by-lifecycle-configuration-for-sprin
maven报错:
Plugin execution not covered by lifecycle configuration:
- 发布docker程序到marathon
ronin47
docker 发布应用
1 发布docker程序到marathon 1.1 搭建私有docker registry 1.1.1 安装docker regisry
docker pull docker-registry
docker run -t -p 5000:5000 docker-registry
下载docker镜像并发布到私有registry
docker pull consol/tomcat-8.0
- java-57-用两个栈实现队列&&用两个队列实现一个栈
bylijinnan
java
import java.util.ArrayList;
import java.util.List;
import java.util.Stack;
/*
* Q 57 用两个栈实现队列
*/
public class QueueImplementByTwoStacks {
private Stack<Integer> stack1;
pr
- Nginx配置性能优化
cfyme
nginx
转载地址:http://blog.csdn.net/xifeijian/article/details/20956605
大多数的Nginx安装指南告诉你如下基础知识——通过apt-get安装,修改这里或那里的几行配置,好了,你已经有了一个Web服务器了。而且,在大多数情况下,一个常规安装的nginx对你的网站来说已经能很好地工作了。然而,如果你真的想挤压出Nginx的性能,你必
- [JAVA图形图像]JAVA体系需要稳扎稳打,逐步推进图像图形处理技术
comsci
java
对图形图像进行精确处理,需要大量的数学工具,即使是从底层硬件模拟层开始设计,也离不开大量的数学工具包,因为我认为,JAVA语言体系在图形图像处理模块上面的研发工作,需要从开发一些基础的,类似实时数学函数构造器和解析器的软件包入手,而不是急于利用第三方代码工具来实现一个不严格的图形图像处理软件......
&nb
- MonkeyRunner的使用
dai_lm
androidMonkeyRunner
要使用MonkeyRunner,就要学习使用Python,哎
先抄一段官方doc里的代码
作用是启动一个程序(应该是启动程序默认的Activity),然后按MENU键,并截屏
# Imports the monkeyrunner modules used by this program
from com.android.monkeyrunner import MonkeyRun
- Hadoop-- 海量文件的分布式计算处理方案
datamachine
mapreducehadoop分布式计算
csdn的一个关于hadoop的分布式处理方案,存档。
原帖:http://blog.csdn.net/calvinxiu/article/details/1506112。
Hadoop 是Google MapReduce的一个Java实现。MapReduce是一种简化的分布式编程模式,让程序自动分布到一个由普通机器组成的超大集群上并发执行。就如同ja
- 以資料庫驗證登入
dcj3sjt126com
yii
以資料庫驗證登入
由於 Yii 內定的原始框架程式, 採用綁定在UserIdentity.php 的 demo 與 admin 帳號密碼: public function authenticate() { $users=array( &nbs
- github做webhooks:[2]php版本自动触发更新
dcj3sjt126com
githubgitwebhooks
上次已经说过了如何在github控制面板做查看url的返回信息了。这次就到了直接贴钩子代码的时候了。
工具/原料
git
github
方法/步骤
在github的setting里面的webhooks里把我们的url地址填进去。
钩子更新的代码如下: error_reportin
- Eos开发常用表达式
蕃薯耀
Eos开发Eos入门Eos开发常用表达式
Eos开发常用表达式
>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
蕃薯耀 2014年8月18日 15:03:35 星期一
&
- SpringSecurity3.X--SpEL 表达式
hanqunfeng
SpringSecurity
使用 Spring 表达式语言配置访问控制,要实现这一功能的直接方式是在<http>配置元素上添加 use-expressions 属性:
<http auto-config="true" use-expressions="true">
这样就会在投票器中自动增加一个投票器:org.springframework
- Redis vs Memcache
IXHONG
redis
1. Redis中,并不是所有的数据都一直存储在内存中的,这是和Memcached相比一个最大的区别。
2. Redis不仅仅支持简单的k/v类型的数据,同时还提供list,set,hash等数据结构的存储。
3. Redis支持数据的备份,即master-slave模式的数据备份。
4. Redis支持数据的持久化,可以将内存中的数据保持在磁盘中,重启的时候可以再次加载进行使用。
Red
- Python - 装饰器使用过程中的误区解读
kvhur
JavaScriptjqueryhtml5css
大家都知道装饰器是一个很著名的设计模式,经常被用于AOP(面向切面编程)的场景,较为经典的有插入日志,性能测试,事务处理,Web权限校验, Cache等。
原文链接:http://www.gbtags.com/gb/share/5563.htm
Python语言本身提供了装饰器语法(@),典型的装饰器实现如下:
@function_wrapper
de
- 架构师之mybatis-----update 带case when 针对多种情况更新
nannan408
case when
1.前言.
如题.
2. 代码.
<update id="batchUpdate" parameterType="java.util.List">
<foreach collection="list" item="list" index=&
- Algorithm算法视频教程
栏目记者
Algorithm算法
课程:Algorithm算法视频教程
百度网盘下载地址: http://pan.baidu.com/s/1qWFjjQW 密码: 2mji
程序写的好不好,还得看算法屌不屌!Algorithm算法博大精深。
一、课程内容:
课时1、算法的基本概念 + Sequential search
课时2、Binary search
课时3、Hash table
课时4、Algor
- C语言算法之冒泡排序
qiufeihu
c算法
任意输入10个数字由小到大进行排序。
代码:
#include <stdio.h>
int main()
{
int i,j,t,a[11]; /*定义变量及数组为基本类型*/
for(i = 1;i < 11;i++){
scanf("%d",&a[i]); /*从键盘中输入10个数*/
}
for
- JSP异常处理
wyzuomumu
Webjsp
1.在可能发生异常的网页中通过指令将HTTP请求转发给另一个专门处理异常的网页中:
<%@ page errorPage="errors.jsp"%>
2.在处理异常的网页中做如下声明:
errors.jsp:
<%@ page isErrorPage="true"%>,这样设置完后就可以在网页中直接访问exc