JSPM

  • ESM via JSPM
  • ES Module Entrypoint
  • Export Map
  • Keywords
  • License
  • Repository URL
  • TypeScript Types
  • README
  • Created
  • Published
  • Downloads 499
  • Score
    100M100P100Q71062F
  • License MIT

一个简单实用且轻量级的 JavaScript 工具库

Package Exports

  • vuekit-js
  • vuekit-js/dist/index.min.js

This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (vuekit-js) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.

Readme

一个简单实用且轻量级的 JavaScript 工具库

如果在使用过程中有任何疑问或建议,请随时通过 longchenxu@126.com 与我联系。

工具库介绍:

  vuekit-js 精心收录了一系列在实际开发场景中千锤百炼的 JavaScript 方法。
  这些方法涵盖了数据处理、DOM 操作、网络请求、浏览器兼容性处理等多个开发中常见的领域。
  无论是复杂的数据清洗与转换,还是繁琐的 DOM 元素操作;无论是便捷的网络请求封装,
  还是巧妙的浏览器差异适配,vuekit-js 都能提供简洁、高效的解决方案。

  它以 "轻量级" 为核心设计理念之一,整个库体积小巧,不会给项目带来额外的负担,却能发挥强大的作用。
  同时,"简单实用" 贯穿于每一个方法的设计与实现中。
  清晰的 API 设计,让开发者无需花费大量时间学习复杂的使用规则,能够快速上手,将更多的精力投入到业务逻辑的实现上。​

  vuekit-js 是开发者在实际开发中的得力伙伴,它凭借着对实际问题的深刻理解和针对性的解决方案。
  帮助开发者解决各种各样的开发难题,提升开发效率,让开发过程更加顺畅和愉悦。

为什么选择 vuekit-js ?

VueKit-JS 是一个面向 Vue/react 应用场景优化的 JavaScript 工具库,通过封装日常开发中常用的函数,让 JavaScript 的使用更高效、轻松。
VueKit-JS 的模块化方法非常适用于:
深度处理数组、对象、字符串和日期等数据
优雅实现节流、防抖、剪贴板、大整数运算等操作
快速获取浏览器信息、滚动位置、元素位置等兼容工具
编写更简洁、语义更明确的函数逻辑

Using npm:

 npm i vuekit-js

Using in Vue. and Using in React.

// 示例:
import { freezeObjectProperties } from 'vuekit-js';

1. deepClone 深度克隆方法

let obj1 = {a: 10, b: 20};
let obj2 = obj1;
console.log(obj1 === obj2); // true

//---------------------------------------------------

let obj3 = {a: 10, b: 20};
let obj4 = deepClone(obj3);
console.log(obj3 === obj4); // false

2. freezeObjectProperties 冻结对象某个属性

// 方法简介: 在vue2项目中由于响应式设计上有一定的缺陷,实际开发中如果使用不当会造成一定程度的性能浪费

/* 举个栗子: 我们在渲染表格数据的时候,往往真正用到响应式数据的一般就那么几个,而后端返回的表格数据通常会有很多,如果不做处理vue就会默认这些数据都需要响应式,这就
造成了不必要的性能浪费,数据量小还好,如果数据量过大会造成一些资源浪费和性能问题。
*/

<tl-table :data='tableData'></tl-table>

data() {
    return {
        tableData: [],
        obj: {}
    }
}

// 使用方法: 
// 1. 数组
let arrayList = [{a: 10, b: 20, c: 30, ...}, {a: 10, b: 20, c: 30, ...}, {a: 10, b: 20, c: 30, ...}]; // 假设这是后端返回的表格数据

// 那么在赋值之前 this.tableData = arrayList; 先调用一个 freezeObjectProperties 这个方法,告诉它那个key需要保留响应式即可
let temp1 = freezeObjectProperties(arrayList, ['c']); // 比如我只想让 c 这个字段拥有响应式,其他都不需要

this.tableData = temp1; // 优化完成 这些数组里面 每个对象里面只有 c 保持响应式


// 2. 对象
let object = {a: 10, b: 20, c: 30};
let temp2 = freezeObjectProperties(object, ['c']);
this.obj = temp2;


// 3. 冻结全部
// 如果想要冻结整个对象,那么第二个参数不传即可
let temp3 = freezeObjectProperties(arrayList);
this.tableData = temp3; // 那么整个对象都是冻结状态

