首页 > 其他分享 >GodoOS本地代理实现详解

GodoOS本地代理实现详解

时间:2025-01-13 23:12:54浏览次数:1  
标签:GodoOS http nil err fmt 详解 proxy 本地 Printf

引言

  • 在现代软件开发中,代理服务器扮演着至关重要的角色。它们可以用于负载均衡、请求转发、缓存、安全控制等多种场景。本文将详细介绍 godoos 中的本地代理实现,包括其架构设计、核心功能以及具体的实现细节。

架构设计

核心组件
ProxyServer 结构体:用于存储服务类型和实际服务对象。
FileServer 结构体:用于存储文件静态服务的信息。
sync.Map:用于全局存储代理服务映射,确保线程安全。

主要功能
HTTP 代理:将请求转发到指定的远程服务器。
UDP 代理:实现 UDP 数据包的转发。
文件静态服务:提供静态文件的访问。

核心功能实现

1. 创建本地代理

CreateLocalProxyHandler 函数用于创建一个新的本地代理。它从 HTTP 请求中解析代理配置,并将其存储到数据库中。然后启动相应的代理服务。

func CreateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
	var lp model.LocalProxy
	err := json.NewDecoder(r.Body).Decode(&lp)
	if err != nil {
		http.Error(w, err.Error(), http.StatusBadRequest)
		return
	}

	err = model.Db.Create(&lp).Error
	if err != nil {
		http.Error(w, err.Error(), http.StatusInternalServerError)
		return
	}

	// 启动代理服务
	go startProxy(lp)

	libs.SuccessMsg(w, lp, "")
}

2. 获取本地代理

GetLocalProxiesHandlerGetLocalProxyHandler 函数分别用于获取所有本地代理和单个本地代理的配置信息。

func GetLocalProxiesHandler(w http.ResponseWriter, r *http.Request) {
// 获取查询参数 page 和 limit
pageStr := r.URL.Query().Get("page")
limitStr := r.URL.Query().Get("limit")

page, err := strconv.Atoi(pageStr)
if err != nil || page < 1 {
	page = 1
}

limit, err := strconv.Atoi(limitStr)
if err != nil || limit < 1 {
	limit = 10
}

// 定义响应结构体
type ProxyResponse struct {
	Proxies []model.LocalProxy `json:"proxies"`
	Total   int64              `json:"total"`
}

// 修改处理函数
proxies, total, err := model.GetLocalProxies(page, limit)
if err != nil {
	http.Error(w, err.Error(), http.StatusInternalServerError)
	return
}

libs.SuccessMsg(w, ProxyResponse{Proxies: proxies, Total: total}, "")
}

UpdateLocalProxyHandler 函数用于更新本地代理的配置。更新后,会停止旧的代理服务并启动新的代理服务。

gofunc UpdateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
var lp model.LocalProxy
err := json.NewDecoder(r.Body).Decode(&lp)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}

err = model.Db.Model(&model.LocalProxy{}).Where("id = ?", lp.ID).Updates(lp).Error
if err != nil {
	http.Error(w, err.Error(), http.StatusInternalServerError)
	return
}

// 停止旧的代理服务
stopProxy(lp.ID)

// 启动新的代理服务
go startProxy(lp)

libs.SuccessMsg(w, lp, "")
}

DeleteLocalProxyHandler 函数用于删除本地代理。删除前会停止相应的代理服务。

gofunc DeleteLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
idStr := r.URL.Query().Get("id")
id, err := strconv.Atoi(idStr)
if err != nil {
http.Error(w, "Invalid ID", http.StatusBadRequest)
return
}

// 停止代理服务
stopProxy(uint(id))

err = model.Db.Delete(&model.LocalProxy{}, uint(id)).Error
if err != nil {
	http.Error(w, err.Error(), http.StatusInternalServerError)
	return
}

libs.SuccessMsg(w, nil, "delete proxy success")
}

