当前位置: 首页 > news >正文

【SpringCloud】-OpenFeign实战及源码解析、与Ribbon结合

一、背景介绍

二、正文

OpenFeign是什么?

OpenFeign(简称Feign)是一个声明式的Web服务客户端,用于简化服务之间的HTTP通信。与Nacos和Ribbon等组件协同,以支持在微服务体系结构中方便地进行服务间的通信; OpenFeign在默认情况下集成了Hystrix,提供了服务容错和服务降级的功能。

OpenFeign的作用是什么?

按照单一职责,也为了满足可复用、可扩展的核心我们可以对整体业务拆分成不同的服务,这样涉及到的一个问题就是某一个服务的逻辑实现需要依托另一个服务的信息,这样服务和服务之间需要通信来进行消息传递,但是服务和服务之间如果直接通信的话耦合关系变强,无法达到高复用的作用。使用了OpenFeign技术来解决这个问题,如果其中一个类需要调用另一个类的某一个方法的话,直接通过OpenFeign这个第三方转发这个调用,减少直接通信带来的耦合关系
在这里插入图片描述通过使用@FeignClien注解标识UserApiService这个客户端,对【internetbar-provider-user】这个服务远程调用。OpenFeign会根据接口自动创建一个实现类,发起HTTP请求去用Get的方式访问远程服务的【/userManage/getUserInfoById/{userId}】这个接口

Feign和OpenFeign的关系?

OpenFeign是对Feign的增强,对mvc注解的增强,那如何理解这句话呢?
共同点:Feign和OpenFeign都是用于简化服务之间HTTP通信的Web服务客户端
不同点:
所属不同:Feign是Netflix开发HTTP客户端;OpenFeign是SpringCloud对Feign的重新实现和增强,OpenFeign引入了SpringMVC的注解,例如:@GetMapping等注解可以类似于SpringMVC的一些效果
Hystrix集成:OpenFeign默认集成Hystrix,提供服务容错和服务降级的功能;Feign中需要显式地添加Hystrix的依赖并配置
在这里插入图片描述


Java还有哪些服务调用方式?

Okhttp、HttpURLConnection、RestTemplate


使用OpenFeign和不使用的区别对比?

我们先从代码形式上来看一下两者如何使用:
基于RestTemplate使用HTTP请求调用服务
在这里插入图片描述

在这里插入图片描述

基于OpenFeign调用
在这里插入图片描述
Feign可以根据接口的定义生成客户端所需要的代码,这部分代码被Feign封装到底层,并且底层同步的实现了负载均衡和服务发现、服务容错等功能,不需要我们像使用RestTemplate一样额外的手动去配置,一个简单的API接口就可以帮助我们做了很多事情。



实战

引入依赖

<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>

在启动类添加@EnableFeignClients注解

@EnableFeignClients(defaultConfiguration = FeignAutoConfiguration.class)

添加FeignClient接口

package cn.itcast.order.clients;import cn.itcast.order.pojo.User;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;/*** @BelongsProject: cloud-demo* @BelongsPackage: cn.itcast.order.clients* @CreateTime: 2023-03-17  12:29* @Description: TODO* @Version: 1.0*/
@FeignClient("userservice")
public interface UserClient {@GetMapping("/user/{id}")User findById(@PathVariable("id") Long id);
}

修改逻辑代码

使用RestTemplate**