3. flatArray 用于扁平化一个多维数组

// 在项目中我们通常会对一些多维数据进行扁平化处理,那么这个时候就需要请出flatArray方法了。

// 使用方法:
// 这个方法一共包含两个参数,参数1必传并且必须是一个数组,参数2是可选,它代表你要扁平子级的key名字 默认是 children
flatArray(参数1,参数2)

let arr = [{a: 10, list: []}, {a: 10, list: [{a: 10, list: [{a: 10, list: []}]}]}]; // 比如这是你要扁平的数据

let temp = flatArray(arr, 'list');

4. createRandomId 得到一个随机唯一id

// 使用方法:
let id1 = createRandomId(); // 随机id且唯一
let id2 = createRandomId(); // 随机id且唯一
let id3 = createRandomId(); // 随机id且唯一
let id4 = createRandomId(); // 随机id且唯一
let id5 = createRandomId(); // 随机id且唯一
let id6 = createRandomId(); // 随机id且唯一

for(let i of []) {
    i.id = createRandomId();
}

5. debounce 和 advanceDebounce 基础函数防抖 和 高级版函数防抖

// 使用方法
// debounce(参数1, 参数2, 餐数3); // 参数1 要防抖的函数,参数2等待的时间(毫秒), 参数3 是否立即执行(可不传默认false)
// advanceDebounce(参数1, 参数2, 餐数3); // 参数1 要防抖的函数,参数2等待的时间(毫秒), 参数3 是否立即执行(可不传默认false) 高级带有 带取消和flush功能


<el-input v-model='value' @input='searchInput'></el-input>

data() {
    return {
        value: '',
        searchInput: ''
    }
}

created() {
    // 基础版
    this.searchInput = debounce(() => {
        // 函数体
    }, 500)

    // 高级版 带取消和flush功能 可以更加灵活的控制防抖行为
    this.searchInput = advanceDebounce(() => {
        // 函数体
    }, 500)

    this.searchInput.cancel(); // 取消
    this.searchInput.flush(); // flush
}

6. throttle 和 throttleOut 和 advanceThrottle 节流函数

// 1. 使用方法  基础版 throttle
<el-input v-model='value' @input='searchInput'></el-input>

data() {
    return {
        value: '',
        searchInput: ''
    }
}

created() {
    // 在单位时间内只执行一次
    this.searchInput = throttle(() => {
        // 函数体
    }, 800);
}


// 2. 综合版 throttleOut 定时器方式 + 最后一次一定执行方式
<el-input v-model='value' @input='searchInput'></el-input>

data() {
    return {
        value: '',
        searchInput: ''
    }
}

created() {
    // 定时器方式 + 最后一次一定执行方式
    this.searchInput = throttleOut(() => {
        // 函数体
    }, 800);
}


// 3. 进阶综合版 advanceThrottle 立即执行 + 最后一次一定执行方式
<el-input v-model='value' @input='searchInput'></el-input>

data() {
    return {
        value: '',
        searchInput: ''
    }
}

created() {
    // 立即执行 + 最后一次一定执行方式
    this.searchInput = advanceThrottle(() => {
        // 函数体
    }, 800);
}

7. _getPathValue 访问深层对象属性

// 使用方法: 在低版本js中不支持可选连操作(?.) , 如果要访问深层对象又要保证不报错,写出来的代码十分臃肿一堆 a && a.b && a.b.c......使代码十分不容易维护

let obj = {
    a: {
        b: {
            c: {
                d: {
                    f: {
                        name:"你好111"
                    }
                }
            }
        }
    }
};

方法1: console.log(obj.a.b.c.d.f.name); // 不推荐 没有容错判断

方法2: console.log(obj && obj.a && obj.a.b &&  obj.a.b.c.....); // 太沉重 不推荐

方法3: try { console.log(obj.a.b.c.d.f.name); } catch(e) { console.log('默认值') }; // 效率差 不优雅 极为不推荐

方法4: console.log(obj?.a?.b?.c?.d?.f?.name); // 可选连操作 高版本才能用 (高版本推荐)

方法5: console.log(_getPathValue(obj, 'a.b.c.d.f.name', '这是默认值')); // 低版本js 推荐使用

8. _numberfilter 数值转换

// 使用方法:
在一些业务场景中需要把小写数字转换成大写数字