HandlerSetProxyStatus 函数用于启用或禁用代理服务。根据代理的状态,启动或停止相应的代理服务。

gofunc HandlerSetProxyStatus(w http.ResponseWriter, r *http.Request) {
idStr := r.URL.Query().Get("id")
if idStr == "" {
libs.ErrorMsg(w, "id is empty")
return
}
id, err := strconv.Atoi(idStr)
if err != nil || id == 0 {
libs.ErrorMsg(w, "id is not number")
return
}
var proxy model.LocalProxy
if err := model.Db.First(&proxy, uint(id)).Error; err != nil {
libs.ErrorMsg(w, "proxy not found")
return
}
if err := model.Db.Model(&model.LocalProxy{}).Where("id = ?", proxy.ID).Update("status", !proxy.Status).Error; err != nil {
libs.ErrorMsg(w, "update proxy status failed")
return
}
if !proxy.Status {
startProxy(proxy)
} else {
stopProxy(proxy.ID)
}
libs.SuccessMsg(w, nil, "")
}

启动代理服务

startProxy 函数根据代理的类型启动相应的代理服务。

func startProxy(proxy model.LocalProxy) {
switch proxy.ProxyType {
case "http":
go func(p model.LocalProxy) {
httpProxyHandler(p)
}(proxy)
case "udp":
go func(p model.LocalProxy) {
udpProxyHandler(p)
}(proxy)
case "file":
go func(p model.LocalProxy) {
fileServerHandler(p)
}(proxy)
default:
fmt.Printf("Unknown proxy type: %s\n", proxy.ProxyType)
}
}

stopProxy 函数根据代理的类型停止相应的代理服务。

func stopProxy(id uint) {
if server, ok := proxyServers.Load(id); ok {
proxyServer, ok := server.(ProxyServer)
if !ok {
fmt.Printf("Failed to load proxy server for ID %d\n", id)
return
}

	switch proxyServer.Type {
	case "http":
		httpServer, ok := proxyServer.Server.(*http.Server)
		if ok {
			// 创建一个上下文,用于传递给 server.Shutdown(ctx)
			ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
			defer cancel()
			if err := httpServer.Shutdown(ctx); err != nil {
				fmt.Printf("Failed to shutdown HTTP server on port %s: %v\n", httpServer.Addr, err)
			} else {
				fmt.Printf("Stopped HTTP server on port %s\n", httpServer.Addr)
			}
		}
	case "udp":
		udpConn, ok := proxyServer.Server.(*net.UDPConn)
		if ok {
			udpConn.Close()
			fmt.Printf("Stopped UDP server on port %d\n", udpConn.LocalAddr().(*net.UDPAddr).Port)
		}
	case "file":
		fileServer, ok := proxyServer.Server.(*FileServer)
		if ok {
			// 创建一个上下文,用于传递给 server.Shutdown(ctx)
			ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
			defer cancel()
			if err := fileServer.Server.Shutdown(ctx); err != nil {
				fmt.Printf("Failed to shutdown file server on port %d: %v\n", fileServer.Port, err)
			} else {
				fmt.Printf("Stopped file server on port %d\n", fileServer.Port)
			}
		}
	default:
		fmt.Printf("Unknown proxy type: %s\n", proxyServer.Type)
	}
	proxyServers.Delete(id)
}
}

HTTP 代理处理函数

httpProxyHandler 函数用于处理 HTTP 代理请求,将请求转发到指定的远程服务器。

