前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >如何优雅地根治null值引起的Bug!

如何优雅地根治null值引起的Bug!

作者头像
Bug开发工程师
发布于 2020-03-12 10:29:06
发布于 2020-03-12 10:29:06
92400
代码可运行
举报
文章被收录于专栏:码农沉思录码农沉思录
运行总次数:0
代码可运行

写在前面

在笔者几年的开发经验中,经常看到项目中存在到处空值判断的情况,这些判断,会让人觉得摸不着头绪,它的出现很有可能和当前的业务逻辑并没有关系。但它会让你很头疼。

有时候,更可怕的是系统因为这些空值的情况,会抛出空指针异常,导致业务系统发生问题。

此篇文章,总结了几种关于空值的处理手法,希望对读者有帮助。

业务中的空值

场景

存在一个 UserSearchService用来提供用户查询的功能:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserSearchService{
List<User> listUser();
Userget(Integer id);
}

问题现场

对于面向对象语言来讲,抽象层级特别的重要。尤其是对接口的抽象,它在设计和开发中占很大的比重,我们在开发时希望尽量面向接口编程。

对于以上描述的接口方法来看,大概可以推断出可能它包含了以下两个含义:

  • listUser(): 查询用户列表
  • get(Integerid): 查询单个用户

在所有的开发中,XP推崇的TDD模式可以很好的引导我们对接口的定义,所以我们将TDD作为开发代码的”推动者”。

对于以上的接口,当我们使用TDD进行测试用例先行时,发现了潜在的问题:

  • listUser() 如果没有数据,那它是返回空集合还是null呢?
  • get(Integerid) 如果没有这个对象,是抛异常还是返回null呢?

深入listUser研究

我们先来讨论

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
listUser()

这个接口,经常看到如下实现:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicList<User> listUser(){
List<User> userList = userListRepostity.selectByExample(newUserExample());
if(CollectionUtils.isEmpty(userList)){//spring util工具类
returnnull;
}
return userList;
}

这段代码返回是null,从我多年的开发经验来讲,对于集合这样返回值,最好不要返回null,因为如果返回了null,会给调用者带来很多麻烦。你将会把这种调用风险交给调用者来控制。

如果调用者是一个谨慎的人,他会进行是否为null的条件判断。如果他并非谨慎,或者他是一个面向接口编程的狂热分子(当然,面向接口编程是正确的方向),他会按照自己的理解去调用接口,而不进行是否为null的条件判断,如果这样的话,是非常危险的,它很有可能出现空指针异常!

基于此,我们将它进行优化:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicList<User> listUser(){
List<User> userList = userListRepostity.selectByExample(newUserExample());
if(CollectionUtils.isEmpty(userList)){
returnLists.newArrayList(); //guava类库提供的方式
}
return userList;
}

对于接口( ListlistUser()),它一定会返回List,即使没有数据,它仍然会返回List(集合中没有任何元素);

通过以上的修改,我们成功的避免了有可能发生的空指针异常,这样的写法更安全!

深入研究get方法

对于接口

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Userget(Integer id)

你能看到的现象是,我给出id,它一定会给我返回User.但事实真的很有可能不是这样的。

我看到过的实现:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicUserget(Integer id){
return userRepository.selectByPrimaryKey(id);//从数据库中通过id直接获取实体对象
}

相信很多人也都会这样写。

通过代码的时候得知它的返回值很有可能是null! 但我们通过的接口是分辨不出来的!

这个是个非常危险的事情。尤其对于调用者来说!

我给出的建议是,需要在接口明明时补充文档,比如对于异常的说明,使用注解@exception:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserSearchService{

/**
   * 根据用户id获取用户信息
   * @param id 用户id
   * @return 用户实体
   * @exception UserNotFoundException
   */
Userget(Integer id);

}

我们把接口定义加上了说明之后,调用者会看到,如果调用此接口,很有可能抛出“UserNotFoundException(找不到用户)”这样的异常。

这种方式可以在调用者调用接口的时候看到接口的定义,但是,这种方式是”弱提示”的!

如果调用者忽略了注释,有可能就对业务系统产生了风险,这个风险有可能导致一个亿!

除了以上这种”弱提示”的方式,还有一种方式是,返回值是有可能为空的。那要怎么办呢?

我认为我们需要增加一个接口,用来描述这种场景.