console.log(_numberfilter('520')); // 五百二十
console.log(_numberfilter('521')); // 五百二十一
console.log(_numberfilter('1314')); // 一千三百十四
console.log(_numberfilter('666')); // 六百六十六

9. _moneyfilter 数字金额读法

// 使用方法:
console.log(_moneyfilter('1314.56'));    // 壹仟叁佰壹拾肆元伍角陆分
console.log(_moneyfilter('1001001.00')); // 壹佰万零壹仟零壹元整
console.log(_moneyfilter(1002003004.6)); // 壹拾亿零贰佰万零叁仟零肆元陆角
console.log(_moneyfilter('0'));          // 零元整
console.log(_moneyfilter('-300.75'));    // 负叁佰元柒角伍分
console.log(_moneyfilter('00123.45'));   // 壹佰贰拾叁元肆角伍分
console.log(_moneyfilter('abc123'));     // 金额错误
console.log(_moneyfilter(null));         // 金额错误
console.log(_moneyfilter('99999999999999.99')); // 玖拾玖兆玖仟玖佰玖拾玖亿玖仟玖佰玖拾玖万玖仟玖佰玖拾玖元玖角玖分

10. _pick 从源对象中选择指定字段构成新对象

// 使用方法
let obj = {
    a: 10,
    b: 20,
    c: 30
};

console.log(_pick(obj, ['c']));
打印结果: {
    "c": 30
}

11. deepFindTreeNodeDom 此方法针对的是 element-ui 中的 el-tree 节点树

// 使用方法
// 在使用el-tree 节点树的时候很多场景就是 默认选中节点树第一个节点并点击
// 那么这个方法就可以很好的帮助你

// 注意: 在使用此方法之前  el-tree 必须添加 default-expand-all 这个属性 否则不生效
<el-tree :data='treeData' ref='tree' default-expand-all></el-tree>

this.$nextTick(() => {
    let ele = deepFindTreeNodeDom(this.$refs.tree.$el);
    ele && ele.click();
})

12. _isEmpty 判断值是否为空 包含 null 、undefined、空数组、空对象、空字符串

// 使用示例 返回 true 表示为空
console.log(_isEmpty([])); // true
console.log(_isEmpty([1])); // false
console.log(_isEmpty({})); // true
console.log(_isEmpty({a: 10})); // false
console.log(_isEmpty([null])); // true
console.log(_isEmpty([undefined])); // true

13. 关于浏览所有方法 all

// 1. 获取页面滚动高度(兼容各种浏览器)
getScrollTop();

// 2. 获取视口宽高(兼容)
getViewportSize();

// 3. 注册事件监听器(兼容 IE)
addEvent(element, type, handler);

// 4. 移除事件监听器(兼容 IE)
removeEvent(element, type, handler)

// 5.  获取元素样式(兼容 IE)
getStyle(element, prop);

// 6. requestAnimationFrame 兼容处理
requestAnimFrame();

// 7. cancelAnimationFrame 兼容处理
cancelAnimFrame();

// 8. 判断浏览器是否支持某个 API / 属性
isSupport(feature);

// 9. 判断是否支持 ES6 箭头函数(间接判断 ES6)
isES6Supported();

// 10. 获取浏览器信息(简单版本)
getBrowserInfo();

// 11. 判断是否是移动端设备
isMobileDevice();

// 12.判断当前是否为触摸设备
isTouchDevice();

// 13. 获取兼容的全屏 API
requestFullscreen(el);

// 14. 退出全屏
exitFullscreen();

// 15. 判断是否处于全屏状态
isFullscreen();

14. 格式化时间 formatDate()(自定义格式)

formatDate();
// 结果示例:2025-05-21 14:30:15

formatDate(new Date(), 'YYYY/MM/DD');
// 结果示例:2025/05/21

formatDate(new Date(), 'HH:mm:ss');
// 结果示例:14:30:15

formatDate(new Date(), 'YYYY年MM月DD日 HH时mm分ss秒');
// 结果示例:2025年05月21日 14时30分15秒

const myDate = new Date('2023-12-01T08:05:09');
formatDate(myDate, 'YYYY-MM-DD HH:mm:ss');
// 结果:2023-12-01 08:05:09

15. 只执行一次的函数 once()