func httpProxyHandler(proxy model.LocalProxy) {
remote, err := url.Parse(proxy.Domain)
if err != nil {
fmt.Printf("Failed to parse remote URL for port %d: %v\n", proxy.Port, err)
return
}

if remote.Scheme == "" {
	fmt.Printf("Remote URL for port %d does not contain a scheme (http/https): %s\n", proxy.Port, proxy.Domain)
	return
}

reverseProxy := httputil.NewSingleHostReverseProxy(remote)

// 启动 HTTP 服务器并监听指定端口
server := &http.Server{
	Addr:    fmt.Sprintf(":%d", proxy.Port),
	Handler: reverseProxy,
}

proxyServers.Store(proxy.ID, ProxyServer{Type: "http", Server: server})

fmt.Printf("Starting HTTP proxy on port %d and forwarding to %s:%d\n", proxy.Port, proxy.Domain, proxy.Port)
if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
	fmt.Printf("Failed to start HTTP proxy on port %d: %v\n", proxy.Port, err)
}
}

udpProxyHandler 函数用于处理 UDP 代理请求,将数据包转发到指定的远程服务器。

func udpProxyHandler(proxy model.LocalProxy) {
localAddr, err := net.ResolveUDPAddr("udp", fmt.Sprintf(":%d", proxy.Port))
if err != nil {
fmt.Printf("Failed to resolve local UDP address for port %d: %v\n", proxy.Port, err)
return
}

remoteAddr, err := net.ResolveUDPAddr("udp", proxy.Domain)
if err != nil {
	fmt.Printf("Failed to resolve remote UDP address for port %d: %v\n", proxy.Port, err)
	return
}

conn, err := net.ListenUDP("udp", localAddr)
if err != nil {
	fmt.Printf("Failed to listen on UDP port %d: %v\n", proxy.Port, err)
	return
}
defer conn.Close()

proxyServers.Store(proxy.ID, ProxyServer{Type: "udp", Server: conn})

buffer := make([]byte, 1024)
for {
	n, addr, err := conn.ReadFromUDP(buffer)
	if err != nil {
		fmt.Printf("Failed to read from UDP: %v\n", err)
		continue
	}

	remoteConn, err := net.DialUDP("udp", nil, remoteAddr)
	if err != nil {
		fmt.Printf("Failed to dial remote UDP: %v\n", err)
		continue
	}

	_, err = remoteConn.Write(buffer[:n])
	if err != nil {
		fmt.Printf("Failed to write to remote UDP: %v\n", err)
		continue
	}

	// 读取远程服务器的响应并转发回客户端
	n, err = remoteConn.Read(buffer)
	if err != nil {
		fmt.Printf("Failed to read from remote UDP: %v\n", err)
		continue
	}

	_, err = conn.WriteToUDP(buffer[:n], addr)
	if err != nil {
		fmt.Printf("Failed to write to client UDP: %v\n", err)
		continue
	}
}
}

fileServerHandler 函数用于提供静态文件的访问。

func fileServerHandler(proxy model.LocalProxy) {
// 创建文件服务器的 HTTP 处理函数
fileHandler := http.FileServer(http.Dir(proxy.Path))

// 启动 HTTP 服务器并监听指定端口
server := &http.Server{
	Addr:    fmt.Sprintf(":%d", proxy.Port),
	Handler: fileHandler,
}

proxyServers.Store(proxy.ID, ProxyServer{Type: "file", Server: &FileServer{Port: int(proxy.Port), Server: server}})

fmt.Printf("Starting file server on port %d serving files from %s\n", proxy.Port, proxy.Path)
if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
	fmt.Printf("Failed to start file server on port %d: %v\n", proxy.Port, err)
}
}

通过本文的介绍,我们详细地了解了 godoos 中本地代理的实现。godoos 通过灵活的代理类型支持 HTTP、UDP 和文件静态服务,并提供了丰富的管理功能,包括创建、更新、删除和设置代理状态。这种设计使得 godoos 能够适应各种复杂的网络环境和需求。
希望本文能帮助你更好地理解和使用 godoos 中的本地代理功能。如果你有任何问题或建议,欢迎在评论区留言交流。

全部代码

package proxy

import (
	"context"
	"encoding/json"
	"fmt"
	"godo/libs"
	"godo/model"
	"net"
	"net/http"
	"net/http/httputil"
	"net/url"
	"strconv"
	"sync"
	"time"
)