引入jdk8的Optional,或者使用guava的Optional.看如下定义:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserSearchService{

/**
   * 根据用户id获取用户信息
   * @param id 用户id
   * @return 用户实体,此实体有可能是缺省值
   */
Optional<User> getOptional(Integer id);
}

Optional有两个含义: 存在 or 缺省。

那么通过阅读接口getOptional(),我们可以很快的了解返回值的意图,这个其实是我们想看到的,它去除了二义性。

它的实现可以写成:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicOptional<User> getOptional(Integer id){
returnOptional.ofNullable(userRepository.selectByPrimaryKey(id));
}

深入入参

通过上述的所有接口的描述,你能确定入参id一定是必传的吗?我觉得答案应该是:不能确定。除非接口的文档注释上加以说明。

那如何约束入参呢?

推荐两种方式:

  • 强制约束
  • 文档性约束(弱提示)

1.强制约束,我们可以通过jsr 303进行严格的约束声明:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserSearchService{
/**
   * 根据用户id获取用户信息
   * @param id 用户id
   * @return 用户实体
   * @exception UserNotFoundException
   */
Userget(@NotNullInteger id);

/**
   * 根据用户id获取用户信息
   * @param id 用户id
   * @return 用户实体,此实体有可能是缺省值
   */
Optional<User> getOptional(@NotNullInteger id);
}

当然,这样写,要配合AOP的操作进行验证,但让spring已经提供了很好的集成方案,在此就不在赘述了。

2.文档性约束

在很多时候,我们会遇到遗留代码,对于遗留代码,整体性改造的可能性很小。

我们更希望通过阅读接口的实现,来进行接口的说明。

jsr 305规范,给了我们一个描述接口入参的一个方式(需要引入库 com.google.code.findbugs:jsr305):

可以使用注解: @Nullable @Nonnull @CheckForNull 进行接口说明。比如:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserSearchService{
/**
   * 根据用户id获取用户信息
   * @param id 用户id
   * @return 用户实体
   * @exception UserNotFoundException
   */
@CheckForNull
Userget(@NonNullInteger id);

/**
   * 根据用户id获取用户信息
   * @param id 用户id
   * @return 用户实体,此实体有可能是缺省值
   */
Optional<User> getOptional(@NonNullInteger id);
}

小结

通过 空集合返回值,Optional,jsr 303,jsr 305这几种方式,可以让我们的代码可读性更强,出错率更低!

  • 空集合返回值 :如果有集合这样返回值时,除非真的有说服自己的理由,否则,一定要返回空集合,而不是null
  • Optional: 如果你的代码是jdk8,就引入它!如果不是,则使用Guava的Optional,或者升级jdk版本!它很大程度的能增加了接口的可读性!
  • jsr 303: 如果新的项目正在开发,不防加上这个试试!一定有一种特别爽的感觉!
  • jsr 305: 如果老的项目在你的手上,你可以尝试的加上这种文档型注解,有助于你后期的重构,或者新功能增加了,对于老接口的理解!

空对象模式

场景

来看一个DTO转化的场景,对象:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@Data
staticclassPersonDTO{
privateString dtoName;
privateString dtoAge;
}

@Data
staticclassPerson{
privateString name;
privateString age;
}

需求是将Person对象转化成PersonDTO,然后进行返回。

当然对于实际操作来讲,返回如果Person为空,将返回null,但是PersonDTO是不能返回null的(尤其Rest接口返回的这种DTO)。

在这里,我们只关注转化操作,看如下代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@Test
publicvoid shouldConvertDTO(){

PersonDTO personDTO = newPersonDTO();

Person person = newPerson();
if(!Objects.isNull(person)){
    personDTO.setDtoAge(person.getAge());
    personDTO.setDtoName(person.getName());
}else{
    personDTO.setDtoAge("");
    personDTO.setDtoName("");
}
}

优化修改

这样的数据转化,可读性非常差,每个字段的判断,如果是空就设置为空字符串(“”)

换一种思维方式进行思考,我们是拿到Person这个类的数据,然后进行赋值操作(setXXX),其实是不关系Person的具体实现是谁的。

那我们可以创建一个Person子类:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
staticclassNullPersonextendsPerson{
@Override
publicString getAge() {
return"";
}

@Override
publicString getName() {
return"";
}
}

它作为Person的一种特例而存在,如果当Person为空的时候,则返回一些 get*的默认行为.

所以代码可以修改为:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@Test
publicvoid shouldConvertDTO(){

PersonDTO personDTO = newPersonDTO();

Person person = getPerson();
   personDTO.setDtoAge(person.getAge());
   personDTO.setDtoName(person.getName());
}