// 1. 
function sayHello(name) {
  console.log('Hello, ' + name);
  return 'Greeted ' + name;
}

const greetOnce = once(sayHello);

console.log(greetOnce('Alice')); // 输出:Hello, Alice\nGreeted Alice
console.log(greetOnce('Bob'));   // 无输出,返回:Greeted Alice

// 2. 
function initializeApp() {
  console.log('App initialized');
  return true;
}

const init = once(initializeApp);

init(); // 输出:App initialized
init(); // 无输出
init(); // 依然无输出

// 3. 
function fetchData() {
  console.log('Fetching data from server...');
  return { success: true, data: [1, 2, 3] };
}

const fetchOnce = once(fetchData);

console.log(fetchOnce()); // 输出:Fetching data from server...,返回数据
console.log(fetchOnce()); // 不再输出日志,返回第一次的结果


// 4. 
function add(a, b) {
  return a + b;
}

const addOnce = once(add);

console.log(addOnce(3, 4)); // 输出:7
console.log(addOnce(5, 6)); // 输出:7(第二次调用不生效,仍返回第一次结果)

16. 复制文本到剪贴板 copyToClipboard()

// 1. 
<button onclick="copyToClipboard('Hello World!')">复制文字</button>

// 2. 
<input id="myInput" value="这是一段测试内容" />
<button onclick="copyToClipboard(document.getElementById('myInput').value)">复制输入内容</button>

// 3. 
async function handleCopy() {
  try {
    await copyToClipboard('复制成功的内容');
    alert('内容已复制到剪贴板');
  } catch (err) {
    alert('复制失败:' + err.message);
  }
}

document.getElementById('copyBtn').addEventListener('click', handleCopy);

// 4. 
<template>
  <button @click="copyText">复制 Vue 中的内容</button>
</template>

<script>
import { copyToClipboard } from './utils/clipboard'; // 假设你放在 utils 文件夹里

export default {
  methods: {
    async copyText() {
      try {
        await copyToClipboard('来自 Vue 的问候');
        this.$toast('复制成功!'); // 假设你有 toast 插件
      } catch (err) {
        console.error('复制失败', err);
      }
    }
  }
}
</script>

17. 数组按条件分组 groupBy()

// 1. 示例 1:按对象属性分组
const users = [
  { name: 'Alice', age: 20 },
  { name: 'Bob', age: 25 },
  { name: 'Charlie', age: 20 },
];

const groupedByAge = groupBy(users, 'age');
console.log(groupedByAge);

//  输出
{
  20: [
    { name: 'Alice', age: 20 },
    { name: 'Charlie', age: 20 }
  ],
  25: [
    { name: 'Bob', age: 25 }
  ]
}

// 2. 按字符串长度分组
const words = ['one', 'two', 'three', 'four', 'five'];

const groupedByLength = groupBy(words, word => word.length);
console.log(groupedByLength);

// 输出
{
  3: ['one', 'two'],
  5: ['three'],
  4: ['four', 'five']
}


// 3. 按数字是否为奇偶数分组
const numbers = [1, 2, 3, 4, 5, 6];

const groupedByEvenOdd = groupBy(numbers, n => (n % 2 === 0 ? 'even' : 'odd'));
console.log(groupedByEvenOdd);

// 输出
{
  odd: [1, 3, 5],
  even: [2, 4, 6]
}

// 4. 按首字母分组
const names = ['Alice', 'Adam', 'Bob', 'Brian', 'Charlie'];

const groupedByFirstLetter = groupBy(names, name => name[0]);
console.log(groupedByFirstLetter);


// 输出
{
  A: ['Alice', 'Adam'],
  B: ['Bob', 'Brian'],
  C: ['Charlie']
}

17. 下载文件 downloadFile()

// 1. 下载远程图片
downloadFile('https://example.com/image.png', 'example.png');

// 2. 下载 Blob 文件(如生成的 PDF)
const blob = new Blob(['Hello world!'], { type: 'text/plain' });
downloadFile(blob, 'hello.txt');

// 下载 Base64 文件
const base64Data = 'data:text/plain;base64,SGVsbG8gd29ybGQh';
downloadFile(base64Data, 'hello.txt');

18 . 通用数组交集 deepIntersection(arr1, arr2) 和 deepDifference(arr1, arr2)

const a = { id: 1 };
const b = { id: 2 };
const c = { id: 3 };
const d = { id: 1 };