// 全局代理服务映射
var proxyServers sync.Map

// ProxyServer 结构体用于存储服务类型和实际服务对象
type ProxyServer struct {
	Type   string
	Server interface{}
}

// FileServer 结构体用于存储文件静态服务的信息
type FileServer struct {
	Port   int
	Server *http.Server
}

// 创建 LocalProxy 的 HTTP 处理函数
func CreateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
	var lp model.LocalProxy
	err := json.NewDecoder(r.Body).Decode(&lp)
	if err != nil {
		http.Error(w, err.Error(), http.StatusBadRequest)
		return
	}

	err = model.Db.Create(&lp).Error
	if err != nil {
		http.Error(w, err.Error(), http.StatusInternalServerError)
		return
	}

	// 启动代理服务
	go startProxy(lp)

	libs.SuccessMsg(w, lp, "")
}

// GetLocalProxiesHandler 获取所有 LocalProxy 的 HTTP 处理函数
func GetLocalProxiesHandler(w http.ResponseWriter, r *http.Request) {
	// 获取查询参数 page 和 limit
	pageStr := r.URL.Query().Get("page")
	limitStr := r.URL.Query().Get("limit")

	page, err := strconv.Atoi(pageStr)
	if err != nil || page < 1 {
		page = 1
	}

	limit, err := strconv.Atoi(limitStr)
	if err != nil || limit < 1 {
		limit = 10
	}

	// 定义响应结构体
	type ProxyResponse struct {
		Proxies []model.LocalProxy `json:"proxies"`
		Total   int64              `json:"total"`
	}

	// 修改处理函数
	proxies, total, err := model.GetLocalProxies(page, limit)
	if err != nil {
		http.Error(w, err.Error(), http.StatusInternalServerError)
		return
	}

	libs.SuccessMsg(w, ProxyResponse{Proxies: proxies, Total: total}, "")
}

// GetLocalProxyHandler 获取单个 LocalProxy 的 HTTP 处理函数
func GetLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
	idStr := r.URL.Query().Get("id")
	id, err := strconv.Atoi(idStr)
	if err != nil {
		http.Error(w, "Invalid ID", http.StatusBadRequest)
		return
	}
	var proxy model.LocalProxy
	err = model.Db.First(&proxy, uint(id)).Error
	if err != nil {
		http.Error(w, err.Error(), http.StatusNotFound)
		return
	}

	libs.SuccessMsg(w, proxy, "")
}

// UpdateLocalProxyHandler 更新 LocalProxy 的 HTTP 处理函数
func UpdateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
	var lp model.LocalProxy
	err := json.NewDecoder(r.Body).Decode(&lp)
	if err != nil {
		http.Error(w, err.Error(), http.StatusBadRequest)
		return
	}

	err = model.Db.Model(&model.LocalProxy{}).Where("id = ?", lp.ID).Updates(lp).Error
	if err != nil {
		http.Error(w, err.Error(), http.StatusInternalServerError)
		return
	}

	// 停止旧的代理服务
	stopProxy(lp.ID)

	// 启动新的代理服务
	go startProxy(lp)

	libs.SuccessMsg(w, lp, "")
}

