目录

课程说明

1、动态查询

1.1、查询好友动态

1.2、查询推荐动态

1.3、根据id查询动态

2、圈子互动

2.1、环境搭建

2.2、动态评论

2.3、点赞

2.4、喜欢

课程说明

  • 圈子动态查询

  • 圈子实现评论

  • 圈子实现点赞、喜欢功能

  • 圈子实现评论

1、动态查询

我的动态:查询个人发布的动态列表(分页查询),和之前实现的好友动态,推荐动态实现逻辑是一致。

1.1、查询好友动态

查询好友动态与查询推荐动态显示的结构是一样的,只是其查询数据源不同

1.1.1、接口文档

API接口文档:http://192.168.136.160:3000/project/19/interface/api/142

1.1.2、代码步骤

  • Controller层接受请求参数

  • Service数据封装

    • 调用API查询好友动态详情数据

    • 调用API查询动态发布人详情

    • 构造VO对象

  • API层根据用户ID查询好友发布动态详情

    • 查询好友时间线表

    • 查询动态详情

1.1.3、代码实现

MovementController

/*** 查询好友动态*/@GetMappingpublic ResponseEntity movements(@RequestParam(defaultValue = "1") Integer page,@RequestParam(defaultValue = "10") Integer pagesize) {PageResult pr = movementService.findFriendMovements(page,pagesize);return ResponseEntity.ok(pr);}

MovementService

//好友动态
public PageResult findFriendMovements(Integer page, Integer pagesize) {//1、获取当前用户idLong userId = UserHolder.getUserId();//2、查询数据列表List<Movement> items = movementApi.findFriendMovements(userId,page,pagesize);//3、非空判断if(CollUtil.isEmpty(items)) {return new PageResult();}//4、获取好友用户idList<Long> userIds = CollUtil.getFieldValues(items, "userId", Long.class);//5、循环数据列表Map<Long, UserInfo> userMaps = userInfoApi.findByIds(userIds, null);List<MovementsVo> vos = new ArrayList<>();for (Movement item : items) {//5、一个Movement构建一个Vo对象UserInfo userInfo = userMaps.get(item.getUserId());MovementsVo vo = MovementsVo.init(userInfo, item);vos.add(vo);}//6、构建返回值return new PageResult(page,pagesize,0L,vos);
}

movementApi

@Override
public List<Movement> findFriendMovements(Long friendId, Integer page, Integer pagesize) {//1、查询好友时间线表Query query = Query.query(Criteria.where("friendId").in(friendId)).skip((page - 1)*pagesize).limit(pagesize).with(Sort.by(Sort.Order.desc("created")));List<MovementTimeLine> lines = mongoTemplate.find(query, MovementTimeLine.class);//2、提取动态id集合List<ObjectId> movementIds = CollUtil.getFieldValues(lines, "movementId", ObjectId.class);//3、根据动态id查询动态详情Query movementQuery = Query.query(Criteria.where("id").in(movementIds));return mongoTemplate.find(movementQuery, Movement.class);
}

1.2、查询推荐动态

推荐动态是通过推荐系统计算出的结果,现在我们只需要实现查询即可,推荐系统在后面的课程中完成。

推荐系统计算完成后,会将结果数据写入到Redis中,数据如下:

192.168.31.81:6379> get MOVEMENTS_RECOMMEND_1
"2562,3639,2063,3448,2128,2597,2893,2333,3330,2642,2541,3002,3561,3649,2384,2504,3397,2843,2341,2249"

可以看到,在Redis中的数据是有多个发布id组成(pid)由逗号分隔。所以实现中需要自己对这些数据做分页处理。

1.2.1、接口文档

API接口文档:http://192.168.136.160:3000/project/19/interface/api/145

1.2.2、代码步骤

  • Controller层接受请求参数

  • Service数据封装

    • 从redis获取当前用户的推荐PID列表

    • 如果不存在,调用API随机获取10条动态数据

    • 如果存在,调用API根据PID列表查询动态数据

    • 构造VO对象

  • API层编写方法

    • 随机获取

    • 根据pid列表查询

1.2.3、代码实现

Constants