const arr1 = [1, 'a', a, b, [1, 2], { id: 3 }];
const arr2 = ['a', 2, d, c, [1, 2]];

console.log('交集:', deepIntersection(arr1, arr2));
console.log('差集:', deepDifference(arr1, arr2));

19. 通用的“首字母大写”函数

// 标准版:单个单词首字母大写
capitalize("hello WORLD");           // "Hello"

// 扩展版:每个单词首字母都大写(Title Case)
capitalizeWords("hello WORLD test"); // "Hello World Test"

// 高级版:支持 Unicode、标点、特殊分隔符
smartCapitalize("l'été d'amour");    // "L'Été D'Amour"
smartCapitalize("hello-world");      // "Hello-World"

20. 是否为 URL

isValidUrl('https://example.com');           // true
isValidUrl('http://localhost:3000/test');    // true
isValidUrl('ftp://192.168.0.1/file.txt');    // true
isValidUrl('/relative/path');                // false
isValidUrl('/relative/path', { allowRelative: true }); // true
isValidUrl('example.com');                   // false
isValidUrl('//example.com', { requireProtocol: false }); // true


| 功能                 | 默认行为 | 可配置                     |
| ------------------ | ---- | ----------------------- |
| 必须包含协议             || 可禁用 `requireProtocol` |
| 支持 IP、localhost、域名 ||                         |
| 支持带端口、路径、参数    ||                         |
| 支持相对路径             || 可启用 `allowRelative`   |
| 支持 `//example.com`    || 可允许无协议                |

21. deepUnique 「数组去重」函数

| 功能             | 说明                                                                    |
| -------------- | --------------------------------------------------------------------- |
| 支持基本类型       | `number`, `string`, `boolean`, `null`, `undefined`, `NaN`, `Symbol`|
| 支持对象、数组、嵌套结构 | `{ id: 1 }`, `[1, 2, [3]]` 等深层结构                            |
| 支持循环引用对象     | 使用 `WeakSet` 避免递归死循环                                        |
| 支持自定义去重依据    | 如根据对象字段去重(`item.id`)或自定义函数                                            |
| 保持原始顺序       | 结果中元素顺序和原数组一致                                             |



//  使用示例

deepUnique([1, 2, 2, 'a', 'a', NaN, NaN, { a: 1 }, { a: 1 }, [1, 2], [1, 2]]);
// => [1, 2, 'a', NaN, { a: 1 }, [1, 2]]


// 按字段去重
const users = [
  { id: 1, name: 'Tom' },
  { id: 2, name: 'Jerry' },
  { id: 1, name: 'Tommy' },
];

deepUnique(users, 'id');
// => [{ id: 1, name: 'Tom' }, { id: 2, name: 'Jerry' }]


// 使用函数作为规则
deepUnique(users, user => `${user.id}-${user.name.length}`);
// 例如:根据 id + 名字长度去重

22. truncateString 按字符数裁剪 + 添加省略号

truncateString('The quick brown fox jumps over the lazy dog', 20);
// => 'The quick brown f...'

truncateString('你好,这是一个测试用例', 8);
// => '你好,这是...'

truncateString('The quick brown fox jumps', 20, { preserveWords: true });
// => 'The quick brown...'

truncateString('这是一个中文的测试句子', 10, { ellipsis: '...' });
// => '这是一个中文的...'

23. truncateByVisualWidth 考虑字符宽度(中英文混排)

truncateByVisualWidth('中英混排 Hello 世界', 10);
// => '中英混排 H...'

truncateByVisualWidth('你好吗123456', 8);
// => '你好吗12...'

24. 大整数 相加 保持精度 bigIntAdd,bigIntSubtract, bigIntMultiply, bigIntDivide

console.log(bigIntAdd('-9999999999999999999', '9999999999999999999')); // "0"   高精度加法(含负数)
console.log(bigIntSubtract('1000000000000', '999999999999'));           // "1" 高精度减法(含负数)
console.log(bigIntMultiply('-123456789', '1000000000000'));            // "-123456789000000000000"  高精度乘法(含负数)
console.log(bigIntDivide('99999999999999999999', '3'));                // "33333333333333333333"  高精度整数除法(返回商,不含小数)

25. 获取 window 滚动位置的通用方法 和 获取任意滚动容器的滚动位置