// DeleteLocalProxyHandler 删除 LocalProxy 的 HTTP 处理函数
func DeleteLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
	idStr := r.URL.Query().Get("id")
	id, err := strconv.Atoi(idStr)
	if err != nil {
		http.Error(w, "Invalid ID", http.StatusBadRequest)
		return
	}

	// 停止代理服务
	stopProxy(uint(id))

	err = model.Db.Delete(&model.LocalProxy{}, uint(id)).Error
	if err != nil {
		http.Error(w, err.Error(), http.StatusInternalServerError)
		return
	}

	libs.SuccessMsg(w, nil, "delete proxy success")
}
func HandlerSetProxyStatus(w http.ResponseWriter, r *http.Request) {
	idStr := r.URL.Query().Get("id")
	if idStr == "" {
		libs.ErrorMsg(w, "id is empty")
		return
	}
	id, err := strconv.Atoi(idStr)
	if err != nil || id == 0 {
		libs.ErrorMsg(w, "id is not number")
		return
	}
	var proxy model.LocalProxy
	if err := model.Db.First(&proxy, uint(id)).Error; err != nil {
		libs.ErrorMsg(w, "proxy not found")
		return
	}
	if err := model.Db.Model(&model.LocalProxy{}).Where("id = ?", proxy.ID).Update("status", !proxy.Status).Error; err != nil {
		libs.ErrorMsg(w, "update proxy status failed")
		return
	}
	if !proxy.Status {
		startProxy(proxy)
	} else {
		stopProxy(proxy.ID)
	}
	libs.SuccessMsg(w, nil, "")

}

// 初始化代理处理函数
func InitProxyHandlers() {
	go InitFrpcServer()
	proxies, _, err := model.GetLocalProxies(1, 1000) // 获取所有代理配置
	if err != nil {
		fmt.Println("Failed to get local proxies:", err)
		return
	}

	for _, proxy := range proxies {
		go startProxy(proxy)
	}
}

// 启动代理服务
func startProxy(proxy model.LocalProxy) {
	switch proxy.ProxyType {
	case "http":
		go func(p model.LocalProxy) {
			httpProxyHandler(p)
		}(proxy)
	case "udp":
		go func(p model.LocalProxy) {
			udpProxyHandler(p)
		}(proxy)
	case "file":
		go func(p model.LocalProxy) {
			fileServerHandler(p)
		}(proxy)
	default:
		fmt.Printf("Unknown proxy type: %s\n", proxy.ProxyType)
	}
}

// 停止代理服务
func stopProxy(id uint) {
	if server, ok := proxyServers.Load(id); ok {
		proxyServer, ok := server.(ProxyServer)
		if !ok {
			fmt.Printf("Failed to load proxy server for ID %d\n", id)
			return
		}

		switch proxyServer.Type {
		case "http":
			httpServer, ok := proxyServer.Server.(*http.Server)
			if ok {
				// 创建一个上下文,用于传递给 server.Shutdown(ctx)
				ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
				defer cancel()
				if err := httpServer.Shutdown(ctx); err != nil {
					fmt.Printf("Failed to shutdown HTTP server on port %s: %v\n", httpServer.Addr, err)
				} else {
					fmt.Printf("Stopped HTTP server on port %s\n", httpServer.Addr)
				}
			}
		case "udp":
			udpConn, ok := proxyServer.Server.(*net.UDPConn)
			if ok {
				udpConn.Close()
				fmt.Printf("Stopped UDP server on port %d\n", udpConn.LocalAddr().(*net.UDPAddr).Port)
			}
		case "file":
			fileServer, ok := proxyServer.Server.(*FileServer)
			if ok {
				// 创建一个上下文,用于传递给 server.Shutdown(ctx)
				ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
				defer cancel()
				if err := fileServer.Server.Shutdown(ctx); err != nil {
					fmt.Printf("Failed to shutdown file server on port %d: %v\n", fileServer.Port, err)
				} else {
					fmt.Printf("Stopped file server on port %d\n", fileServer.Port)
				}
			}
		default:
			fmt.Printf("Unknown proxy type: %s\n", proxyServer.Type)
		}
		proxyServers.Delete(id)
	}
}