privatePerson getPerson(){
returnnewNullPerson();   //如果Person是null ,则返回空对象
}

其中 getPerson()方法,可以用来根据业务逻辑获取Person有可能的对象(对当前例子来讲,如果Person不存在,返回Person的的特例NUllPerson),如果修改成这样,代码的可读性就会变的很强了。

使用Optional可以进行优化

空对象模式,它的弊端在于需要创建一个特例对象,但是如果特例的情况比较多,我们是不是需要创建多个特例对象呢,虽然我们也使用了面向对象的多态特性,但是,业务的复杂性如果真的让我们创建多个特例对象,我们还是要再三考虑一下这种模式,它可能会带来代码的复杂性。

对于上述代码,还可以使用Optional进行优化。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@Test
publicvoid shouldConvertDTO(){

PersonDTO personDTO = newPersonDTO();

Optional.ofNullable(getPerson()).ifPresent(person -> {
      personDTO.setDtoAge(person.getAge());
      personDTO.setDtoName(person.getName());
});
}

privatePerson getPerson(){
returnnull;
}

Optional对空值的使用,我觉得更为贴切,它只适用于”是否存在”的场景。

如果只对控制的存在判断,我建议使用Optional。

Optioanl的正确使用

Optional如此强大,它表达了计算机最原始的特性(0 or 1),那它如何正确的被使用呢!

Optional不要作为参数

如果你写了一个public方法,这个方法规定了一些输入参数,这些参数中有一些是可以传入null的,那这时候是否可以使用Optional呢?

给的建议是: 一定不要这样使用!

举个例子:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserService{
List<User> listUser(Optional<String> username);
}

这个例子的方法 listUser,可能在告诉我们需要根据username查询所有数据集合,如果username是空,也要返回所有的用户集合.

当我们看到这个方法的时候,会觉得有一些歧义:

“如果username是absent,是返回空集合吗?还是返回全部的用户数据集合?”

Optioanl是一种分支的判断,那我们究竟是关注 Optional还是Optional.get()呢?

给大家的建议是,如果不想要这样的歧义,就不要使用它!

如果你真的想表达两个含义,就給它拆分出两个接口:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserService{
List<User> listUser(String username);
List<User> listUser();
}

我觉得这样的语义更强,并且更能满足 软件设计原则中的 “单一职责”。

如果你觉得你的入参真的有必要可能传null,那请使用jsr 303或者jsr 305进行说明和验证!

请记住! Optional不能作为入参的参数!

Optinal作为返回值

当个实体的返回

那Optioanl可以做为返回值吗?

其实它是非常满足是否存在这个语义的。

你如说,你要根据id获取用户信息,这个用户有可能存在或者不存在。

你可以这样使用:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserService{
Optional<User> get(Integer id);
}

当调用这个方法的时候,调用者很清楚get方法返回的数据,有可能不存在,这样可以做一些更合理的判断,更好的防止空指针的错误!

当然,如果业务方真的需要根据id必须查询出User的话,就不要这样使用了,请说明,你要抛出的异常.

只有当考虑它返回null是合理的情况下,才进行Optional的返回

集合实体的返回

不是所有的返回值都可以这样用的!如果你返回的是集合:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
publicinterfaceUserService{
Optional<List<User>> listUser();
}

这样的返回结果,会让调用者不知所措,是否我判断Optional之后,还用进行isEmpty的判断呢?

这样带来的返回值歧义!我认为是没有必要的。

我们要约定,对于List这种集合返回值,如果集合真的是null的,请返回空集合(Lists.newArrayList);

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Optional<User> userOpt = ...

如果有这样的变量userOpt,请记住 :

  • 一定不能直接使用get ,如果这样用,就丧失了Optional本身的含义 ( 比如userOp.get() )
  • 不要直接使用getOrThrow ,如果你有这样的需求:获取不到就抛异常。那就要考虑,是否是调用的接口设计的是否合理

getter中的使用

对于一个java bean,所有的属性都有可能返回null,那是否需要改写所有的getter成为Optional类型呢?

给大家的建议是,不要这样滥用Optional.

即便 我java bean中的getter是符合Optional的,但是因为java bean 太多了,这样会导致你的代码有50%以上进行Optinal的判断,这样便污染了代码。(我想说,其实你的实体中的字段应该都是由业务含义的,会认真的思考过它存在的价值的,不能因为Optional的存在而滥用)