// 页面滚动位置
console.log(getScrollPosition()); 
// {scrollLeft: ..., scrollTop: ...}

// 某个滚动元素
const container = document.querySelector('.scroll-container');
console.log(getScrollPosition(container));

26. 通用的滚动到顶部方法,兼容各种浏览器,支持传入目标元素(默认为 window),还支持可选的平滑滚动动画。

// 滚动整个页面到顶部(瞬间)
scrollToTop();

// 滚动整个页面到顶部(平滑滚动)
scrollToTop(window, true);

// 滚动指定元素到顶部
const div = document.querySelector('.scroll-container');
scrollToTop(div, true);

27. 获取元素位置和尺寸

const el = document.querySelector('.my-element');
const pos = getElementPosition(el);
console.log(`元素左上角坐标:(${pos.left}, ${pos.top})`);
console.log(`元素尺寸:宽=${pos.width}px 高=${pos.height}px`);

28. 加密 解密 函数

let reson = _enciphered("你好"); // 加密函数
let result = _decrypt(reson); // 解密函数

29. _retryAsync 异步重试函数

// 最大重试次数

// 指数退避机制

// 条件判断是否继续重试

// onRetry 回调通知

// 取消机制(中止重试)

// 超时机制(每次请求超时中止)

const controller = new AbortController();

retryAsync(
  () =>
    fetch('https://xxxxx.com/api', {
      signal: controller.signal,
    }),
  {
    retries: 5,
    delay: 500,
    timeout: 3000, // 每次超时时间
    exponentialBackoff: true,
    signal: controller.signal,
    onRetry: (err, attempt) => {
      console.log(`${attempt} 次尝试失败:`, err.message);
    },
  }
).catch(console.error);

// 可手动取消
setTimeout(() => {
  controller.abort();
  console.warn('已手动中止重试流程');
}, 5000);

30. chunkArray 将数组按指定大小分组

const data = [1, 2, 3, 4, 5, 6, 7];
const chunkSize = 3;

const result = chunkArray(data, chunkSize);

console.log(result);
// 输出: [[1, 2, 3], [4, 5, 6], [7]]

31. compact 移除假值元素

const data = [0, 1, false, 2, '', 3, null, undefined, NaN, 4];

const result = compact(data);

console.log(result);
// 输出: [1, 2, 3, 4]

32. mergeDeep 深度合并两个对象

const obj1 = {
  user: {
    name: 'Alice',
    info: {
      age: 25,
      city: 'New York'
    }
  },
  theme: 'light'
};

const obj2 = {
  user: {
    info: {
      age: 30,       // 会覆盖 obj1.user.info.age
      country: 'USA' // 会新增
    }
  },
  theme: 'dark'       // 会覆盖 theme
};

const merged = mergeDeep(obj1, obj2);

console.log(merged);
// 输出:
// {
//   user: {
//     name: 'Alice',
//     info: {
//       age: 30,
//       city: 'New York',
//       country: 'USA'
//     }
//   },
//   theme: 'dark'
// }

33. omit 返回排除指定 key 的新对象

const user = {
  id: 1,
  name: 'Alice',
  password: 'secret',
  email: 'alice@example.com'
};

const safeUser = omit(user, ['password', 'email']);

console.log(safeUser);
// 输出:
// {
//   id: 1,
//   name: 'Alice'
// }

// 这个方法 正好 跟 _pick 反过来

34. pickBy 根据条件筛选对象属性

const user = {
  id: 1,
  name: 'Alice',
  age: 25,
  isActive: true,
  password: 'secret'
};

// 示例:筛选出值为 "truthy" 的字段
const truthyUserData = pickBy(user, value => Boolean(value));

console.log(truthyUserData);
// 输出:
// {
//   id: 1,
//   name: 'Alice',
//   age: 25,
//   isActive: true,
//   password: 'secret'
// }
// (因为所有字段都是 truthy,这里只是演示)

// 示例:只保留类型为 string 的字段
const stringFields = pickBy(user, (value) => typeof value === 'string');

console.log(stringFields);
// 输出:
// {
//   name: 'Alice',
//   password: 'secret'
// }

35. memoize 缓存函数的返回值

// 这个方法 跟 vue 的计算属性 蛮像的