// HTTP 代理处理函数
func httpProxyHandler(proxy model.LocalProxy) {
	remote, err := url.Parse(proxy.Domain)
	if err != nil {
		fmt.Printf("Failed to parse remote URL for port %d: %v\n", proxy.Port, err)
		return
	}

	if remote.Scheme == "" {
		fmt.Printf("Remote URL for port %d does not contain a scheme (http/https): %s\n", proxy.Port, proxy.Domain)
		return
	}

	reverseProxy := httputil.NewSingleHostReverseProxy(remote)

	// 设置请求头
	// reverseProxy.Director = func(req *http.Request) {
	// 	req.Header.Add("X-Forwarded-For", req.RemoteAddr)
	// 	req.Header.Add("X-Real-IP", req.RemoteAddr)
	// 	req.Host = remote.Host
	// }

	// 启动 HTTP 服务器并监听指定端口
	server := &http.Server{
		Addr:    fmt.Sprintf(":%d", proxy.Port),
		Handler: reverseProxy,
	}

	proxyServers.Store(proxy.ID, ProxyServer{Type: "http", Server: server})

	fmt.Printf("Starting HTTP proxy on port %d and forwarding to %s:%d\n", proxy.Port, proxy.Domain, proxy.Port)
	if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
		fmt.Printf("Failed to start HTTP proxy on port %d: %v\n", proxy.Port, err)
	}
}

// UDP 代理处理函数
func udpProxyHandler(proxy model.LocalProxy) {
	localAddr, err := net.ResolveUDPAddr("udp", fmt.Sprintf(":%d", proxy.Port))
	if err != nil {
		fmt.Printf("Failed to resolve local UDP address for port %d: %v\n", proxy.Port, err)
		return
	}

	remoteAddr, err := net.ResolveUDPAddr("udp", proxy.Domain)
	if err != nil {
		fmt.Printf("Failed to resolve remote UDP address for port %d: %v\n", proxy.Port, err)
		return
	}

	conn, err := net.ListenUDP("udp", localAddr)
	if err != nil {
		fmt.Printf("Failed to listen on UDP port %d: %v\n", proxy.Port, err)
		return
	}
	defer conn.Close()

	proxyServers.Store(proxy.ID, ProxyServer{Type: "udp", Server: conn})

	buffer := make([]byte, 1024)
	for {
		n, addr, err := conn.ReadFromUDP(buffer)
		if err != nil {
			fmt.Printf("Failed to read from UDP: %v\n", err)
			continue
		}

		remoteConn, err := net.DialUDP("udp", nil, remoteAddr)
		if err != nil {
			fmt.Printf("Failed to dial remote UDP: %v\n", err)
			continue
		}

		_, err = remoteConn.Write(buffer[:n])
		if err != nil {
			fmt.Printf("Failed to write to remote UDP: %v\n", err)
			continue
		}

		// 读取远程服务器的响应并转发回客户端
		n, err = remoteConn.Read(buffer)
		if err != nil {
			fmt.Printf("Failed to read from remote UDP: %v\n", err)
			continue
		}

		_, err = conn.WriteToUDP(buffer[:n], addr)
		if err != nil {
			fmt.Printf("Failed to write to client UDP: %v\n", err)
			continue
		}
	}
}

// 文件静态服务处理函数
func fileServerHandler(proxy model.LocalProxy) {
	// 创建文件服务器的 HTTP 处理函数
	fileHandler := http.FileServer(http.Dir(proxy.Path))

	// 启动 HTTP 服务器并监听指定端口
	server := &http.Server{
		Addr:    fmt.Sprintf(":%d", proxy.Port),
		Handler: fileHandler,
	}

	proxyServers.Store(proxy.ID, ProxyServer{Type: "file", Server: &FileServer{Port: int(proxy.Port), Server: server}})

	fmt.Printf("Starting file server on port %d serving files from %s\n", proxy.Port, proxy.Path)
	if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
		fmt.Printf("Failed to start file server on port %d: %v\n", proxy.Port, err)
	}
}

标签:GodoOS,http,nil,err,fmt,详解,proxy,本地,Printf
From: https://www.cnblogs.com/xpbb/p/18669591