package cn.itcast.order.service;import cn.itcast.order.clients.UserClient;
import cn.itcast.order.mapper.OrderMapper;
import cn.itcast.order.pojo.Order;
import cn.itcast.order.pojo.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class OrderService {@Autowiredprivate OrderMapper orderMapper;@Autowiredprivate RestTemplate template;public Order queryOrderById(Long orderId){//1、根据用户id查询用户的订单信息Order order = orderMapper.findById(orderId);//2、利用RestTemplate发起http请求,查询用户信息String url="http://userservice/user/"+order.getUserId();User forObject = template.getForObject(url, User.class);//3、将user信息封装到Order中order.setUser(forObject);// 返回return order;}
}

使用Feign接口

package cn.itcast.order.service;import cn.itcast.order.clients.UserClient;
import cn.itcast.order.mapper.OrderMapper;
import cn.itcast.order.pojo.Order;
import cn.itcast.order.pojo.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class OrderService {@Autowiredprivate OrderMapper orderMapper;@Autowiredprivate UserClient userClient;public Order queryOrderById(Long orderId) {//1、查询订单Order order = orderMapper.findById(orderId);//2、用Feign远程调用User user = userClient.findById(order.getUserId());//3、封装user到Orderorder.setUser(user);//4、返回return order;}
}

自定义配置

配置日志(以debug级别输出 )

当我们进行服务调用的过程中有可能出现接口调用失败的问题,或者我们想要输出请求或响应的详细信息,我们可以配置日志

日志级别有哪些?

NONE:不记录任何日志(默认值)
BASIC:记录请求方法、URL、响应状态代码及执行时间
HEADERS:记录BASIC级别的基础上,记录请求和响应的header
FULL:记录请求和响应的header、body和元数据


方式一:配置文件

全局:所有服务生效
feign:client:config:default:loggerLevel: FULL
#某服务生效
feign:client:config:userservice:loggerLevel: FULL

方式二:配置类

添加配置类

package cn.itcast.order.config;import feign.Logger;
import org.springframework.context.annotation.Bean;/*** @BelongsProject: cloud-demo* @BelongsPackage: cn.itcast.order.config* @CreateTime: 2023-03-17  13:05* @Description: TODO* @Version: 1.0*/
public class DefaultFeignConfiguration {@Beanpublic Logger.Level feignLogLevel() {return Logger.Level.BASIC;}
}

方式三:在启动类上添加注解@EnableFeignClients

全局配置:对所有生效

@EnableFeignClients(defaultConfiguration =FeignAutoConfiguration.class)

局部配置:只对userservice服务生效

@EnableFeignClients(value = “userservice”, defaultConfiguration = FeignAutoConfiguration.class)


性能优化

底层的客户端实现:
URLConnection:默认实现,不支持连接池
Apache HttpClient:支持连接池
OKHttp:支持连接池


优化包括
使用连接池代替默认的URLConnection
日志级别,最好用basic或none


实操——Feign工程化

引入feign-httpClient依赖

 <dependency><groupId>io.github.openfeign</groupId><artifactId>feign-httpclient</artifactId>
</dependency>

配置文件开启httpClient功能,设置连接池参数

feign:client:config:default:  #default全局的配置loggerLevel: BASIC  #日志级别(基本的请求和响应信息)httpclient:enabled: true #开启feign对HttpClient的支持max-connections: 200  #最大的连接数max-connections-per-route: 50 #每个路径的最大连接数

目的:将公共的配置抽取出来

新建一个Maven项目,服务名叫——feign-api

引入依赖

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><parent><artifactId>cloud-demo</artifactId><groupId>cn.itcast.demo</groupId><version>1.0</version></parent><modelVersion>4.0.0</modelVersion><artifactId>feign-api</artifactId><properties><maven.compiler.source>8</maven.compiler.source><maven.compiler.target>8</maven.compiler.target></properties><dependencies><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign</artifactId></dependency></dependencies></project>

第一步、公有接口——工程化核心

在项目中创建一个Feign客户端接口,通过使用@FeignClient注解来声明要调用的服务和服务接口

package cn.itcast.feign.clients;import cn.itcast.feign.pojo.User;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;/*** @BelongsProject: cloud-demo* @BelongsPackage: cn.itcast.order.clients* @CreateTime: 2023-03-17  12:29* @Description: TODO* @Version: 1.0*/
@FeignClient("userservice")
public interface UserClient {@GetMapping("/user/{id}")User findById(@PathVariable("id") Long id);
}

第二步、配置类—用于配置Feign客户端的一些属性,如日志级别等

package cn.itcast.feign.config;import feign.Logger;
import org.springframework.context.annotation.Bean;/*** @BelongsProject: cloud-demo* @BelongsPackage: cn.itcast.order.config* @CreateTime: 2023-03-17  13:05* @Description: TODO* @Version: 1.0*/
public class DefaultFeignConfiguration {@Beanpublic Logger.Level feignLogLevel() {return Logger.Level.BASIC;}
}

实体类

package cn.itcast.feign.pojo;import lombok.Data;@Data
public class User {private Long id;private String username;private String address;
}

orderservice服务

第三步、启动类—启用Feign客户端

在Spring Boot应用程序中,通过使用@EnableFeignClients注解启用Feign客户端,并指定要扫描的Feign客户端接口所在的包。

package cn.itcast.order;import cn.itcast.feign.clients.UserClient;
import com.netflix.loadbalancer.IRule;
import com.netflix.loadbalancer.RandomRule;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.cloud.openfeign.EnableFeignClients;
import org.springframework.cloud.openfeign.FeignAutoConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.web.client.RestTemplate;@MapperScan("cn.itcast.order.mapper")
@SpringBootApplication
@EnableFeignClients(clients = UserClient.class,defaultConfiguration = FeignAutoConfiguration.class)public class OrderApplication {public static void main(String[] args) {SpringApplication.run(OrderApplication.class, args);}/*** @Description: 将RestTemplate注册到容器中* @Date: 2023/3/13 11:28* @return: org.springframework.web.client.RestTemplate**/@Bean@LoadBalancedpublic RestTemplate restTemplate() {return new RestTemplate();}/*** @Description: 修改负载均衡策略方式* @Date: 2023/3/13 19:56* @return: com.netflix.loadbalancer.IRule**/@Beanpublic IRule randomRule() {return new RandomRule();}
}

实体类

package cn.itcast.order.pojo;import cn.itcast.feign.pojo.User;
import lombok.Data;@Data
public class Order {private Long id;private Long price;private String name;private Integer num;private Long userId;private User user;
}

Mapper类

package cn.itcast.order.mapper;import cn.itcast.order.pojo.Order;
import org.apache.ibatis.annotations.Select;public interface OrderMapper {@Select("select * from tb_order where id = #{id}")Order findById(Long id);
}

Controller类

package cn.itcast.order.web;import cn.itcast.order.pojo.Order;
import cn.itcast.order.service.OrderService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
@RequestMapping("/order")
public class OrderController {@Autowiredprivate OrderService orderService;@GetMapping("/{orderId}")public Order queryOrderByUserId(@PathVariable("orderId") Long orderId) {// 根据id查询订单并返回return orderService.queryOrderById(orderId);}
}

第四步、使用Feign客户端—service类

在需要调用远程服务的地方,注入Feign客户端并使用

package cn.itcast.order.service;import cn.itcast.feign.clients.UserClient;
import cn.itcast.feign.pojo.User;
import cn.itcast.order.mapper.OrderMapper;
import cn.itcast.order.pojo.Order;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class OrderService {@Autowiredprivate OrderMapper orderMapper;@Autowiredprivate UserClient userClient;public Order queryOrderById(Long orderId) {//1、查询订单Order order = orderMapper.findById(orderId);//2、用Feign远程调用User user = userClient.findById(order.getUserId());//3、封装user到Orderorder.setUser(user);//4、返回return order;}
}

通过上述的四个步骤(公有接口、配置类、启动类—启用Feign客户端、使用Feign客户端—service类)我们就实现了Feign的工程化,以声明式的方式定义了服务调用接口,配置了Feign客户端

基于工程化的思想,所有的服务调用被抽成公共的API,单独放在一个包中,我们只需要调用公共接口,不需要写太多重复代码,哪个服务需要调用引用对应的包就可以了,避免冗余代码,这也是设计模式思想(可复用)的体现。


集成Hystrix

我们需要先了解雪崩效应,通过OpenFeign进行多服务之间调用,如果服务提供者出现了故障将会导致服务消费者不可用。比方说login服务需要调用user服务获取登录用户的基本信息,此时user服务可能服务挂掉或者被攻击了,此时login服务可能将会处于等待过程,我们可以使用Hystrix进行熔断,当远程服务调用失败或超时时,熔断器会出发,并执行降级逻辑。
关于OpenFeign和Hystrix的集成使用请等待我之后关于Hystrix的博客详解哦~~~


源码解析

1、Spring启动时初始化-生成代理类

在服务启动时,Spring会加载配置并且初始化Spring容器、扫描并加载项目当中用到的组件,此时OpenFeign的配置也会被程序加载(包括:自定义Feign配置类、负载均衡配置、熔断器配置等),扫描带有@FeignClient注解的接口,Feign会使用动态代理生成代理类并注册到Spring容器中

2、服务发现和负载均衡

Nacos注册中心:如果项目中用到了Nacos注册中心,Feign会与Nacos集成通过服务名进行服务发现
Ribbon负载均衡:如果项目中使用了Ribbon,可以实现对服务实例的负载均衡
大家可以看下面这张图片,我们点进OpenFeign的依赖,会发现里面自动集成了Ribbon
在这里插入图片描述
那Feign和Ribbon、RestTemplate三者之间的关系是什么样的呢?
在这里插入图片描述

3、请求处理

场景:服务A调用服务B
当服务A通过OpenFeign的方式调用服务B时,实际上是通过代理类最终调用LoadBalancerFeignClient的execute方法,execute方法内部整合了Ribbon去实现负载均衡,最终会找到真实要请求的服务地址,从而发送RibbonRequest请求,返回响应内容给服务A
这也是Feign的核心代码:
在这里插入图片描述


为什么Feign第一次调用耗时很长?

这个问题是我在网上看到的一个问题,大家也可以参考参考图中说到的原因:
在这里插入图片描述



三、总结

OpenFeign帮助我们简化了服务之间的通信,同时继承了负载均衡、熔断器等功能,帮助我们更好的实现服务调用。之后我将会针对OpenFeign中集成的Hystrix做分享,大家敬请期待


如果有想要交流的内容欢迎在评论区进行留言,如果这篇文档受到了您的喜欢那就留下你点赞+收藏+评论脚印支持一下博主~


相关文章:

【SpringCloud】-OpenFeign实战及源码解析、与Ribbon结合

一、背景介绍 二、正文 OpenFeign是什么&#xff1f; OpenFeign&#xff08;简称Feign&#xff09;是一个声明式的Web服务客户端&#xff0c;用于简化服务之间的HTTP通信。与Nacos和Ribbon等组件协同&#xff0c;以支持在微服务体系结构中方便地进行服务间的通信&#xff1b…...

走进数字金融峰会,为金融科技数字化赋能

12月20—21日&#xff0c;FSIDigital数字金融峰会在上海圆满召开。本次峰会包含InsurDigital数字保险峰会和B&SDigital数字银行与证券峰会2场平行峰会&#xff1b;吸引了近600位来自保险、银行、证券以及金融科技等行业的领导者和专家齐聚一堂&#xff0c;共同探讨金融业数…...

docker-compose部署kafka

docker-compose.yml配置 version: "3" services:kafka:image: bitnami/kafka:latestports:- 7050:7050environment:- KAFKA_ENABLE_KRAFTyes- KAFKA_CFG_PROCESS_ROLESbroker,controller- KAFKA_CFG_CONTROLLER_LISTENER_NAMESCONTROLLER- KAFKA_CFG_LISTENERSPLAIN…...

Spark与Hadoop的关系和区别

在大数据领域&#xff0c;Spark和Hadoop是两个备受欢迎的分布式数据处理框架&#xff0c;它们在处理大规模数据时都具有重要作用。本文将深入探讨Spark与Hadoop之间的关系和区别&#xff0c;以帮助大家的功能和用途。 Spark和Hadoop简介 1 Hadoop Hadoop是一个由Apache基金会…...

蓝桥杯-Excel地址[Java]

目录&#xff1a; 学习目标&#xff1a; 学习内容&#xff1a; 学习时间&#xff1a; 题目&#xff1a; 题目描述: 输入描述: 输出描述: 输入输出样例: 示例 1: 运行限制: 题解: 思路: 学习目标&#xff1a; 刷蓝桥杯题库日记 学习内容&#xff1a; 编号96题目Ex…...

OSPF多区域配置-新版(12)

目录 整体拓扑 操作步骤 1.基本配置 1.1 配置R1的IP 1.2 配置R2的IP 1.3 配置R3的IP 1.4 配置R4的IP 1.5 配置R5的IP 1.6 配置R6的IP 1.7 配置PC-1的IP地址 1.8 配置PC-2的IP地址 1.9 配置PC-3的IP地址 1.10 配置PC-4的IP地址 1.11 检测R5与PC1连通性 1.12 检测…...

华为---USG6000V防火墙web基本配置示例

目录 1. 实验要求 2. 配置思路 3. 网络拓扑图 4. USG6000V防火墙端口和各终端相关配置 5. 在USG6000V防火墙web管理界面创建区域和添加相应端口 6. 给USG6000V防火墙端口配置IP地址 7. 配置通行策略 8. 测试验证 8.1 逐个删除策略&#xff0c;再看各区域终端通信情况 …...

Ksher H5页面支付实例指导 (PHP实现)

背景 前两天&#xff0c;公司的项目&#xff0c;为了满足泰国客户的支付需求&#xff0c;要求使用 Ksher (开时支付) 对接任务突然就给了鄙人&#xff0c;一脸懵 … 通过了解客户的使用场景、以及参考官网指导 发现&#xff1a;Ksher支付最令人满意的便是 —— 提供了便捷的 支…...

https密钥认证、上传镜像实验

一、第一台主机通过https密钥对认证 1、安装docker服务 &#xff08;1&#xff09;安装环境依赖包 yum -y install yum-utils device-mapper-persistent-data lvm2 &#xff08;2&#xff09;设置阿里云镜像源 yum-config-manager --add-repo http://mirrors.aliyun.com/do…...

three.js使用精灵模型Sprite渲染森林

效果&#xff1a; 源码&#xff1a; <template><div><el-container><el-main><div class"box-card-left"><div id"threejs" style"border: 1px solid red"></div><div class"box-right&quo…...

什么是数据可视化?数据可视化的流程与步骤

前言 数据可视化将大大小小的数据集转化为更容易被人脑理解和处理的视觉效果。可视化在我们的日常生活中非常普遍&#xff0c;但它们通常以众所周知的图表和图形的形式出现。正确的数据可视化以有意义和直观的方式为复杂的数据集提供关键的见解。 数据可视化定义 数据可视化…...

2022年山东省职业院校技能大赛高职组云计算赛项试卷第二场-容器云

2022年山东省职业院校技能大赛高职组云计算赛项试卷 目录 【赛程名称】云计算赛项第二场-容器云 需要竞赛软件包以及资料可以私信博主&#xff01; 【赛程名称】云计算赛项第二场-容器云 【赛程时间】2022-11-27 09:00:00至2022-11-27 16:00:00 说明&#xff1a;完成本任务…...

Unity3D 中播放 RTSP 监控视频

【Unity 3D】怎么在 WebGL 中低延迟播放 RTSP 监控 - 简书[Unity 3D] 开箱即食的头部监控厂商 SDK 集成框架 - 简书 Unity3d Windows播放视频&#xff08;视频流&#xff09;功能组/插件支持对比_ffmpeg for unity-CSDN博客Unity UMP打包黑屏问题总结-CSDN博客Unity Universal…...

[spark] DataFrame 的 checkpoint

在 Apache Spark 中&#xff0c;DataFrame 的 checkpoint 方法用于强制执行一个物理计划并将结果缓存到分布式文件系统&#xff0c;以防止在计算过程中临时数据丢失。这对于长时间运行的计算过程或复杂的转换操作是有用的。 具体来说&#xff0c;checkpoint 方法执行以下操作&…...

flask文件夹列表改进版--Bug追踪

把当前文件夹下的所有文件夹和文件列出来&#xff0c;允许点击返回上层目录&#xff0c;允许点击文件夹进入下级目录并显示此文件夹内容 允许点击文件进行下载 from flask import Flask, render_template, send_file, request, redirect, url_for import osapp Flask(__name_…...

Elasticsearch之常用DSL语句

目录 1. Elasticsearch之常用DSL语句 1.1 操作索引 1.2 文档操作 1.3 DSL查询 1.4 搜索结果处理 1.5 数据聚合 1. Elasticsearch之常用DSL语句 1.1 操作索引 mapping是对索引库中文档的约束&#xff0c;常见的mapping属性包括&#xff1a; - type&#xff1a;字段数据类…...

鸿蒙实战-库的调用(ArkTS)

整体框架搭建 主页面、本地库组件页面、社区库组件页面三个页面组成&#xff0c;主页面由Navigation作为根组件实现全局标题&#xff0c;由Tabs组件实现本地库和社区库页面的切换。 // MainPage.ets import { Outer } from ../view/OuterComponent; import { Inner } from ..…...

观察者模式学习

观察者模式&#xff08;Observer Design Pattern&#xff09;也被称为发布订阅模式&#xff08;Publish-Subscribe Design Pattern&#xff09;。在 GoF 的《设计模式》一书中&#xff0c;它的定义是这样的&#xff1a; Define a one-to-many dependency between objects so th…...

人工智能_机器学习078_聚类算法_概念介绍_聚类升维_降维_各类聚类算法_有监督机器学习_无监督机器学习---人工智能工作笔记0118

首先看一下什么是聚类,我们可以进入sklearn的官网去看看 可以看到这里,首先classification 这个分类我们学完了,然后就是regression回归我们也学完了对吧,其实我们现实生活中的,大部分问题就是 这两种问题就可以解决了. 然后我们再来看一个: clustering,这个就是聚类对吧.聚类算…...

基于AR+地图导航的景区智慧导览设计

随着科技的飞速发展&#xff0c;智慧旅游已经成为现代旅游业的一个重要趋势。在这个背景下&#xff0c;景区智慧导览作为智慧旅游的核心组成部分&#xff0c;正逐渐受到越来越多游客的青睐。本文将深入探讨地图导航软件在景区智慧导览中的应用&#xff0c;并分析其为游客和景区…...

涡轮发动机建模这事听起来玄乎,实际拆开了看也就是搭积木。咱今天用Simulink搞个能喘气的发动机模型,重点在让这铁疙瘩遇到负载突变时还能稳住转速不抖腿

【空气涡轮发动机Matlab/simulink动态仿真模型】 1、部件级模型;进气道&#xff0c;涡轮&#xff0c;气室&#xff0c;压气机&#xff0c;尾喷管&#xff0c;转子动力学模块&#xff0c;容积模块 2、PID控制器: 输出扭矩阶跃扰动下&#xff0c;维持转速恒定先给各个部件开户口簿…...

从SEED-Labs实验到实战:手把手教你编写无零字节的x86 Shellcode(附完整代码)

从SEED-Labs实验到实战&#xff1a;手把手教你编写无零字节的x86 Shellcode&#xff08;附完整代码&#xff09; 当你第一次看到"Shellcode"这个词时&#xff0c;可能会联想到某种神秘的编程黑魔法。实际上&#xff0c;它是安全研究中最具实用价值的技能之一——一段…...

以采购管理系统为例,构建多角色AI智能体协作系统

成果演示&#xff08;基于 Trae Solo&#xff09; 1.构建智能体Trae Solo 支持智能生成智能体&#xff0c;输入角色及职能描述&#xff0c;即可得到角色智能体。在此构建需求分析智能体、架构设计智能体、前端智能体、后端智能体进行演示。2.创建任务 本文依照需求分析、架构设…...

阅读APP书源完全指南:3种快速导入方法与问题解决方案

阅读APP书源完全指南&#xff1a;3种快速导入方法与问题解决方案 【免费下载链接】Yuedu &#x1f4da;「阅读」自用书源分享 项目地址: https://gitcode.com/gh_mirrors/yu/Yuedu 「阅读」APP书源开源项目为小说爱好者提供了一个强大的解决方案&#xff0c;让您能够在一…...

揭秘AI教材写作:低查重率,用AI轻松搞定教材编写难题!

编写教材需要丰富的资料作为基础&#xff0c;但传统的资料整理方式早已无法满足当前的需求。过去&#xff0c;课标文件、学术论文和教学案例等信息零散地散落在多个平台上&#xff0c;比如知网和教研网站&#xff0c;这让我们花费数天才筛选到有用的内容。而即使所有资料都收集…...

GraphRAG实战:我是如何用它分析公司内部文档,让客服响应时间缩短近30%的

GraphRAG实战&#xff1a;我是如何用它分析公司内部文档&#xff0c;让客服响应时间缩短近30%的 作为一家中型电商企业的技术负责人&#xff0c;我最近半年一直在与客服团队的一个顽固问题搏斗&#xff1a;每当新品上线或促销活动期间&#xff0c;客服人员需要花费大量时间在不…...

告别单调闪烁!用GD32F303的TIMER高级功能玩转PWM:实现S形曲线呼吸灯与多灯同步效果

解锁GD32F303定时器高阶玩法&#xff1a;S形曲线PWM与多灯协同控制艺术 呼吸灯效果在嵌入式设备中早已司空见惯&#xff0c;但大多数实现仍停留在简单的线性渐变阶段。当LED亮度以恒定速率变化时&#xff0c;人眼会感知到明显的"机械感"——就像早期数字音乐缺少模拟…...

G-Helper终极指南:如何免费优化华硕笔记本性能,告别Armoury Crate臃肿体验

G-Helper终极指南&#xff1a;如何免费优化华硕笔记本性能&#xff0c;告别Armoury Crate臃肿体验 【免费下载链接】g-helper Lightweight, open-source control tool for ASUS laptops and ROG Ally. Manage performance modes, fans, GPU, battery, and RGB lighting across …...

前端Word文档生成革命:3分钟掌握纯JavaScript专业文档导出创新方案

前端Word文档生成革命&#xff1a;3分钟掌握纯JavaScript专业文档导出创新方案 【免费下载链接】DOCX.js Generate Microsoft Word DOCX files in pure client-side JavaScript. Try in Chrome 项目地址: https://gitcode.com/gh_mirrors/do/DOCX.js 还在为Word文档导出…...

2026届最火的五大降AI率助手推荐

Ai论文网站排名&#xff08;开题报告、文献综述、降aigc率、降重综合对比&#xff09; TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 减低AIGC检测率要从多方面入手&#xff0c;首先&#xff0c;调整句式结构&#xff0c;避开过…...