// 示例1: 
// 一个耗时的平方函数(模拟计算)
function slowSquare(n) {
    console.log(`Calculating square of ${n}`);
    return n * n;
}

const memoizedSquare = memoize(slowSquare);

console.log(memoizedSquare(5)); // 输出:Calculating square of 5\n25
console.log(memoizedSquare(5)); // 输出:25(不再调用 slowSquare)
console.log(memoizedSquare(6)); // 输出:Calculating square of 6\n36


// 示例2: 
function add(a, b) {
    console.log(`Calculating sum of ${a} and ${b}`);
    return a + b;
}

const memoizedAdd = memoize(add, (a, b) => `${a},${b}`);

console.log(memoizedAdd(1, 2)); // 输出:Calculating sum of 1 and 2\n3
console.log(memoizedAdd(1, 2)); // 输出:3(缓存命中)
console.log(memoizedAdd(2, 1)); // 输出:Calculating sum of 2 and 1\n3(key 不同)

// 注意事项:
// 默认 key 是第一个参数 args[0],适合单参数函数。
// 如果函数有多个参数,建议传入 resolver 自定义生成缓存 key。
// 缓存使用的是 Map,可以支持任意类型作为 key(包括对象,但要注意引用地址)。

36. compose 函数组合(从右往左执行)

const double = x => x * 2;
const square = x => x * x;
const increment = x => x + 1;

// 组合函数:先执行 increment,再执行 square,最后执行 double
const composedFn = compose(double, square, increment);

const result = composedFn(3);

console.log(result);
// 执行过程:
// increment(3) => 4
// square(4) => 16
// double(16) => 32
// 最终输出: 32

// compose(double, square, increment) 会从右往左执行函数,即先执行 increment,再执行 square,最后执行 double。

// 传入初始值 3,依次执行三个函数,得到最终结果。

37. isEqual 深度比较两个值是否相等

// 基础类型比较
console.log(isEqual(1, 1));        // true
console.log(isEqual(1, '1'));      // false
console.log(isEqual(null, null));  // true
console.log(isEqual(null, undefined)); // false

// 对象比较
const obj1 = { a: 1, b: { c: 2 } };
const obj2 = { a: 1, b: { c: 2 } };
const obj3 = { a: 1, b: { c: 3 } };

console.log(isEqual(obj1, obj2)); // true
console.log(isEqual(obj1, obj3)); // false

// 数组比较
const arr1 = [1, 2, [3, 4]];
const arr2 = [1, 2, [3, 4]];
const arr3 = [1, 2, [4, 3]];

console.log(isEqual(arr1, arr2)); // true
console.log(isEqual(arr1, arr3)); // false


// 基础类型相同且值相等时返回 true
// 不同类型或任一为 null 时返回 false
// 对象和数组会递归比较属性和值
// 只要有一个属性值不相等,返回 false

38. noop 空函数

// 用作默认回调函数,避免调用时出错
function doSomething(callback = noop) {
  console.log('Doing something...');
  callback();
}

doSomething(); // 不传回调,不会报错,也不会执行任何操作

doSomething(() => {
  console.log('Callback executed!');
});

// 输出:
// Doing something...
// Callback executed!

// noop 是一个什么都不做的空函数,常用作默认参数,避免调用时出现 undefined is not a function 错误。

// 适合占位、默认回调、事件处理等场景。

39. shuffle 洗牌算法打乱数组顺序

const original = [1, 2, 3, 4, 5];

const shuffled = shuffle(original);

console.log('原始数组:', original);     // [1, 2, 3, 4, 5]
console.log('打乱后的数组:', shuffled); // 例如:[3, 1, 5, 2, 4]

// shuffle 会返回一个新数组,不会修改原始数组(使用了扩展运算符 [...] 拷贝)。 
// 使用的是经典的 Fisher–Yates 洗牌算法,能确保随机性均匀。
// 每次运行可能都会得到不同的结果。

40. zip, unzip 将多个数组按索引位置组合 | 将组合后的数组拆分还原

import { unzip } from 'vuekit-js'; 

const names = ['Alice', 'Bob', 'Charlie'];
const ages = [25, 30, 28];
const cities = ['NY', 'LA', 'Chicago'];

const zipped = zip(names, ages, cities);

console.log(zipped);
// 输出:
// [
//   ['Alice', 25, 'NY'],
//   ['Bob', 30, 'LA'],
//   ['Charlie', 28, 'Chicago']
// ]