我们应该更关注于业务,而不只是空值的判断。

不要在getter中滥用Optional.

小结

可以这样总结Optional的使用:

  • 当使用值为空的情况,并非源于错误时,可以使用Optional!
  • Optional不要用于集合操作!
  • 不要滥用Optional,比如在java bean的getter中!
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
点个在看吧,证明你还爱我
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-12-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 码农沉思录 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
见微知著:细粒度图像分析进展
作者简介:魏秀参,南京大学计算机系机器学习与数据挖掘所(LAMDA)博士生,专攻计算机视觉和机器学习。曾在国际顶级期刊和会议发表多篇学术论文,并两次获得国际计算机视觉相关竞赛冠亚军。 责编:何永灿,欢迎人工智能领域技术投稿、约稿、给文章纠错,请发送邮件至heyc@csdn.net 本文为《程序员》原创文章,未经允许不得转载,更多精彩文章请订阅2017年《程序员》 有别于通用图像分析任务,细粒度图像分析的所属类别和粒度更为精细,它不仅能在更细分的类别下对物体进行识别,就连相似度极高的同一物种也能区别开来。
用户1737318
2018/06/06
1.6K0
细粒度图像分析_图像分类研究现状
Awesome Fine-grained Visual Classification Awesome Fine-Grained Image Analysis – Papers, Codes and Datasets—-weixiushen
全栈程序员站长
2022/11/09
1.3K0
细粒度图像分析_图像分类研究现状
资源 | 最新的细粒度图像分析资源
图来自http://www.weixiushen.com/project/Awesome_FGIA/Awesome_FGIA.html
kbsc13
2019/08/16
1.5K0
细粒度分类你懂吗?——fine-gained image classification
我们在路边看到萌犬可爱至极,然后却不知道这个是哪种狗;看见路边的一个野花却不知道叫什么名字,吃着一种瓜,却不知道是甜瓜还是香瓜傻傻分不清……
机器学习算法工程师
2018/07/27
1.1K0
细粒度分类你懂吗?——fine-gained image classification
【每周CV论文推荐】初学细粒度分类值得阅读的文章
欢迎来到《每周CV论文推荐》。在这个专栏里,还是本着有三AI一贯的原则,专注于让大家能够系统性完成学习,所以我们推荐的文章也必定是同一主题的。
用户1508658
2022/11/07
4210
【每周CV论文推荐】初学细粒度分类值得阅读的文章
双线性汇合(bilinear pooling)在细粒度图像分析及其他领域的进展综述
细粒度图像分类旨在同一大类图像的确切子类。由于不同子类之间的视觉差异很小,而且容易受姿势、视角、图像中目标位置等影响,这是一个很有挑战性的任务。因此,类间差异通常比类内差异更小。双线性汇合(bilinear pooling)计算不同空间位置的外积,并对不同空间位置计算平均汇合以得到双线性特征。外积捕获了特征通道之间成对的相关关系,并且这是平移不变的。双线性汇合提供了比线性模型更强的特征表示,并可以端到端地进行优化,取得了和使用部位(parts)信息相当或甚至更高的性能。
SIGAI学习与实践平台
2018/12/17
2.7K0
双线性汇合(bilinear pooling)在细粒度图像分析及其他领域的进展综述
【新智元干货】计算机视觉必读:目标跟踪、网络压缩、图像分类、人脸识别等
【新智元导读】深度学习目前已成为发展最快、最令人兴奋的机器学习领域之一。本文以计算机视觉的重要概念为线索,介绍深度学习在计算机视觉任务中的应用,包括网络压缩、细粒度图像分类、看图说话、视觉问答、图像理解、纹理生成和风格迁移、人脸识别、图像检索、目标跟踪等。 网络压缩(network compression) 尽管深度神经网络取得了优异的性能,但巨大的计算和存储开销成为其部署在实际应用中的挑战。有研究表明,神经网络中的参数存在大量的冗余。因此,有许多工作致力于在保证准确率的同时降低网路复杂度。 低秩近似
新智元
2018/03/28
1.2K0
【新智元干货】计算机视觉必读:目标跟踪、网络压缩、图像分类、人脸识别等
学界 |「分段映射」帮助利用少量样本习得新类别细粒度分类器
作者:Xiu-Shen Wei等 机器之心编译 参与:Pedro、路 近日,来自南京大学、旷视和阿德莱德大学的研究者在 arXiv 上发布论文,提出一种利用少量样本学习新类别细粒度分类器的新方法 FSFG 模型,该方法包含两个模块:双线性特征学习模块和分类器映射模块。后者中的「分段映射」功能是该模型的关键部分,它通过一种参数更少的方式学习一组相对易实现的子分类器,进而生成决策边界。 细粒度图像识别是一个重要的计算机视觉问题。得益于复杂深层网络结构的应用,该问题解决方案的表现也越来越好。训练这种分类算法所需
机器之心
2018/06/12
1.2K0
基于素描图的细粒度图像检索【附PPT与视频资料】
近年来,随着监控摄像头的普及与应用,监控摄像头系统在打击罪犯和刑侦安全方面起到了至关重要的作用。利用监控系统查找犯罪嫌疑人,从而侦破案件已经成为公安机关的重要破案手段。这一重要应用使得行人重识别问题得到广泛关注。行人重识别是指给定行人在某一监控摄像头下的图片,利用计算机视觉算法在其余监控摄像头下识别出这一特定行人。
马上科普尚尚
2020/05/11
7450
基于素描图的细粒度图像检索【附PPT与视频资料】
【图像分类】细粒度图像分类是什么,有什么方法,发展的怎么样
欢迎大家来自《图像分类》专栏,今天讲述细粒度图像分类问题,这是计算机视觉领域一项极具挑战的研究课题,本文介绍了细粒度图像分类算法的发展现状、相关数据集和竞赛,供大家参考学习。
用户1508658
2019/07/30
3.4K0
【图像分类】细粒度图像分类是什么,有什么方法,发展的怎么样
ECCV2020 Spotlight | 图像定位上的细粒化区域相似性自监督
本文介绍一篇我们发表于ECCV 2020的论文《Self-supervising Fine-grained Region Similarities for Large-scale Image Localization》,很荣幸该论文被收录为spotlight presentation。
AI科技评论
2020/08/17
1.1K0
ECCV2020 Spotlight | 图像定位上的细粒化区域相似性自监督
fine-gained image classification
我们在路边看到萌犬可爱至极,然后却不知道这个是哪种狗;看见路边的一个野花却不知道叫什么名字,吃着一种瓜,却不知道是甜瓜还是香瓜傻傻分不清……
机器学习算法工程师
2018/08/06
4050
fine-gained image classification
CNCC2017中的深度学习与跨媒体智能
01 传统方法与深度学习 图像分割 图像分割是医疗图像中一个很重要的任务,通常分为分割,配准,可视化几个子任务。这里贴一张广义的图像分割的图: 存在的困难: 不同目标区域亮度一致,区分度小, 不同目标
用户1332428
2018/03/09
1.5K0
CNCC2017中的深度学习与跨媒体智能
CNCC2017中的深度学习与跨媒体智能
梦里茶
2017/12/29
2K0
CNCC2017中的深度学习与跨媒体智能
基于PyTorch、易上手,细粒度图像识别深度学习工具库Hawkeye开源
细粒度图像识别 [1] 是视觉感知学习的重要研究课题,在智能新经济和工业互联网等方面具有巨大应用价值,且在诸多现实场景已有广泛应用…… 鉴于当前领域内尚缺乏该方面的深度学习开源工具库,南京理工大学魏秀参教授团队用时近一年时间,开发、打磨、完成了 Hawkeye——细粒度图像识别深度学习开源工具库,供相关领域研究人员和工程师参考使用。本文是对 Hawkeye 的详细介绍。
机器之心
2022/12/15
7690
基于PyTorch、易上手,细粒度图像识别深度学习工具库Hawkeye开源
细粒度情感分析在到餐场景中的应用
经典的细粒度情感分析(ABSA,Aspect-based Sentiment Analysis)主要包含三个子任务,分别为属性抽取、观点抽取以及属性-观点对的情感倾向判定三个级联任务。
美团技术团队
2021/12/13
1.8K0
细粒度情感分析在到餐场景中的应用
​多所高校联合揭秘 TRAC 框架 | 探索 LVLMs 在细粒度视觉描述生成中的潜力!
在他们的真实描述中提供了广泛的概述。特别是,它们在生成细粒度描述方面的效力受到其固有的简洁性、简单性和多目标关注性的显著限制。此外,传统的评估指标如BLEU 和 SPICE(Brocker等人,2016)不适合评估细粒度描述,因为它们过于依赖这些粗略的真实描述,缺乏必要的粒度。
AIGC 先锋科技
2024/07/08
2450
​多所高校联合揭秘 TRAC 框架 | 探索 LVLMs 在细粒度视觉描述生成中的潜力!
北大提出PKU FG-XMedia:细粒度跨媒体检索数据集和评测基准
跨媒体检索(Cross-media Retrieval)是指用户给定任意一种媒体类型数据作为查询样例,系统检索得到与查询样例相关的各种媒体数据。如图 1 所示,当用户给定一张灰背鸥(Slaty-backed Gull)的图像作为查询样例,检索结果包含了图像、文本、视频和音频 4 种媒体数据。现有跨媒体检索研究一般聚焦在粗粒度跨媒体检索(Coarse-grained Cross-media Retrieval),只是将灰背鸥的图像作为鸟的图像进行分析检索,因此检索结果中会包含各种相似鸟类的媒体数据(如灰翅鸥、银鸥、加州海鸥等),而不是灰背鸥的图像、文本、视频和音频数据,如图 1(a) 所示。为了克服上述问题,本文提出了细粒度跨媒体检索(Fine-grained Cross-media Retrieval),即用户给定任意一种媒体类型数据作为查询样例,系统检索得到与查询样例细粒度类别相同的各种媒体数据,如图 1(b) 所示,检索得到灰背鸥的图像、文本、视频和音频数据。
机器之心
2019/10/17
1.4K0
最新ICCV2021 | GAN少样本(26)实例可控(27)视图合成(28)细粒度生成(29)细粒度检索(30)生成对抗GAN
二十六、少样本生成 72、LoFGAN: Fusing Local Representations for Few-shot Image Generation 给定新的、训练未知的类别里的少数可用图像,少样本图像生成,旨在为该类别生成更多数据。以前工作试图通过使用可调整的加权系数来融合这些图像。然而从全局角度来看,不同图像之间存在严重的语义错位,使得生成质量和多样性较差。 为此提出 LocalFusion Generative Adversarial Network (LoFGAN),将这些可用的图像作为一
公众号机器学习与AI生成创作
2022/05/27
8960
最新ICCV2021 | GAN少样本(26)实例可控(27)视图合成(28)细粒度生成(29)细粒度检索(30)生成对抗GAN
CCAF 新框架,利用 CLIP 学习细粒度语义特征,增强跨摄像头识别 !
凡本公众号注明“来源:XXX(非集智书童)”的作品,均转载自其它媒体,版权归原作者所有,如有侵权请联系我们删除,谢谢。
集智书童公众号
2024/06/25
1K0
CCAF 新框架,利用 CLIP 学习细粒度语义特征,增强跨摄像头识别 !
推荐阅读
见微知著:细粒度图像分析进展
1.6K0
细粒度图像分析_图像分类研究现状
1.3K0
资源 | 最新的细粒度图像分析资源
1.5K0
细粒度分类你懂吗?——fine-gained image classification
1.1K0
【每周CV论文推荐】初学细粒度分类值得阅读的文章
4210
双线性汇合(bilinear pooling)在细粒度图像分析及其他领域的进展综述
2.7K0
【新智元干货】计算机视觉必读:目标跟踪、网络压缩、图像分类、人脸识别等
1.2K0
学界 |「分段映射」帮助利用少量样本习得新类别细粒度分类器
1.2K0
基于素描图的细粒度图像检索【附PPT与视频资料】
7450
【图像分类】细粒度图像分类是什么,有什么方法,发展的怎么样
3.4K0
ECCV2020 Spotlight | 图像定位上的细粒化区域相似性自监督
1.1K0
fine-gained image classification
4050
CNCC2017中的深度学习与跨媒体智能
1.5K0
CNCC2017中的深度学习与跨媒体智能
2K0
基于PyTorch、易上手,细粒度图像识别深度学习工具库Hawkeye开源
7690
细粒度情感分析在到餐场景中的应用
1.8K0
​多所高校联合揭秘 TRAC 框架 | 探索 LVLMs 在细粒度视觉描述生成中的潜力!
2450
北大提出PKU FG-XMedia:细粒度跨媒体检索数据集和评测基准
1.4K0
最新ICCV2021 | GAN少样本(26)实例可控(27)视图合成(28)细粒度生成(29)细粒度检索(30)生成对抗GAN
8960
CCAF 新框架,利用 CLIP 学习细粒度语义特征,增强跨摄像头识别 !
1K0
相关推荐
见微知著:细粒度图像分析进展
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验