package com.tanhua.commons.utils;//常量定义
public class Constants {//手机APP短信验证码CHECK_CODE_public static final String SMS_CODE = "CHECK_CODE_";//推荐动态public static final String MOVEMENTS_RECOMMEND = "MOVEMENTS_RECOMMEND_";//推荐视频public static final String VIDEOS_RECOMMEND = "VIDEOS_RECOMMEND_";//圈子互动KEYpublic static final String MOVEMENTS_INTERACT_KEY = "MOVEMENTS_INTERACT_";//动态点赞用户HashKeypublic static final String MOVEMENT_LIKE_HASHKEY = "MOVEMENT_LIKE_";//动态喜欢用户HashKeypublic static final String MOVEMENT_LOVE_HASHKEY = "MOVEMENT_LOVE_";//视频点赞用户HashKeypublic static final String VIDEO_LIKE_HASHKEY = "VIDEO_LIKE";//访问用户public static final String VISITORS = "VISITORS";//关注用户public static final String FOCUS_USER = "FOCUS_USER_{}_{}";//初始化密码public static final String INIT_PASSWORD = "123456";//环信用户前缀public static final String HX_USER_PREFIX = "hx";//jwt加密盐public static final String JWT_SECRET = "itcast";//jwt超时时间public static final int JWT_TIME_OUT = 3_600;
}

MovementController

/*** 推荐动态*/
@GetMapping("/recommend")
public ResponseEntity recommend(@RequestParam(defaultValue = "1") Integer page,@RequestParam(defaultValue = "10") Integer pagesize) {PageResult pr = movementService.findRecommendMovements(page,pagesize);return ResponseEntity.ok(pr);
}

MovementService

//推荐动态
public PageResult findRecommendMovements(Integer page, Integer pagesize) {String redisKey = "MOVEMENTS_RECOMMEND_" + UserHolder.getUserId();String redisData = this.redisTemplate.opsForValue().get(redisKey);List<Movement> list = Collections.EMPTY_LIST;if(StringUtils.isEmpty(redisData)){list = movementApi.randomMovements(pagesize);}else {String[] split = redisData.split(",");if ((page-1) * pagesize > split.length) {return new PageResult();}List<Long> pids = Arrays.stream(split).skip((page - 1) * pagesize).limit(pagesize).map(e -> Convert.toLong(e)).collect(Collectors.toList());list = movementApi.findByPids(pids);}List<Long> userIds = CollUtil.getFieldValues(list, "userId", Long.class);//5、循环数据列表Map<Long, UserInfo> userMaps = userInfoApi.findByIds(userIds, null);List<MovementsVo> vos = new ArrayList<>();for (Movement item : list) {//5、一个Movement构建一个Vo对象UserInfo userInfo = userMaps.get(item.getUserId());MovementsVo vo = MovementsVo.init(userInfo, item);vos.add(vo);}//6、构建返回值return new PageResult(page,pagesize,0L,vos);
}

movementApi

//随机获取
public List<Movement> randomMovements(Integer counts) {TypedAggregation aggregation = Aggregation.newAggregation(Movement.class,Aggregation.sample(counts));AggregationResults<Movement> movements = mongoTemplate.aggregate(aggregation,Movement.class);return movements.getMappedResults();
}//根据PID查询
public List<Movement> findByPids(List<Long> pids) {Query query = Query.query(Criteria.where("pId").in(pids));return mongoTemplate.find(query, Movement.class);
}

1.3、根据id查询动态

根据id查询动态:当手机端查看评论内容时(需要根据动态id,查询动态详情),后续再去查询评论列表

1.3.1、接口文档

API接口文档:http://192.168.136.160:3000/project/19/interface/api/151

1.3.2、代码实现

MovementController

 /*** 根据id查询动态*/@GetMapping("/{id}")public ResponseEntity findById(@PathVariable("id") String movementId) {MovementsVo vo = movementService.findMovementById(movementId);return ResponseEntity.ok(vo);}

MovementService

public MovementsVo findMovementById(String movementId) {Movement movements = movementApi.findById(movementId);if(movements == null) {return null;}else {UserInfo userInfo = userInfoApi.findById(movements.getUserId());return MovementsVo.init(userInfo,movements);}
}

movementApi

@Override
public Movement findById(String movementId) {return mongoTemplate.findById(movementId,Movement.class);
}

2、圈子互动

点赞、喜欢、评论等均可理解为用户对动态的互动。

数据库表:quanzi_comment

将数据记录到表中:保存到MongoDB中
互动表需要几张:需要一张表即可(quanzi_comment)
里面的数据需要分类:通过字段commentType 1-点赞,2-评论,3-喜欢
{"_id" : ObjectId("5fe7f9263c851428107cd4e8"),"publishId" : ObjectId("5fae53947e52992e78a3afa5"),"commentType" : 1,"userId" : NumberLong(1),"publishUserId" : NumberLong(1),"created" : NumberLong(1609038118275),"likeCount" : 0,"_class" : "com.tanhua.domain.mongo.Comment"
}

