【前端】JavaScript中的柯里化(Currying)详解及实现
文章目录
- 💯前言
- 💯什么是柯里化?
- 💯柯里化的特点
- 💯柯里化的简单示例
- 💯通用的柯里化实现
- 💯柯里化让代码更易读的原因
- 💯柯里化是否总是更易读?
- 💯柯里化的实现思路
- 💯减少 `return` 的场景
- 💯柯里化的实际用途
- 💯柯里化的缺点
- 💯总结
💯前言
- 在 JavaScript 编程中,函数是极为灵活而强大的工具。近年来,函数式编程风格的流行逐渐改变了开发者对代码结构和设计模式的理解。在函数式编程的诸多技术中,柯里化(Currying)占据了一个十分重要的位置。
对于那些希望编写简洁、优雅、可重用代码的开发者来说,柯里化无疑是一个值得深入研究的概念。本文旨在全面阐述柯里化的理论基础、实现方式
、实际应用场景及其潜在的优势与局限性,以期为读者提供系统化的认知。
JavaScript
💯什么是柯里化?
柯里化是一种将接收多个参数的函数转换为一系列只接收单一参数的函数的技术。这种技术在函数式编程中极为常见,其核心思想是将多参数函数拆分为多个一元函数,使得每次调用时仅需处理一个参数,从而逐步构建出完整的结果。通过这样逐层的拆解与组合,柯里化有效地降低了函数的复杂性,并且提高了函数复用的灵活性。
简单来说,假如我们有一个多参数函数 f(a, b, c)
,经过柯里化之后,它会被转换为 f(a)(b)(c)
的形式。这意味着,每次函数调用仅接受一个参数并返回一个新的函数,直到最终所有的参数都被提供并得到结果。这种逐次处理参数的过程不仅是代码层面的优化,更是一种思维方式上的进步。
柯里化不仅是一种函数转换的技术,它更是一种对函数调用逻辑的重新架构和思考方式。通过逐步处理每个参数,柯里化实现了函数的高度灵活性和精细控制。在实际开发中,柯里化常常与高阶函数
结合使用,以实现复杂逻辑的灵活组合,从而构建出更为优雅和可维护的代码架构。其应用场景涵盖了从简单的数学运算到复杂的事件处理,广泛的实践证明了柯里化在函数式编程中的巨大潜力。
💯柯里化的特点
- 逐步传递参数:柯里化的核心在于函数每次只接受一个参数,而不是一次性接受全部参数。这使得函数调用可以逐步进行,有助于简化逻辑和减少出错概率。
- 延迟计算:柯里化允许函数在传入部分参数后不立即执行,而是返回一个新的函数,以便稍后继续接收剩余的参数。这种
延迟计算
机制对提高程序灵活性和响应性极为有利。 - 提高代码复用性:柯里化通过逐步传递参数,可以方便地创建一系列配置相似但略有不同的函数,从而极大地提高了代码复用性。
- 提升代码的可测试性和可维护性:柯里化后的函数往往较小,且专注于处理单一任务。这种单一功能性的函数更容易进行单元测试,也使得代码维护更加简便。
💯柯里化的简单示例
为了更好地理解柯里化,我们来看一个具体的例子。假设我们有一个简单的加法函数:
function add(a, b) {return a + b;
}
这个函数接受两个参数 a
和 b
,直接返回两者的和。如果我们将它柯里化,可以这样做:
function curriedAdd(a) {return function(b) {return a + b;};
}// 调用方式
const addFive = curriedAdd(5); // 返回一个新函数
console.log(addFive(3)); // 输出 8
在这个例子中,curriedAdd
是一个柯里化后的函数,它每次接收一个参数,返回一个新的函数,直到所有参数都被处理完成。通过柯里化,我们可以创建出例如 addFive
这样有特定用途的新函数,使代码变得更加简洁和可复用。
柯里化的这种方式在某些场景下非常有用,例如当你需要生成一系列类似但略有不同的函数时,柯里化可以方便地固定某些参数值,而不必重复编写函数逻辑。这使得代码显得更加优雅,避免了逻辑上的冗余,同时使得代码变得更加模块化。
💯通用的柯里化实现
如果你需要对任意的多参数函数进行柯里化,可以编写一个通用的柯里化函数。下面是一个实现的例子:
function curry(func) {return function curried(...args) {if (args.length >= func.length) {// 如果传入的参数足够,直接调用原始函数return func.apply(this, args);} else {// 否则返回一个函数,等待更多参数return function(...nextArgs) {return curried(...args, ...nextArgs);};}};
}// 示例
function multiply(a, b, c) {return a * b * c;
}const curriedMultiply = curry(multiply);// 调用方式
console.log(curriedMultiply(2)(3)(4)); // 输出 24
console.log(curriedMultiply(2, 3)(4)); // 输出 24
console.log(curriedMultiply(2, 3, 4)); // 输出 24
在这个通用的柯里化函数中,curry(func)
返回一个新的函数 curried
,它会检查当前传递的参数数量是否足够调用原函数 func
。如果参数足够,就调用原函数并返回结果;否则,返回一个新的函数用于接收剩余的参数。
这种通用柯里化实现不仅可以用于简单的加法或者乘法函数,还可以应用于更加复杂的场景,例如事件处理、API 请求等。在这些场景中,柯里化能够有效地分离参数逻辑,使代码结构更具层次性和可读性。
💯柯里化让代码更易读的原因
- 逐步传参,降低复杂度
传统的函数设计往往要求一次性传入所有参数,这对某些场景来说,直接写出所有参数并不是那么直观或易读。例如:
calculate(10, 20, 30);
在这种情况下,开发者需要明确这些参数分别代表什么,这容易导致代码可读性降低。而柯里化后的函数可以逐步传入参数,每一步都非常明确:
calculate(10)(20)(30);
这种方式每次只关注一个参数,使得代码更加清晰,符合逐步解决问题的逻辑,也让每个函数在功能上保持了单一性,从而降低了整体的复杂度。这种形式尤其适用于处理一系列逐步应用的操作,例如数学计算、字符串处理、或分步骤处理的业务逻辑。
- 逻辑分离,提升复用性
柯里化还使得代码的复用性大大增强。通过柯里化后的函数,我们可以生成特定用途的小工具函数。例如:
const add = (a) => (b) => a + b;const addFive = add(5); // 固定 a 为 5
console.log(addFive(10)); // 输出 15
通过柯里化,我们可以轻松地创建具有特定用途的工具函数,这种简化有助于减少代码的重复,提升开发效率,特别是在需要多次调用相似逻辑的场景下。柯里化后的函数往往具备单一职责,便于模块化和单独测试,这种模块化设计使得开发者能够更加专注于每个独立功能模块的实现。
- 更贴近业务逻辑
柯里化可以使代码的调用方式与业务需求更紧密结合。例如,我们可以定义一个用于记录日志的函数:
const log = (level) => (time) => (message) =>console.log(`[${level}] [${time}] ${message}`);const errorLog = log("ERROR");
const nowErrorLog = errorLog(new Date().toISOString());nowErrorLog("Something went wrong!");
在这个例子中,我们逐步设置日志的级别、时间戳,最终传入具体的日志内容。每一步的调用逻辑都非常清晰,这种设计符合人类的思维方式,使得代码的可读性大大增强。
柯里化使得函数调用更加接近自然语言的描述,特别是在需要复杂参数配置的业务逻辑中,可以清晰地表达每个步骤的含义。这样的分步配置方式对于某些高度定制化的功能,诸如日志管理、API 请求设置等,显得尤为高效和灵活。
💯柯里化是否总是更易读?
- 对于简单场景:增加嵌套可能显得多余
对于一些简单的函数,柯里化可能会使代码变得冗长而无益。例如,简单的加法函数直接写成:
function add(a, b) {return a + b;
}
在这种情况下,柯里化的多层嵌套可能显得过于繁琐,不如直接使用普通函数来得更为简洁。在这种情况下,增加额外的嵌套不仅不会带来实质性的好处,还可能使代码更加难以理解。
- 对于复杂场景:柯里化让逻辑更直观
当函数的参数较多,或者需要灵活地部分应用参数时,柯里化可以帮助逐步清晰地拆分逻辑,显著提升代码的可读性和维护性。对于复杂的业务场景,参数的逐步应用能够显著增强代码的可维护性和逻辑清晰度,从而便于团队协作与理解。
💯柯里化的实现思路
假如一个函数有五个参数,柯里化之后,这个函数会被逐层嵌套为四个返回函数,每一层函数依次接收一个参数。例如:
function curriedAdd(a) {return function(b) {return function(c) {return function(d) {return function(e) {return a + b + c + d + e;};};};};
}console.log(curriedAdd(1)(2)(3)(4)(5)); // 输出 15
在这里,每个函数层级都会接收一个参数,直到所有参数都传入完成。这种设计符合柯里化的基本定义,使得每次调用函数时的逻辑都清晰明确。
对于复杂的业务逻辑,通过柯里化,我们可以逐步将一个多参数的问题分解为更小、更简单的部分,使得每一层函数的职责都变得单一且明确,这样的代码更符合“单一职责原则”,有利于代码的复用和单独测试。
💯减少 return
的场景
如果多层嵌套显得过于复杂,可以通过改进的柯里化实现,允许一次性传入多个参数,从而减少显式嵌套:
function curry(func) {return function curried(...args) {if (args.length >= func.length) {return func.apply(this, args);} else {return function(...nextArgs) {return curried(...args, ...nextArgs);};}};
}// 示例
const add = (a, b, c, d, e) => a + b + c + d + e;const curriedAdd = curry(add);console.log(curriedAdd(1, 2)(3)(4, 5)); // 输出 15
console.log(curriedAdd(1)(2)(3, 4, 5)); // 输出 15
这种改进的柯里化方法结合了传统函数和柯里化的优点,让开发者可以灵活选择是逐步传参还是一次性传入多个参数。这样的灵活性在实际开发中尤为重要,因为它让函数的使用更加便捷,适应不同的使用场景。
💯柯里化的实际用途
-
函数复用
- 柯里化允许你生成固定部分参数的新函数,从而极大地提升了函数的复用性。例如:
const log = (level, message) => console.log(`[${level}] ${message}`); const info = log.bind(null, "INFO"); const error = log.bind(null, "ERROR");info("This is an informational message."); error("This is an error message。");
通过预绑定部分参数,我们可以轻松创建一些特定用途的函数,例如
info
和error
,这些函数共享原始的log
逻辑,但具有不同的参数配置。 -
部分应用
- 通过柯里化,你可以生成特定用途的函数。例如:
const fetchWithBaseURL = curry(fetch)("https://api.example.com");fetchWithBaseURL("/users").then(response => response.json()).then(data => console.log(data));
通过柯里化,我们可以创建一个基于特定基 URL 的
fetch
请求函数,使得后续的 API 调用更加简单和清晰,避免了重复代码。 -
事件处理和回调
- 柯里化函数可以提前绑定部分参数,从而减少代码冗余,使代码更加模块化和灵活。例如在事件处理场景中,可以用柯里化将事件和处理逻辑分离:
const addEventListenerCurried = (element) => (event) => (handler) => {element.addEventListener(event, handler); };const button = document.querySelector("#myButton"); const onClick = addEventListenerCurried(button)("click");onClick(() => alert("Button clicked!"));
通过这样的方式,
addEventListenerCurried
将元素、事件类型和处理逻辑分开,使得每一步都非常清晰且容易复用。
💯柯里化的缺点
- 增加嵌套,容易混淆:对于一些初学者来说,多层嵌套的函数写法可能不太容易理解,尤其是嵌套层数较多时。
- 不适用于所有场景:对于简单的函数或不需要逐步传参的场景,柯里化显得
不必要且冗长
。 - 调试复杂度增加:由于柯里化函数返回的是一层层嵌套的函数,调试时可能难以直观地查看所有的调用与参数,这增加了调试的难度。
- 性能开销:在性能敏感的场景下,柯里化可能带来额外的函数调用开销,尤其是在
深度嵌套
的情况下,可能会影响代码的执行效率。
💯总结
柯里化是一种极具威力的函数式编程技术,通过将多参数函数逐步分解为单参数函数,能够有效提升代码的复用性和可读性。在JavaScript
中,柯里化能够帮助开发者更为模块化和清晰地组织代码,特别是当需要部分应用或者分步处理逻辑时,其优势尤为显著。
尽管柯里化并非适用于所有情况,在某些简单场景下可能显得复杂化
,但掌握柯里化的思想能够极大丰富开发者解决问题的方式。当使用柯里化时,关键在于权衡其带来的灵活性与复杂度,根据具体情境选择最适合的代码组织方式。希望本文的详细阐述能够帮助你更好地理解和运用柯里化,提升代码的质量和可维护性。
柯里化不仅是技术上的转变,更是一种思维方式的革新。它使得函数更加灵活且高度可复用,并且让代码逻辑更接近自然语言描述。通过合理应用柯里化,你可以编写更加优雅、扩展性更强且可维护性更高的代码。无论是在前端还是后端开发中,柯里化都为你提供了一种全新的思维范式,帮助你编写出更加简洁、优雅且强大的代码解决方案。
相关文章:
【前端】JavaScript中的柯里化(Currying)详解及实现
博客主页: [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: 前端 文章目录 💯前言💯什么是柯里化?💯柯里化的特点💯柯里化的简单示例💯通用的柯里化实现💯柯里化让代码更易读的原因💯…...
解决 docker 部署 vsftpd 速度慢问题
解决 docker 部署 vsftpd 速度慢问题 Docker 部署 ftp version: 3.8services:ftps:image: fauria/vsftpdcontainer_name: my-ftpsenvironment:- FTP_USERyourusername- FTP_PASSyourpassword- PASV_ADDRESS192.168.0.123 # 使用环境变量或直接指定IP地址- PASV_MIN_PORT4900…...
Java基础夯实——2.9 多线程如何共享数据
在 Java 多线程编程中,共享数据通过以下几种方式实现: 1. 使用共享对象 多个线程可以通过引用同一个对象来实现数据共享。例如: class SharedData {private int count;public synchronized void increment() {count;}public synchronized …...
【Leetcode Top 100】234. 回文链表
问题背景 给你一个单链表的头节点 h e a d head head,请你判断该链表是否为 回文链表(回文 序列是向前和向后读都相同的序列)。如果是,返回 t r u e true true;否则,返回 f a l s e false false。 数据…...
GitLab指定用户分配合并权限
进入项目 -》 Project Settings Repository -》展开 Protected branches -》 添加要保护的分支,设置角色 管理用户角色权限 查看到不同用户的角色,一般设置Developer只有Merger Request权限,Maintainer还有Merge审批权限 GitLab 中的权限…...
五,[GXYCTF2019]Ping Ping Ping1
进入靶场,有提示 我们在url试着输入本地IP,返回了ping命令 既然要在url处传参,那就用postman,再输入ip127.0.0.1 & ls,试着列出目录内容 ok,好像是个脏话,它过滤了空格 试着穿越又看到了脏话࿰…...
基于STM32的智能无人机自主飞行与目标识别系统设计
目录 引言系统需求分析 2.1 功能需求 2.2 硬件需求 2.3 软件需求系统设计 3.1 总体架构 3.2 各模块设计系统实现 4.1 硬件实现 4.2 软件实现系统调试与优化总结与展望 1. 引言 随着无人机技术的快速发展,无人机在军事侦察、环境监测、物流配送等领域的应用逐渐增多…...
C 语言数组与函数:核心要点深度剖析与高效编程秘籍
我的个人主页 我的专栏:C语言,希望能帮助到大家!!!点赞❤ 收藏❤ 目录 引言数组基础 2.1 数组的定义与初始化 2.2 一维数组的基本操作 2.3 二维数组及其应用 2.4 数组与指针的关系函数基础 3.1 函数的定义与调用 3.2…...
汽车轮毂结构分析有哪些?国产3D仿真分析实现静力学+模态分析
本文为CAD芯智库原创,未经允许请勿复制、转载! 之前分享了如何通过国产三维CAD软件如何实现「汽车/汽配行业产品设计」,兼容NX(UG)、Creo(Proe),轻松降低企业上下游图纸交互成本等。…...
解决jupyter notebook 新建或打开.ipynb 报500 : Internal Server Error(涉及jinja2兼容性问题)
报错: [E 10:09:52.362 NotebookApp] 500 GET /notebooks/Untitled16.ipynb?kernel_namepyt hon3 (::1) 93.000000ms refererhttp://localhost:8888/tree ...... 重点是: from .exporters import * File "C:\ProgramData\Anaconda3\lib\site-p…...
【若依ruoyi Vue前端线上个人服务器部署】以及常见报错问题解决
提示:【若依ruoyi Vue前端线上个人服务器部署】以及常见报错问题解决 文章目录 前言一、若依ruoyi Vue前端部署常见两种错误1、404问题2、找不到….模块 二、使用步骤(正式开始)1.修改vue.config.js中的publicPath属性。2.修改router/index.j…...
Python学习第十天--处理CSV文件和JSON数据
CSV:简化的电子表格,被保存为纯文本文件 JSON:是一种数据交换格式,易于人阅读和编写,同时也易于机器解析和生成,以JavaScript源代码的形式将信息保存在纯文本文件中 一、csv模块 CSV文件中的每行代表电…...
python基础(一)
python语言特点 解释型语言代码执行过程中通过解释器将代码转换为机器语言,并立即执行;编译型语言执行前需要经过编译整个代码文件为机器语言的可执行文件,然后执行能找出大部分错误错误处理解释型语言在运行时发现错误,编译型语…...
go-carbon v2.5.0 发布,轻量级、语义化、对开发者友好的 golang 时间处理库
carbon 是一个轻量级、语义化、对开发者友好的 Golang 时间处理库,提供了对时间穿越、时间差值、时间极值、时间判断、星座、星座、农历、儒略日 / 简化儒略日、波斯历 / 伊朗历的支持。 carbon 目前已捐赠给 dromara 开源组织,已被 awesome-go 收录&am…...
守护进程
目录 守护进程 前台进程 后台进程 session(进程会话) 前台任务和后台任务比较好 本质 绘画和终端都关掉了,那些任务仍然在 bash也退了,然后就托孤了 编辑 守护进程化---不想受到任何用户登陆和注销的影响编辑 如何…...
学习日记_20241126_聚类方法(自组织映射Self-Organizing Maps, SOM)
前言 提醒: 文章内容为方便作者自己后日复习与查阅而进行的书写与发布,其中引用内容都会使用链接表明出处(如有侵权问题,请及时联系)。 其中内容多为一次书写,缺少检查与订正,如有问题或其他拓展…...
【接口自动化测试】一文从0到1详解接口测试协议!
接口自动化测试是软件开发过程中重要的环节之一。通过对接口进行测试,可以验证接口的功能和性能,确保系统正常运行。本文将从零开始详细介绍接口测试的协议和规范。 定义接口测试协议 接口测试协议是指用于描述接口测试的规范和约定。它包含了接口的请求…...
安全设备-日志审计-系统安装部署配置
3.1 系统安装部署概述 通过系统初始化安装部署,可实现对系统的基础管理工作。系统安装基本部署涉及功能有时间配置、 资产组、资产、用户组、用户、时间配置等) 3.2 系统安装部署配置举例 3.2.1 用户场景 本阶段进行系统安装,进行相关设…...
【ArcGIS Pro】实现一下完美的坐标点标注
在CAD里利用湘源可以很快点出一个完美的坐标点标注。 但是在ArcGIS Pro中要实现这个效果却并不容易。 虽然有点标题党,这里就尽量在ArcGIS Pro中实现一下。 01 标注实现方法 首先是准备工作,准备一个点要素图层,包含xy坐标字段。 在地图框…...
Unity项目性能优化列表
1、对象池 2、检查内存是否泄露。内存持续上升(闭包、委托造成泄露) 3、检查DrawCall数量,尽量减少SetPassCall 4、尽量多的利用四种合批 动态合批(Dynamic Batching)静态合批(Static Batching)GPUInstancingSRP Batcher 动态合批消耗内存把多个网格组合在一起合并…...
【系统架构设计师】高分论文:论软件架构的生命周期
更多内容请见: 备考系统架构设计师-专栏介绍和目录 文章目录 摘要正文摘要 2022 年5月,本人所在的某集团公司承接了财务共享服务平台综合管理系统的项目开发,该项目主要实现财务系统主流业务的集成共享。本人担任项目组成员中的系统架构设计师一职,全面负责项目的全生命周…...
流量控制和拥塞控制的区别
流量控制和拥塞控制是TCP协议中两个重要的机制,它们分别用于解决不同的问题。 流量控制 流量控制的目的是防止发送方发送数据过快,导致接收方来不及接收,从而避免分组丢失。流量控制是通过滑动窗口机制实现的,接收方在返回的ACK…...
CSS 背景、阴影和混合模式
网站的好坏在于细节,在实现页面里某个组件的布局并写完样式之后,不要急着继续,有意识地训练自己,以挑剔的眼光审视刚刚完成的代码。 1 背景与渐变 background-image 指定文件或者生成的颜色渐变为背景图片。 background-origin…...
第49届ICPC亚洲区域赛,非凸科技再次支持上海赛站
11月16日-17日,第49届ICPC国际大学生程序设计竞赛亚洲区域赛上海站在上海大学宝山校区成功举办,来自全国各地222所高校、中学、企业的352支参赛队伍同台竞技。非凸科技高度重视ICPC竞赛,再次荣膺上海赛站合作伙伴,共同推动全球信息…...
良好的并发编程习惯之封闭(Confinement)
创作内容丰富的干货文章很费心力,感谢点过此文章的读者,点一个关注鼓励一下作者,激励他分享更多的精彩好文,谢谢大家! “共享可变状态”有两个要点:“共享”和“可变”。封闭的策略是:不共享就完…...
docker镜像、容器、仓库介绍
docker docker介绍docker镜像命令docker容器命令docker仓库 docker介绍 官网 Docker 是一种开源的容器化平台,用于开发、部署和运行应用。它通过将应用程序及其依赖项打包到称为“容器”的单一包中,使得应用能够在任何环境下运行,不受底层系…...
写个添加球队和展示球队的功能--laravel与inertia
先展示下最终效果,如下是展示球队的界面 如下是添加球队的界面 界面样式没怎么调整,不要在意这些细节。先说说操作流程 首先需要登录,没注册就注册一个账号。登录界面就不展示了。然后选中”NbaBasketballTeams“这个选项,就进入了展示球队的界面。然后点击…...
自制Windows系统(十)
上图 (真的不是Windows破解版) 开源地址:仿Windows...
World of Warcraft /script SetRaidTarget(“target“, n, ““) n=8,7,6,5,4,3,2,1,0
魔兽世界执行当前目标标记方法 /script SetRaidTarget("target", n, "") n8,7,6,5,4,3,2,1,0 解析这个lua脚本 D:\Battle.net\World of Warcraft\_classic_\Interface\AddOns\wMarker wMarker.lua /script SetRaidTarget("target", 8, &quo…...
Rust中Tracing 应用指南
欢迎来到这篇全面的Rust跟踪入门指南。Rust 的tracing是一个用于应用程序级别的诊断和调试的库。它提供了一种结构化的、异步感知的方式来记录日志和跟踪事件。与传统的日志记录相比,tracing能够更好地处理复杂的异步系统和分布式系统中的事件跟踪,帮助开…...
教育网站建设情况报告/产品品牌策划方案
问题你需要执行简单的时间转换,比如天到秒,小时到分钟等的转换。解决方案为了执行不同时间单位的转换和计算,请使用 datetime 模块。 比如,为了表示一个时间段,可以创建一个 timedelta 实例,就像下面这样&a…...
尼高网站设计公司/香水推广软文
大家好,我是乔戈里。最近有一位小学弟成功去了腾讯,洋哥找他要了一些面试相关的资料,内容涵盖计算机基础、Java、分布式、大厂面经,质量非常高!!!不要再看那些到处拼凑出来的面试题了࿰…...
厦门建设局耿家强/seo链接优化建议
曾经多次我的鼠标都是因为滚轴坏了而作废,我想这也是大部分小伙伴会遇到的问题。最近我的无线鼠标摔了一下,滚轴坏了。这次闲来无事,索性直接拆机,探索探索,看看可以不可以修好。结果还真的被朕修好了ahahah࿰…...
自己做的网站加载慢的原因/百度关键词优化和百度推广
https://www.zhihu.com/question/34183746javaScript原型、原型链的定义?prototype:每个函数都有一个prototype(显式原型),这个属性是一个对象(属性的集合),默认有一个叫做constructor(和_proto_࿰…...
光谷做网站推广公司/绍兴seo
aspx是页面文件ascx是用户控件,用户控件必须嵌入到aspx中才能使用。 ascx是用户控件,相当于模板 其实ascx你可以理解为Html里的一部分代码,只是嵌到aspx里而已,因为aspx内容多的时候实在是不太好管理,而且你把公共的Html部分写成ascx也可以公用在很多asp…...
北京天通苑 做网站/百度排行
error C2664: CreateThread2010-06-24 17:21用VC创建新线程易出的问题error C2664: CreateThread : cannot convert parameter 3 from unsigned long (void *) to unsigned long (__stdcall *)(void *) 收藏 今天使用 CreateThread 方法创建线程遇到问题,后解决&…...