const combined = [
  ['Alice', 25, 'NY'],
  ['Bob', 30, 'LA'],
  ['Charlie', 28, 'Chicago']
];

const [names, ages, cities] = unzip(combined);

console.log(names);  // ['Alice', 'Bob', 'Charlie']
console.log(ages);   // [25, 30, 28]
console.log(cities); // ['NY', 'LA', 'Chicago']

// zip(...arrays):按对应索引组合多个数组,常用于将多列数据打包成表格形式。

// unzip(array):与 zip 相反,常用于还原结构或做表格数据的“转置”。

41. regexList 常用正则表达式

  // digits: /^\d+$/, // 只包含数字
  // integer: /^-?\d+$/, // 整数,可带负号
  // positiveInteger: /^[1-9]\d*$/, // 正整数,不含0开头
  // float: /^-?\d+(\.\d+)?$/, // 浮点数,可带负号
  // letters: /^[a-zA-Z]+$/, // 只包含字母
  // lowercaseLetters: /^[a-z]+$/, // 只包含小写字母
  // uppercaseLetters: /^[A-Z]+$/, // 只包含大写字母
  // lettersAndDigits: /^[a-zA-Z0-9]+$/, // 只包含字母和数字
  // englishSentence: /^[a-zA-Z\s.,?!]+$/, // 英文句子,允许空格和常用标点
  // email: /^[a-zA-Z0-9_.+-]+@[a-zA-Z0-9-]+\.[a-zA-Z0-9-.]+$/, // 邮箱地址
  // phoneCN: /^1[3-9]\d{9}$/, // 中国手机号码
  // landlineCN: /^(\d{3,4}-)?\d{7,8}$/, // 固定电话(带区号)
  // url: /^(https?:\/\/)?([\w-]+\.)+[\w-]+(\/[\w\-./?%&=]*)?$/, // URL 地址
  // ipv4: /^((25[0-5]|2[0-4]\d|1\d{2}|[1-9]?\d)(\.|$)){4}$/, // IPv4地址
  // idCard15: /^\d{15}$/, // 15位身份证号
  // idCard18: /^\d{17}[\dXx]$/, // 18位身份证号
  // date: /^\d{4}-(0[1-9]|1[0-2])-(0[1-9]|[12]\d|3[01])$/, // 日期格式 YYYY-MM-DD
  // time: /^([01]\d|2[0-3]):([0-5]\d):([0-5]\d)$/, // 时间格式 HH:MM:SS
  
  // postalCodeCN: /^\d{6}$/, // 中国邮政编码
  // licensePlateCN: /^[京津沪渝冀豫云辽黑湘皖鲁新苏浙赣鄂桂甘晋蒙陕吉闽贵粤青藏川宁琼使领A-Z]{1}[A-Z]{1}[A-Z0-9]{4}[A-Z0-9挂学警港澳]{1}$/, // 中国车牌号
  // passwordSimple: /^(?=.*[A-Za-z])(?=.*\d)[A-Za-z\d]{6,20}$/, // 密码6-20位,含字母和数字
  // passwordMedium: /^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)[A-Za-z\d]{8,20}$/, // 密码8-20位,含大小写字母和数字
  // passwordStrong: /^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)(?=.*[@$!%*?&])[A-Za-z\d@$!%*?&]{8,}$/, // 强密码,含大小写字母、数字和特殊符号,8位以上
  // htmlTag: /<\/?[\w\s="'-]*>/, // HTML标签匹配
  // chineseChar: /[\u4e00-\u9fa5]/, // 匹配中文字符
  // blank: /^\s*$/, // 全空白字符
  // uuid: /^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i, // UUID格式
  // base64: /^(?:[A-Za-z0-9+\/]{4})*(?:[A-Za-z0-9+\/]{2}==|[A-Za-z0-9+\/]{3}=)?$/, // Base64编码
  // hexColor: /^#?([a-fA-F0-9]{6}|[a-fA-F0-9]{3})$/, // Hex颜色值
  // macAddress: /^([0-9A-Fa-f]{2}[:-]){5}([0-9A-Fa-f]{2})$/, // MAC地址

const email = 'test@example.com';
if (regexList.email.test(email)) {
  console.log('邮箱格式正确');
} else {
  console.log('邮箱格式错误');
}