数据存储位置:redismongodb

mongodb中的数据

  • 在动态详情Movement表中,加入喜欢,点赞,评论数量:检查数据库访问压力

    • 互动操作的时候,不要忘记对上面的字段进行维护

  • 圈子互动的表 comment

  • 互动完成(点赞,喜欢):不仅要将数据保存到mongo中,需要记录到redis中

  • 页面查询圈子列表时,可以从redis中判断是否有点赞,和喜欢历史

2.1、环境搭建

2.1.1 创建API接口

public interface CommentApi {}

2.1.2 创建API实现类

@DubboService
public class CommentApiImpl implements CommentApi {@Autowiredprivate MongoTemplate mongoTemplate;
}

2.1.3 Movement对象

@Data
@NoArgsConstructor
@AllArgsConstructor
@Document(collection = "movement")
public class Movement implements java.io.Serializable {private ObjectId id; //主键id//redis实现,使用Mongodb实现private Long pid; //Long类型,用于推荐系统的模型(自动增长)private Long created; //发布时间private Long userId;private String textContent; //文字private List<String> medias; //媒体数据,图片或小视频 urlprivate String longitude; //经度private String latitude; //纬度private String locationName; //位置名称private Integer state = 0;//状态 0:未审(默认),1:通过,2:驳回//补充字段private Integer likeCount = 0; //点赞数private Integer commentCount = 0; //评论数private Integer loveCount = 0; //喜欢数//根据评论类型,获取对应的互动数量public Integer statisCount(Integer commentType) {if (commentType == CommentType.LIKE.getType()) {return this.likeCount;} else if (commentType == CommentType.COMMENT.getType()) {return this.commentCount;} else {return loveCount;}}
}

2.1.4 实体类对象

