怎样用此列表来排序彼列表 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 private List<TagGroup> reorder (Long id, List<TagGroup> entities) { final AccountSettings accountSettings = getAccountSetting( id); final List<Long> tagGroupOrders = accountSettings.getTagGroupOrders(); if (CollectionUtils.isNotEmpty(tagGroupOrders)) { entities.sort((o1, o2) -> { int index1 = tagGroupOrders.indexOf(o1.getTagGroupId()); int index2 = tagGroupOrders.indexOf(o2.getTagGroupId()); if (index1 == -1 || index2 == -1 ) { return sortByModifiedTime(o1, o2); } return index1 - index2; }); } return entities; }private static int sortByModifiedTime (TagGroup o1, TagGroup o2) { final long time1 = o1.getLastModifiedTime().getTime(); final long time2 = o2.getLastModifiedTime().getTime(); if (time1 < time2) { return -1 ; } else if (time1 > time2) { return 1 ; } else { return 0 ; } }
怎样每次生成一把锁实例 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 @Slf4j @Component @Scope("prototype") public class LockRunner { private RedissonClient redissonClient; @Autowired public LockRunner (RedissonClient redissonClient) { this .redissonClient = redissonClient; } @WithSpan public void runInLock (@SpanAttribute("lockConfig") LockKey lockKeyConfig, Runnable runnable) { final String lockKey = lockKeyConfig.getLockKey(); RLock lock = redissonClient.getLock(lockKey); lock.lock(); try { runnable.run(); } finally { unlock(lock); } } @WithSpan public <V> V callInLock (@SpanAttribute("lockConfig") LockKey lockKeyConfig, Callable<V> callable) { final String lockKey = lockKeyConfig.getLockKey(); RLock lock = redissonClient.getLock(lockKey); lock.lock(); try { return callable.call(); } catch (Exception e) { throw new SystemInternalException ("callable.call()失败:" + e.getMessage(), e); } finally { unlock(lock); } } private void unlock (RLock lock) { log.info("unlock for: {}" , lock.getName()); lock.unlock(); } }@Component public class LockRunnerFactory { private final ApplicationContext applicationContext; @Autowired public LockRunnerFactory (ApplicationContext applicationContext) { this .applicationContext = applicationContext; } public LockRunner createLockRunner () { return applicationContext.getBean(LockRunner.class); } }
一个更好的设计是去掉@Scope("prototype")
和LockRunnerFactory
,LockRunner
的可变状态都被参数化了,所有最好的设计实际上是:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 @Data public class LockConfig { public static final String LOCK_SUFFIX = "-lock" ; private final int expireSecs; private final String lockKey; private LockConfig (final int expireSecs, final String lockKey) { this .expireSecs = expireSecs; this .lockKey = lockKey; } public LockConfig (final int expireSecs, final Object... objects) { this (expireSecs, Joiner.on("-" ).join(objects) + LOCK_SUFFIX); } }@Data public class LockKey { public static final String LOCK_SUFFIX = "-lock" ; private final String lockKey; public LockKey (final Object... objects) { lockKey = Joiner.on("-" ).join(objects) + LOCK_SUFFIX; } }
在还原一个字段的时候,怎么确定它有没有被修改过? 一个字段是 ownerId。它曾经是1,无意之中被设置为0,能不能恢复为1? 假设有另一个字段为 ownerName,有意设置 ownerId 的时候,一定会清空 ownerName。则可以做这样的比对:
1 2 3 if (owner == 0 && StringUtils.isNotBlank(ownerName)) { owner = 1 ; }
使用一个特定参数来排除某个 jar 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 def jars = fileTree(dir: 'addtionnallib' , include: ['*.jar' ]) jars.each { File file -> println("jars:${file}" ) } ext.excludeHighJar = false ext.excludeHighJar = project.hasProperty('excludeHighJar' ) ? project.getProperty('excludeHighJar' ) : false if (excludeHighJar && excludeHighJar.trim()) { jars = jars.minus(file("addtionnallib/a.jar" )) } else { jars = jars.minus(file("addtionnallib/a.fat.jar" )) } api files(jars) println "Excluding jar: ${excludeHighJar}"
把多重 group by 转成单重 group by 老的group by a, b, day
,实际产出是一个以 day 聚合的曲线图。 a、b出现在这里本义是为了按照 a, b 聚合,然后选出特定的 a、b 组合,这实际上是一种叉乘关系。 所以如果我们能转化 a、b 到where 里,我们就能直接group by day
。
基于函数来管理 go 版本(适用于 Linux) 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 go_version_manage () { case $1 in install) __go_install_version "$2 " ;; switch) __go_switch_version "$2 " ;; list) __go_list_versions ;; *) echo "Usage: go_version_manage [install|switch|list] <version>" return 1 ;; esac }__go_install_version () { local version=$1 if ! [[ "$version " =~ ^[0-9]+\.[0-9]+(\.[0-9]+)?$ ]]; then echo "Invalid version format. Expected format: 1.22.0 or 1.22" return 1 fi local install_dir="/opt/go-versions/go$version " local tar_file="go$version .linux-amd64.tar.gz" local url="https://go.dev/dl/$tar_file " sudo mkdir -p "$install_dir " || return 1 if ! (wget -qO- "$url " | sudo tar -xz -C "$install_dir " --strip-components 1); then echo "Failed to install Go $version " sudo rm -rf "$install_dir " return 1 fi echo "Successfully installed Go $version in $install_dir " if [ -n "$ZSH_VERSION " ]; then rehash elif [ -n "$BASH_VERSION " ]; then hash -r fi }__go_switch_version () { local version=$1 local go_path="/opt/go-versions/go$version " if [ -d "$go_path " ]; then export GOROOT="$go_path " export PATH="$GOROOT /bin:$PATH " echo "Switched to Go $version " go version else echo "Go $version not installed. Available versions:" __go_list_versions return 1 fi }__go_list_versions () { echo "Installed Go versions:" ls -1 /opt/go-versions | sed 's/go//' | sort -V }_go_version_manage () { local cur=${words[CURRENT]} case ${words[2]} in switch) if [ -d /opt/go-versions ]; then COMPREPLY=($(ls /opt/go-versions | grep '^go' | cut -c3- | grep "^$cur " )) fi ;; install) local remote_versions=$(curl -s https://go.dev/VERSION?m=text | grep -Eo 'go[0-9]+\.[0-9]+(\.[0-9]+)?' | sed 's/go//' | sort -Vr | uniq ) COMPREPLY=($(compgen -W "$remote_versions " -- "$cur " )) ;; *) COMPREPLY=($(compgen -W "install switch list" -- "$cur " )) ;; esac }
范式是为了平衡读和写 写多的时候,人们就不喜欢冗余列,因为这会增加同步的难度,而且扩大事务的范围。 读多的时候,大家就不喜欢不符合第一范式的列,因为这会导致列内部不可查询。
OLAP 数仓查询应用倾向于多冗余,但是加字段和回溯数据比较麻烦。
使用移动来代替删除 使用移动来代替恢复。