相关文章

  • GodoOS 知识库实现文件系统监控与同步机制
    引言在软件开发中,文件系统的实时监控和同步是一项关键任务。无论是为了实现增量备份、日志分析还是数据同步,都需要一种高效且可靠的方法来跟踪文件的更改。本文将深入探讨如何使用Go语言及其fsnotify库实现一个强大的文件系统监控和同步系统,并进一步优化其性能和可靠性。......
  • 扩散模型原理详解
    引言        扩散模型(DiffusionModels,DM)是一类基于深度学习的生成模型,其核心思想是通过模拟物理扩散过程,将数据逐步转化为噪声,然后学习逆向过程,从噪声中逐步恢复出原始数据,从而实现高质量的生成效果。扩散模型在图像生成、语音合成、文本生成等多个领域取得了显著......
  • 详解Redis的Set类型及相关命令
    目录SADDSMEMEBERSSISMEMBERSCARDSPOPSMOVESREMSINTERSINTERSTORESUNIONSUNIONSTORESDIFFSDIFFSTORE内部编码应用场景集合类型是保存多个字符串类型的元素的,但和列表类型不同的是,集合中元素之间是⽆序的,且元素不允许重复。⼀个集合中最多可以存储个元素。......
  • React中Element&Fiber对象、WorkInProgress双缓存、Reconcile&Render&Commit、第一次
    基础概念Element对象与Fiber对象Element对象与Fiber对象Element对象定义React的Element对象是一个描述用户界面(UI)的普通JavaScript对象,通常由React.createElement或JSX语法生成。作用它是React应用中的一种描述UI的不可变数据结构。表示一个虚拟DOM......
  • 360度测评系统详解
    360度测评系统详解作者:开源测评系统FreeHR360,https://github.com/FreeHrTools/FreeHR360360度测评,又称360度绩效评估或全方位评估,是一种综合性的测评方法,旨在通过多元化的反馈机制,全面、客观地评估一个人的工作表现与能力发展。以下是对360度测评的详细解析:一、定义与起源3......
  • 为你的Blazor程序加入本地化多语言功能
    本地化本地化是为给定语言和地区定制应用程序的过程.BootstrapBlazor组件允许您将其UI元素转换为所需的语言。这包括按钮、过滤器操作符属性等文本。组件内部默认使用当前请求UI文化语言,本文将向您展示如何在应用程序中使用此功能:BootstrapBlazor组件库简介BootstrapBl......
  • 【自动化测试】—— Appium安装配置保姆教程(图文详解)
    目录一.环境准备二.JDK安装1.JDK下载2.JDK安装3.配置环境4.验证安装三.AndroidSDK安装1.下载AndroidSDK2.安装AndroidSDK3.安装工具4.配置环境5.验证安装四.NodeJS安装1.下载NodeJS2.安装NodeJS3.验证安装4.安装淘宝镜像五.Python及依......
  • 【详解】Hadoop执行start-all.sh时namenode没有启动
    目录Hadoop执行start-all.sh时namenode没有启动1.检查Hadoop配置文件1.1core-site.xml1.2hdfs-site.xml1.3yarn-site.xml1.4mapred-site.xml2.检查日志文件2.1查看NameNode日志2.2常见错误及解决方法2.2.1端口被占用2.2.2文件系统不一致3.检查环境变量......
  • 如何本地部署Fiora聊天应用?随时随地享受远程在线聊天的便捷与乐趣
    文章目录前言1.关于Fiora2.安装Docker3.本地部署Fiora4.使用Fiora5.cpolar内网穿透工具安装6.创建远程连接公网地址7.固定UptimeKuma公网地址前言本文主要介绍如何在本地服务器使用Docker部署开源在线聊天工具Fiora,结合cpolar内网穿透工具轻松打造你的专属社交网络......
  • 【从零开始系列】Qwen2.5 & Llama-Factory:开源语言大模型+训练平台——(超详细、最新版
    目录一、简介        1.Qwen2.5:开源模型        2.LLaMA-Factory:微调工具二、环境搭建       1.Python和Pytorch版本        2.llamafactory项目克隆安装       3.其他重要库安装三、模型微调       1.预训练模......