package com.tanhua.domain.mongo;import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.bson.types.ObjectId;
import org.springframework.data.mongodb.core.mapping.Document;/*** 圈子互动表(点赞,评论,喜欢)*/
@Data
@NoArgsConstructor
@AllArgsConstructor
@Document(collection = "comment")
public class Comment implements java.io.Serializable{private ObjectId id;private ObjectId publishId;    //发布idprivate Integer commentType;   //评论类型,1-点赞,2-评论,3-喜欢private String content;        //评论内容  private Long userId;           //评论人   private Long publishUserId;    //被评论人IDprivate Long created;          //发表时间private Integer likeCount = 0; //当前评论的点赞数}

2.1.5 VO对象

@Data
@NoArgsConstructor
@AllArgsConstructor
public class CommentVo implements Serializable {private String id; //评论idprivate String avatar; //头像private String nickname; //昵称private String content; //评论private String createDate; //评论时间private Integer likeCount; //点赞数private Integer hasLiked; //是否点赞(1是,0否)public static CommentVo init(UserInfo userInfo, Comment item) {CommentVo vo = new CommentVo();BeanUtils.copyProperties(userInfo, vo);BeanUtils.copyProperties(item, vo);vo.setHasLiked(0);Date date = new Date(item.getCreated());vo.setCreateDate(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(date));vo.setId(item.getId().toHexString());return vo;}
}

2.1.6 CommentType枚举

/*** 评论类型:1-点赞,2-评论,3-喜欢*/
public enum CommentType {LIKE(1), COMMENT(2), LOVE(3);int type;CommentType(int type) {this.type = type;}public int getType() {return type;}
}

2.2、动态评论

功能包括:查询评论列表,发布评论,对评论点赞和取消点赞。

2.2.1 分页列表查询

CommentController

//分页查询评理列表
@GetMapping
public ResponseEntity findComments(@RequestParam(defaultValue = "1") Integer page,@RequestParam(defaultValue = "10") Integer pagesize,String movementId) {PageResult pr = commentsService.findComments(movementId,page,pagesize);return ResponseEntity.ok(pr);
}

CommentService

//分页查询评理列表
public PageResult findComments(String movementId, Integer page, Integer pagesize) {//1、调用API查询评论列表List<Comment> list = commentApi.findComments(movementId,CommentType.COMMENT,page,pagesize);//2、判断list集合是否存在if(CollUtil.isEmpty(list)) {return new PageResult();}//3、提取所有的用户id,调用UserInfoAPI查询用户详情List<Long> userIds = CollUtil.getFieldValues(list, "userId", Long.class);Map<Long, UserInfo> map = userInfoApi.findByIds(userIds, null);//4、构造vo对象List<CommentVo> vos = new ArrayList<>();for (Comment comment : list) {UserInfo userInfo = map.get(comment.getUserId());if(userInfo != null) {CommentVo vo = CommentVo.init(userInfo, comment);vos.add(vo);}}//5、构造返回值return new PageResult(page,pagesize,0l,vos);
}

CommentAPI

//分页查询
public List<Comment> findComments(String movementId, CommentType commentType, Integer page, Integer pagesize) {//1、构造查询条件Query query = Query.query(Criteria.where("publishId").is(new ObjectId(movementId)).and("commentType").is(commentType.getType())).skip((page -1) * pagesize).limit(pagesize).with(Sort.by(Sort.Order.desc("created")));//2、查询并返回return mongoTemplate.find(query,Comment.class);
}

2.2.2 发布评论

CommentController

@RestController
@RequestMapping("/comments")
public class CommentsController {@Autowiredprivate CommentsService commentsService;/*** 发布评论*/@PostMappingpublic ResponseEntity saveComments(@RequestBody Map map) {String movementId = (String )map.get("movementId");String comment = (String)map.get("comment");commentsService.saveComments(movementId,comment);return ResponseEntity.ok(null);}
}

CommentService

@Service
@Slf4j
public class CommentsService {@DubboReferenceprivate CommentApi commentApi;//发布评论public void saveComments(String movementId, String comment) {//1、获取操作用户idLong userId = UserHolder.getUserId();//2、构造CommentComment comment1 = new Comment();comment1.setPublishId(new ObjectId(movementId));comment1.setCommentType(CommentType.COMMENT.getType());comment1.setContent(comment);comment1.setUserId(userId);comment1.setCreated(System.currentTimeMillis());//3、调用API保存评论Integer commentCount = commentApi.save(comment1);log.info("commentCount = " + commentCount);}
}

CommentAPI

//发布评论,并获取评论数量
public Integer save(Comment comment) {//1、查询动态Movement movement = mongoTemplate.findById(comment.getPublishId(), Movement.class);//2、向comment对象设置被评论人属性if(movement != null) {comment.setPublishUserId(movement.getUserId());}//3、保存到数据库mongoTemplate.save(comment);//4、更新动态表中的对应字段Query query = Query.query(Criteria.where("id").is(comment.getPublishId()));Update update = new Update();if(comment.getCommentType() == CommentType.LIKE.getType()) {update.inc("likeCount",1);}else if (comment.getCommentType() == CommentType.COMMENT.getType()){update.inc("commentCount",1);}else {update.inc("loveCount",1);}//设置更新参数FindAndModifyOptions options = new FindAndModifyOptions();options.returnNew(true) ;//获取更新后的最新数据Movement modify = mongoTemplate.findAndModify(query, update, options, Movement.class);//5、获取最新的评论数量,并返回return modify.statisCount(comment.getCommentType() );
}

测试API代码

@RunWith(SpringRunner.class)
@SpringBootTest(classes = AppServerApplication.class)
public class CommentApiTest {@DubboReferenceprivate CommentApi commentApi;@Testpublic void testSave() {Comment comment = new Comment();comment.setCommentType(CommentType.COMMENT.getType());comment.setUserId(106l);comment.setCreated(System.currentTimeMillis());comment.setContent("测试评论");comment.setPublishId();commentApi.save(comment);}
}

2.3、点赞

2.3.1、编写Controller

修改MovementsController代码,添加点赞与取消点赞方法

/*** 点赞*/
@GetMapping("/{id}/like")
public ResponseEntity like(@PathVariable("id") String movementId) {Integer likeCount = commentsService.likeComment(movementId);return ResponseEntity.ok(likeCount);
}/*** 取消点赞*/
@GetMapping("/{id}/dislike")
public ResponseEntity dislike(@PathVariable("id") String movementId) {Integer likeCount = commentsService.dislikeComment(movementId);return ResponseEntity.ok(likeCount);
}

2.3.2、编写Service

创建CommentService,添加点赞与取消点赞方法

//动态点赞
public Integer likeComment(String movementId) {//1、调用API查询用户是否已点赞Boolean hasComment = commentApi.hasComment(movementId,UserHolder.getUserId(),CommentType.LIKE);//2、如果已经点赞,抛出异常if(hasComment) {throw  new BusinessException(ErrorResult.likeError());}//3、调用API保存数据到MongodbComment comment = new Comment();comment.setPublishId(new ObjectId(movementId));comment.setCommentType(CommentType.LIKE.getType());comment.setUserId(UserHolder.getUserId());comment.setCreated(System.currentTimeMillis());Integer count = commentApi.save(comment);//4、拼接redis的key,将用户的点赞状态存入redisString key = Constants.MOVEMENTS_INTERACT_KEY + movementId;String hashKey = Constants.MOVEMENT_LIKE_HASHKEY + UserHolder.getUserId();redisTemplate.opsForHash().put(key,hashKey,"1");return count;
}//取消点赞
public Integer dislikeComment(String movementId) {//1、调用API查询用户是否已点赞Boolean hasComment = commentApi.hasComment(movementId,UserHolder.getUserId(),CommentType.LIKE);//2、如果未点赞,抛出异常if(!hasComment) {throw new BusinessException(ErrorResult.disLikeError());}//3、调用API,删除数据,返回点赞数量Comment comment = new Comment();comment.setPublishId(new ObjectId(movementId));comment.setCommentType(CommentType.LIKE.getType());comment.setUserId(UserHolder.getUserId());Integer count = commentApi.delete(comment);//4、拼接redis的key,删除点赞状态String key = Constants.MOVEMENTS_INTERACT_KEY + movementId;String hashKey = Constants.MOVEMENT_LIKE_HASHKEY + UserHolder.getUserId();redisTemplate.opsForHash().delete(key,hashKey);return count;
}

2.3.3、修改API服务

判断Comment数据是否存在

//判断comment数据是否存在
public Boolean hasComment(String movementId, Long userId, CommentType commentType) {Criteria criteria = Criteria.where("userId").is(userId).and("publishId").is(new ObjectId(movementId)).and("commentType").is(commentType.getType());Query query = Query.query(criteria);return mongoTemplate.exists(query,Comment.class); //判断数据是否存在
}

删除互动数据

//删除
public Integer delete(Comment comment) {//1、删除Comment表数据Criteria criteria = Criteria.where("userId").is(comment.getUserId()).and("publishId").is(comment.getPublishId()).and("commentType").is(comment.getCommentType());Query query = Query.query(criteria);mongoTemplate.remove(query,Comment.class);//2、修改动态表中的总数量Query movementQuery = Query.query(Criteria.where("id").is(comment.getPublishId()));Update update = new Update();if(comment.getCommentType() == CommentType.LIKE.getType()) {update.inc("likeCount",-1);}else if (comment.getCommentType() == CommentType.COMMENT.getType()){update.inc("commentCount",-1);}else {update.inc("loveCount",-1);}//设置更新参数FindAndModifyOptions options = new FindAndModifyOptions();options.returnNew(true) ;//获取更新后的最新数据Movement modify = mongoTemplate.findAndModify(movementQuery, update, options, Movement.class);//5、获取最新的评论数量,并返回return modify.statisCount(comment.getCommentType() );
}

2.3.4、修改查询动态点赞数

修改之前的查询圈子列表代码,从redis查询是否具有操作记录

2.4、喜欢

喜欢和取消喜欢:和刚才的点赞与取消点赞基本上市一模一样的!操作的类型comment_type=3,操作的字段loveCount

MovementsController

修改MovementsController代码,添加喜欢与取消喜欢方法

/*** 喜欢*/
@GetMapping("/{id}/love")
public ResponseEntity love(@PathVariable("id") String movementId) {Integer likeCount = commentsService.loveComment(movementId);return ResponseEntity.ok(likeCount);
}/*** 取消喜欢*/
@GetMapping("/{id}/unlove")
public ResponseEntity unlove(@PathVariable("id") String movementId) {Integer likeCount = commentsService.disloveComment(movementId);return ResponseEntity.ok(likeCount);
}

CommentService

修改CommentService,添加点赞与取消点赞方法

/*** 喜欢*/
@GetMapping("/{id}/love")
public ResponseEntity love(@PathVariable("id") String movementId) {Integer likeCount = commentsService.loveComment(movementId);return ResponseEntity.ok(likeCount);
}/*** 取消喜欢*/
@GetMapping("/{id}/unlove")
public ResponseEntity unlove(@PathVariable("id") String movementId) {Integer likeCount = commentsService.disloveComment(movementId);return ResponseEntity.ok(likeCount);
}

【探花交友】day05—圈子互动相关推荐

  1. 黑马探花交友----3.圈子-发布动态点赞评论

    学习目标: 圈子功能说明 圈子技术实现 圈子技术方案 圈子实现发布动态 圈子实现好友动态 圈子实现推荐动态 圈子实现点赞.喜欢功能 圈子实现评论 圈子实现评论的点赞 1.功能说明 探花交友项目中的圈子 ...

  2. 探花交友_第7章-完善消息功能以及个人主页

    探花交友_第7章-完善消息功能以及个人主页 文章目录 探花交友_第7章-完善消息功能以及个人主页 1.消息点赞.喜欢.评论列表 1.1.dubbo服务 1.1.1.定义接口 1.1.2.编写实现 1. ...

  3. 探花交友_第6章_圈子互动(新版)

    探花交友_第6章_圈子互动(新版) 文章目录 探花交友_第6章_圈子互动(新版) 课程说明 1. 动态查询 1.1 查询好友动态 1.1.1 接口文档 1.1.2 代码步骤 1.1.3 代码实现 ta ...

  4. java 探花交友项目day5 推荐好友列表 MongoDB集群 发布动态,查询动态 圈子功能

    推荐好友列表 需求分析 推荐好友:分页形式查询推荐的用户列表,根据评分排序显示 代码实现: tanhuaController: /**  * 查询分页推荐好友列表  */ @GetMapping(&q ...

  5. 探花交友_第5章_圈子、小视频功能实现

    探花交友_第5章_圈子.小视频功能实现 文章目录 探花交友_第5章_圈子.小视频功能实现 1.圈子点赞实现分析 2.点赞 2.1.定义枚举 2.2.dubbo服务 2.2.1.定义接口 2.2.2.编 ...

  6. 探花交友_第4章_圈子功能实现

    探花交友_第4章_圈子功能实现 文章目录 探花交友_第4章_圈子功能实现 1.抽取common工程 1.1.创建my-tanhua-common工程 1.2.通用枚举 1.3.抽取mapper 1.4 ...

  7. java项目---探花交友

    一.项目介绍 1.项目背景 在线社交是互联网时代的产物,已成为互联网用户的基础需求之一.移动互联网自2003年起快速发展, 促使在线社交逐渐从PC端转移至移动端.移动社交最初以熟人社交为主,以维系熟人 ...

  8. 探花交友_第2章_环境搭建(新版)

    探花交友_第2章_环境搭建(新版) 文章目录 探花交友_第2章_环境搭建(新版) 课程介绍 <探花交友> 1.项目介绍 1.1.项目背景 1.2.市场分析 1.3.目标用户群体 1.4.使 ...

  9. 全网最全-探花交友项目-面试总结-简历优化

    项目名称: 远亲不如近邻-"有好邻"社区社交服务APP 移动短视频社交平台 O2O同城社交APP "曲园社交读书APP 高校留学生社交APP 项目简介: 本项目是一个在线 ...

最新文章

  1. cython安装、使用
  2. COM线程模型的行为
  3. java开发错误_每个Java开发人员都必须避免的9个安全错误
  4. 【OpenCV 例程200篇】76. OpenCV 实现图像傅里叶变换
  5. 离散数学之集合论 【上】
  6. An Easy Problem(信息学奥赛一本通-T1223)
  7. Mac ssh-key 对应git账户
  8. 诗歌rails之获取本地ip地址
  9. Entity Framework Codefirst的配置步骤
  10. (7)雅思屠鸭第七天:阅读中538个考点词一网打尽
  11. 青春是一场永志的劫数
  12. JWT实现接口双重认证,提供安全又不复杂的接口安全能力
  13. 文本分类入门(二)——训练
  14. 第7-5课:格式化文件读写与 PCX 图像文件
  15. coap php 发送,CoAP 客户端
  16. linux tar.xz 解压命令,Linux下.tar.xz文件的解压教程详解
  17. 在线预览doc,docx文档
  18. Tekton实战案例--S2I
  19. windows聚焦照片_如何拍摄始终聚焦的照片
  20. Revit翻模 | 千呼万唤始上架的『图转喷淋』到底妙在哪里?

热门文章

  1. JavaWeb实现打印
  2. 知网文献使用EndNote X9在word插入参考文献
  3. ubuntu18.04 littlevgl运行环境搭建
  4. 2021年起重机械指挥考试总结及起重机械指挥考试技巧
  5. windows 工具命令 cmd
  6. PFSO-T5,一种OLED材料
  7. android集合方法,android中intent传递list或者对象的方法
  8. 打败你的十五个坏习惯
  9. 网页设计基础(HTML)
  10. 64位系统下同时使用64位和32位的eclipse