用new和delete运算符动态分配内存空间的方法编写程序。
从键盘输入33整型数组的数据,并计算出所有元素之和,打印出最大值和最小值。
输入输出要用流运算符实现。
1

**正文**《ADS工具入门教程》ADS,全称Arm Development Studio,是Arm公司推出的一款强大的嵌入式系统开发和调试工具。
它为基于Arm架构的芯片提供了全面的软件开发支持,包括C/C++编译器、调试器、性能分析器等功能。
本教程将引导您了解如何使用ADS进行高效的开发工作。
一、ADS安装与配置下载并安装ADS软件,确保您的计算机满足其系统需求。
安装完成后,启动ADS,进行必要的环境配置,包括设置编译器路径、目标硬件配置以及工程模板等。
这一步对于确保项目能够正确构建和链接至关重要。
二、创建新工程在ADS中,新建一个工程是开始项目的第一步。
通过“File”菜单选择“New Project”,然后按照向导提示选择合适的工程类型(如应用或库项目),设置工程名称和位置。
接着,指定要使用的处理器型号,这会影响到编译器的配置和产生的代码目标架构。
三、添加源代码在新创建的工程中,可以添加源代码文件(.c或.c++)和头文件(.h)。
通过“Project”菜单的“Add Files to Group”选项,选择要包含的文件。
记得将源代码组织到适当的文件夹结构中,以便于管理和维护。
四、编译与链接完成代码添加后,可以进行编译和链接操作。
点击“Build”菜单的“Build Project”或使用快捷键,ADS会自动执行预处理、编译、汇编和链接的步骤。
如果出现错误,ADS会提供详细的错误报告,帮助定位问题。
五、调试设置ADS的强大之处在于其调试功能。
在工程属性中,配置调试器设置,如GDB服务器端口、目标设备连接方式等。
设置完后,可以在源代码中设置断点,使用“Debug”菜单的“Start Debugging”启动调试会话。
在调试过程中,可以查看变量值、单步执行、调用堆栈和内存查看等功能。
六、性能分析除了基本的开发和调试,ADS还提供了性能分析工具。
通过配置性能分析器,可以收集CPU使用率、指令执行统计等数据,帮助优化代码性能。
在分析结果中,可以找到程序的瓶颈,指导优化工作。
七、示例解析在本教程的压缩包文件"ads_tutorial"中,包含了使用ADS进行开发的实例项目。
这些示例覆盖了从简单的Hello World程序到复杂功能的实现,详细展示了ADS的各个功能。
通过对这些示例的学习和实践,您可以更深入地理解ADS的工作流程和使用技巧。
总结,ADS作为一款强大的嵌入式开发工具,不仅提供了完整的开发环境,还包括了丰富的调试和分析功能。
通过本教程的学习,您将能够熟练掌握ADS的基本操作,并利用它来开发高效、可靠的Arm架构嵌入式系统。
记得结合实际项目不断练习,提升自己的开发技能。
2025/6/15 22:25:19 294KB
1

本例中敏感词ciku.txt放在C盘根目录下,采用的ActiveXObject插件获取本地文件内容。
使用此插件不需网上下插件,直接用如下js代码即可。
浏览器需修改interner安全选项的级别,启用ActiveX才能获取到代码中的ActiveXObject插件。
如下图所示:js代码实现如下:[removed] // -------------- 全局变量,用来判断文本域中是否包含脏词,默认为false,即不包含脏词------- var isDirty = false; //使用ActiveX读取本地文件获取dirt在JavaScript中,有时我们需要在用户提交表单时检查输入内容是否包含敏感词,以防止不当内容的发布。
本文介绍了一种使用ActiveXObject技术在IE浏览器中实现这一功能的方法。
ActiveXObject是Internet Explorer特有的,它允许JavaScript与本地操作系统交互,如读取本地文件。
我们需要在本地(例如C盘根目录)创建一个名为`ciku.txt`的文本文件,其中包含了我们定义的敏感词。
然后,通过JavaScript的ActiveXObject来读取这个文件的内容。
以下是关键的JavaScript代码:```javascriptvar isDirty = false;function readFile() { var fso = new ActiveXObject("Scripting.FileSystemObject"); var openF = fso.OpenTextFile("c:\\ciku.txt", 1); var cikuStr = openF.ReadAll(); return cikuStr;}````readFile()`函数使用`Scripting.FileSystemObject`对象打开并读取`ciku.txt`文件,然后将敏感词返回给其他函数处理。
当用户尝试提交表单时,会触发`submitForm1()`函数。
这个函数首先获取用户在表单文本域中输入的内容,然后检查是否包含敏感词。
如果发现敏感词,它会调用`filterWord()`函数替换敏感词,并弹出提示让用户确认是否继续提交。
如果用户确认,表单会被提交;
否则,提交操作会被取消。
`filterWord()`函数接收用户输入的内容,读取敏感词列表,然后对每个敏感词调用`filterOneWord()`函数进行替换。
`filterOneWord()`使用正则表达式和`replace()`方法将敏感词替换为星号("**")。
表单部分的HTML代码如下:```html<form name="message_board" id="message_board" action="aaa.html"> <textarea name="message" id="message" cols="50" rows="10">"This is you post messsage"——phpdream</textarea> <br/> <input type="button" value="提交留言" id="submitMessage" onclick="submitForm1()" /></form>```需要注意的是,由于ActiveXObject是IE特有的,这种方法在其他非IE浏览器(如Chrome、Firefox、Safari或Edge)上无效。
为了实现跨浏览器兼容,可以考虑使用其他技术,如FileReader API(适用于现代浏览器)或者将敏感词库保存在服务器端,通过Ajax请求获取。
此外,为了使用ActiveXObject,用户需要在浏览器的安全设置中启用ActiveX控件,这可能带来安全风险,因此在实际应用中应谨慎使用,并确保对用户有明确的提示和说明。
2025/6/15 22:22:47 97KB
1

在Android应用开发中,集成百度地图API是一项常见的需求,它能为用户提供丰富的地图服务功能,如定位、导航、路线规划等。
本篇文章将详细介绍如何在Android项目中进行百度地图的开发,以及一些关键的技术点。
我们需要在百度地图开放平台(http://lbsyun.baidu.com/)注册账号并创建应用,获取API密钥。
这个API密钥是连接我们应用与百度地图服务的关键,确保了应用的合法性。
完成注册后,根据应用的包名和签名信息生成相应的密钥,并在项目中正确配置。
接下来,我们将在AndroidManifest.xml文件中添加必要的权限。
包括访问网络、读写外部存储、获取地理位置等,例如:```xml<uses-permission android:name="android.permission.INTERNET" /><uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" /><uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />```同时,添加百度地图SDK所需的库依赖:```xml<meta-data android:name="com.baidu.lbsapi.API_KEY" android:value="你的API_KEY" />```然后,在布局文件中加入MapView组件,这是显示百度地图的核心组件:```xml<com.baidu.mapapi.map.MapView android:id="@+id/bmapView" android:layout_width="match_parent" android:layout_height="match_parent" android:clickable="true" />```在Activity或Fragment中初始化MapView,并设置地图的显示样式和级别:```javaMapView mapView = findViewById(R.id.bmapView);BaiduMap baiduMap = mapView.getMap();baiduMap.setMapType(BaiduMap.MAP_TYPE_NORMAL); // 设置为普通地图baiduMap.setZoom(15); // 设置地图缩放级别```为了让应用能够获取用户的位置信息,我们需要开启定位服务:```javaLocationClient locationClient = new LocationClient(this);locationClient.registerListener(new MyLocationListener()); // 自定义监听器locationClient.start(); // 开启定位```编写`MyLocationListener`类,重写`onReceiveLocation`方法,接收定位结果:```javapublic class MyLocationListener implements BDLocationListener { @Override public void onReceiveLocation(BDLocation location) { if (location != null) { // 使用获得的经纬度设置地图中心点 LatLng myLocation = new LatLng(location.getLatitude(), location.getLongitude()); MapStatusUpdate update = MapStatusUpdateFactory.newLatLng(myLocation); baiduMap.setMapStatus(update); } }}```为了给用户提供更丰富的地图交互体验,可以添加覆盖物(Overlay)、信息窗口(InfoWindow)等功能。
例如,自定义一个Marker来表示用户当前位置:```javaOverlayOptions markerOption = new MarkerOptions() .position(myLocation) .icon(BitmapDescriptorFactory.fromResource(R.drawable.marker));baiduMap.addOverlay(markerOption);```此外,还可以通过百度地图API实现路线规划,比如驾车、步行、公交等。
调用`DrivingRoutePlanOption`、`WalkingRou
2025/6/15 22:19:09 5.65MB
1
简介:
在.NET框架中,C#语言的类(class)属于引用类型。
这意味着当你声明一个类的实例时,实际上是在堆上创建一个对象,并在栈上创建一个引用指向这个对象。
因此,当你将对象作为参数传递给函数时,实际上是传递了这个引用的副本,而不是对象本身。
这就是所谓的"传引用"或"按引用传递"。
让我们深入探讨一下这个问题,以标题和描述中给出的代码为例:```csharpclass Program{ static void Main(string[] args) { TestClass objA = new TestClass(); objA.Name = "I am ObjA"; Console.WriteLine(String.Format("In Main:{0}", objA.Name)); TestFun(objA); Console.WriteLine(String.Format("In Main:{0}", objA.Name)); Console.Read(); } static void TestFun(TestClass obj) { obj.Name = "I am be modified in TestFun"; Console.WriteLine(String.Format("In TestFun:{0}", obj.Name)); } public class TestClass { public string Name { get; set; } }}```在这个例子中,`TestFun`函数接收到`objA`的引用副本`obj`。
当在`TestFun`中修改`obj.Name`时,实际上是修改了`objA`引用的对象,因为它们都指向同一个堆上的实例。
因此,`Main`函数中再次打印`objA.Name`时,值已经被修改为"I am be modified in TestFun"。
然而,如果我们更改`TestFun`的实现:```csharpstatic void TestFun(TestClass obj){ TestClass objB = new TestClass(); obj = objB; obj.Name = "I am ObjB"; Console.WriteLine(String.Format("In TestFun:{0}", obj.Name));}```这里我们创建了一个新的`TestClass`实例`objB`,然后让`obj`引用`objB`。
虽然在`TestFun`内部`obj`的值改变了,但这不会影响`Main`函数中的`objA`,因为`objA`仍然指向原始的`TestClass`实例。
所以,`Main`函数中的`objA.Name`输出仍然是"I am ObjA",因为`objA`并没有被修改指向新创建的`objB`。
这个现象可以用内存模型来解释,就像描述中提到的那样。
在调用`TestFun`时,`objA`的地址被复制到`obj`,但是`objA`本身并未改变。
在`TestFun`中,`obj`被重新分配给`objB`的地址,但`objA`仍然指向原始对象,所以`Main`中的`objA`不会受到影响。
C#中的对象参数传递特性对于理解和调试代码非常重要。
理解这种行为可以帮助我们避免意外地修改了原本不想修改的对象,同时也能有效地利用引用传递来共享和修改数据。
在编写函数时,要清楚地知道参数是值类型(value type,如int、struct)还是引用类型(reference type,如class),因为这将直接影响到参数的处理方式和函数的行为。
2025/6/15 20:02:36 44KB
1
简介:
根据给定的信息,我们可以从《赵丽词汇5500讲义精校版》中提炼出以下相关的知识点: ### 1. 单词理解和运用 #### 1.1 bleak(荒凉的、萧条的) - **含义**:形容一个地方缺乏生机、显得荒凉或者萧条。
- **例句**:The living conditions in the village were bleak due to the harsh winter. #### 1.2 haunted(闹鬼的、萦绕心头的) - **含义**:指某个地方被认为有鬼魂出没或某事长期困扰人心。
- **例句**:The old mansion is said to be haunted by the ghost of its former owner. #### 1.3 gaunt(憔悴的、瘦削的) - **含义**:形容人因病痛或饥饿而显得憔悴、瘦削的样子。
- **例句**:After the long illness, his face had become gaunt and his eyes sunken. #### 1.4 acumen(敏锐、机智) - **含义**:指个人在处理问题时展现出的敏锐洞察力和智慧。
- **例句**:Her business acumen was evident in her quick decision-making abilities. #### 1.5 abnormal(不正常的、异常的) - **含义**:指与正常情况不符的状态或行为。
- **例句**:The patients body temperature remained abnormally high for several days, reaching up to 40.5 degrees Celsius. #### 1.6 abundant(丰富的、充裕的) - **含义**:指某种资源或条件极其丰富、充裕。
- **例句**:Texas is abundant in natural resources such as oil and gas. #### 1.7 access(进入、获取) - **含义**:指能够到达某个地点或获得某种资源的能力。
- **例句**:Access to some parts of South America is still difficult due to dense forests covering large areas. #### 1.8 account(考虑、账户) - **含义**:指在做出决定前考虑某个因素,或指银行账户等。
- **例句**:I hope my teacher will take my recent illness into account when judging my examination performance. #### 1.9 adjust(调整、调节) - **含义**:指根据特定环境或需求改变某些设置或状态。
- **例句**:My camera can be adjusted to take pictures in cloudy or sunny conditions. #### 1.10 adapt(适应、调节) - **含义**:指个体或物体根据环境的变化进行相应的调整以更好地生存或工作。
- **例句**:Newcomers found it impossible to adapt themselves sufficiently to the climate to make permanent homes in the new country. ### 2. 句子理解与应用 #### 2.1 题目示例分析 - **题目**:“Living in that _____ house over there has nearly driven the hero of the story mad.” - **选项**:A. bleak B. haunted C. gaunt D. acumen - **正确答案**:B. haunted - **解析**:根据句子意思,“在那里居住几乎让故事中的主人公发疯”,最符合这个情境的是“haunted”,即闹鬼的房子。
#### 2.2 题目示例分析 - **题目**:“His body temperature is ______ for three days, the highest point reaching to 40.5 degrees Celsius.” - **选项**:A. uncommon B. disordered C. abnormal D. extraordinary - **正确答案**:C. abnormal - **解析**:体温持续多天处于较高水平,显然不符合正常状态,因此“abnormal”最为恰当。
通过以上知识点的详细解析,我们可以看出,《赵丽词汇5500讲义精校版》不仅包含了丰富的英语词汇,还提供了大量实用的练习题来帮助学习者理解和掌握这些词汇在不同语境下的用法。
这对于提高英语水平尤其是词汇量非常有帮助。
2025/6/15 20:01:22 99KB
1
简介:
在Android开发中,`DialogFragment`是Android SDK提供的一种用于创建弹出式对话框的组件。
它是`Fragment`的一个子类,与传统的`Dialog`相比,`DialogFragment`具有更多的优势,尤其是在处理屏幕旋转和生命周期管理方面。
本篇文章将深入探讨`DialogFragment`的核心概念、优点、使用方法以及常见应用场景。
`DialogFragment`简介:`DialogFragment`继承自`Fragment`,它不仅拥有`Fragment`的所有功能,还能展示一个模态对话框。
它将对话框和碎片的特性结合在一起,使得对话框可以在不同的屏幕配置下保持一致的行为,同时能够更好地适应Android应用的动态布局需求。
`DialogFragment`的优点:1. 生命周期管理:由于`DialogFragment`继承自`Fragment`,它拥有完整的生命周期回调,可以方便地处理如暂停、恢复、销毁等状态变化,使得对话框在不同场景下的行为更加可控。
2. 自动处理配置变更:当设备发生横竖屏切换时,`DialogFragment`会自动处理对话框的关闭和重新显示,而普通的`Dialog`则需要开发者手动处理。
3. 易于集成:`DialogFragment`可以像普通`Fragment`一样添加到Activity的布局中,也可以单独显示为对话框,增加了代码的复用性。
4. 更好的主题定制:`DialogFragment`支持自定义对话框样式,可以方便地调整对话框的外观和行为。
使用`DialogFragment`的步骤:1. 创建`DialogFragment`子类:你需要创建一个新的类,继承自`DialogFragment`,并重写必要的方法,如`onCreateDialog()`,在这个方法中构建你的对话框实例。
2. 实现`onCreateDialog()`:在这个方法中,你可以使用`AlertDialog.Builder`或自定义的`Dialog`来创建对话框,并设置其标题、消息、按钮等元素。
3. 显示`DialogFragment`:你可以通过`FragmentManager`和`FragmentTransaction`将`DialogFragment`添加到当前的Activity中,或者使用静态方法`show()`直接显示对话框。
示例代码:```javapublic class MyDialogFragment extends DialogFragment { @NonNull @Override public Dialog onCreateDialog(Bundle savedInstanceState) { AlertDialog.Builder builder = new AlertDialog.Builder(requireContext()); builder.setMessage("这是一条对话框消息") .setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // 确定按钮的点击事件 } }) .setNegativeButton("取消", null); return builder.create(); }}```在Activity中显示`DialogFragment`:```javaMyDialogFragment dialogFragment = new MyDialogFragment();dialogFragment.show(getSupportFragmentManager(), "my_dialog");````DialogFragment`的常见应用场景:- 提示用户确认操作,如删除、保存等。
- 展示设置选项或偏好设置。
- 获取用户输入,如登录、注册表单。
- 显示帮助信息或关于应用的详情。
总结来说,`DialogFragment`是Android开发中处理对话框的理想选择,它的灵活性、易用性和强大的功能使其在各种应用场合都能发挥出色的效果。
通过理解其工作原理和实践操作,开发者可以更高效地利用这一组件,提升应用的用户体验。
2025/6/15 19:58:08 1.42MB
1
简介:
在编程领域,尤其是使用C++这种面向对象的语言时,"无法实例化抽象类"是一个常见的错误,这通常发生在尝试创建一个声明为抽象的类的实例时。
在C++中,抽象类是通过包含至少一个纯虚函数来定义的。
这些类不能被实例化,因为它们没有具体的实现,而主要是作为基类来使用,为派生类提供接口定义。
标题"无法实例化抽象类"指出的问题可能源于以下几个方面:1. **纯虚函数**:一个类如果包含至少一个纯虚函数(即声明为`virtual void func() = 0;`的函数),那么这个类就会被视为抽象类。
抽象类不能用于创建对象,只能作为其他类的基类。
2. **错误的实例化尝试**:可能是开发者尝试直接使用`new`关键字或在栈上创建抽象类的对象,这是不合法的。
例如,`AbstractClass* ptr = new AbstractClass();` 或 `AbstractClass obj;` 都会导致编译错误。
3. **调用约定**:在标签"VC10.0 C++ Win32 programming"中,提到的调用约定(Calling Convention)可能与问题有关。
不同的调用约定会影响函数参数的传递方式,如果错误地指定调用约定,可能会导致链接错误,但这通常不会直接影响抽象类的实例化问题。
4. **参数匹配**:描述中的"检查调用约定,参数等"暗示可能存在参数类型或数量不匹配的问题。
虽然这不是直接与抽象类实例化相关的错误,但错误的函数签名可能导致编译错误,特别是当涉及到虚函数的重写时。
5. **派生类的实现**:如果一个派生类没有实现其基类的所有纯虚函数,那么这个派生类也会变成抽象类。
确保所有的纯虚函数都有具体实现,否则编译器会报错。
6. **模板和抽象类**:如果抽象类被用作模板的参数,确保在实例化模板时,模板参数满足抽象类的要求,即提供所有纯虚函数的实现。
7. **编译器和版本问题**:VC10.0指的是Visual Studio 2010,不同版本的编译器可能对C++标准的支持程度不同,或者存在一些已知的bug。
确保编译器设置正确,并且更新到最新的服务包和补丁。
解决此类问题通常需要检查代码中抽象类的定义,确保所有纯虚函数在需要的地方得到了实现,同时检查调用的函数签名是否正确,参数类型和数量是否匹配。
此外,查阅编译器的错误信息也能帮助定位问题所在。
对于提供的PDF文件"cannot-instantiate-abstract-class.pdf",可能包含更详细的解释和示例,阅读它将有助于深入理解抽象类和实例化抽象类的限制。
建议结合文档内容,根据具体情况分析和解决问题。
2025/6/15 19:57:52 40KB
1
简介:
slickerPicker(测试版) 一个使用Vanilla Javascript制作的简单快速的颜色选择器插件设置插件包括slickerPicker.js和slickerPicker.css < script type = "text/javascript" src = "slickerPicker.js" > < / script >< link rel = "stylesheet" href = "slickerPicker.css" > 实例化slickerPicker。
slickerPicker将仅接受选项对象的1个参数您必须包括要将选择器附加到的元素的目标ID。
new slickerPicker ( { target : "target" } ) ;
2025/6/15 19:51:31 14KB
1
简介:
在本文中,我们将深入探讨如何使用Qt框架与Video for Linux 2(V4L2)接口相结合,实现在Linux系统上显示摄像头视频流。
V4L2是Linux内核提供的一种标准接口,用于与视频捕获设备(如摄像头)进行交互,而Qt则是一个跨平台的C++图形用户界面应用程序开发框架。
我们需要了解V4L2的基本概念。
V4L2是V4L(Video4Linux)的升级版,提供了更多的功能,包括对多种视频格式的支持、多设备并发访问以及高级缓冲区管理。
它通过/dev/videoX设备节点与摄像头通信,X为设备编号。
接下来,我们要引入Qt。
Qt库提供了一套完整的图形用户界面工具,包括窗口、控件、布局等,以及多媒体模块(QMultimedia),可以方便地处理音频和视频数据。
在Qt中,我们可以通过QCamera类来操作摄像头,并使用QCameraViewfinder或QVideoWidget来显示视频流。
实现"v4l2摄像头显示视频流"的关键步骤如下:1. **初始化Qt环境**:确保系统已安装Qt库,然后创建一个Qt项目,选择合适的Qt版本和构建系统。
2. **导入相关模块**:在代码中导入必要的Qt模块,如`<QtWidgets>`(用于窗口和控件)、`<QCamera>`(摄像头操作)和`<QCameraViewfinder>`(显示视频流)。
3. **创建QCamera对象**:使用QCamera类创建一个摄像头对象,传入设备ID(通常是"/dev/video0")作为参数。
例如: ```cpp QCamera camera(new QCamera("/dev/video0", this)); ``` 如果需要检测可用摄像头,可以使用`QCameraInfo`类列出所有设备。
4. **设置视频源**:V4L2摄像头作为视频源,可以通过设置`QCamera::setCaptureDevice`方法来实现: ```cpp camera.setCaptureDevice(QCamera::CaptureDevice::DeviceType, "video0"); ```5. **启动相机**:在确保设置正确后,启动相机: ```cpp camera.start(); ```6. **显示视频流**:创建一个`QCameraViewfinder`实例并将其设置为相机的视图finder,然后将视图finder添加到窗口布局中: ```cpp QCameraViewfinder *viewfinder = new QCameraViewfinder(this); camera.setViewfinder(viewfinder); layout->addWidget(viewfinder); // 假设layout是窗口的布局 ```7. **处理错误和状态改变**:为QCamera对象添加信号连接,以便在出现错误或状态改变时进行相应的处理。
8. **关闭相机**:在应用退出或不再需要视频流时,记得停止并释放相机资源: ```cpp camera.stop(); delete camera; ```以上就是使用Qt结合V4L2显示摄像头视频流的基本步骤。
实际应用中可能还需要处理分辨率设置、帧率控制、色彩格式转换等更复杂的细节。
同时,为了保证兼容性和稳定性,可能需要针对不同的硬件和驱动进行适配。
此外,还可以利用QMediaPlayer和QVideoSurfaceFormat等类来实现自定义的视频播放器功能。
通过这些知识,开发者可以构建出功能丰富的摄像头应用,不仅限于简单的视频显示,还能进行录像、图像处理等多种功能。
对于嵌入式系统或者需要在Linux环境下处理摄像头数据的应用来说,Qt结合V4L2是一个高效且灵活的选择。
2025/6/15 19:50:07 12KB
1
共 146 条记录 首页 上一页 下一页 尾页
在日常工作中,钉钉打卡成了我生活中不可或缺的一部分。然而,有时候这个看似简单的任务却给我带来了不少烦恼。 每天早晚,我总是得牢记打开钉钉应用,点击"工作台",再找到"考勤打卡"进行签到。有时候因为工作忙碌,会忘记打卡,导致考勤异常,影响当月的工作评价。而且,由于我使用的是苹果手机,有时候系统更新后,钉钉的某些功能会出现异常,使得打卡变得更加麻烦。 另外,我的家人使用的是安卓手机,他们也经常抱怨钉钉打卡的繁琐。尤其是对于那些不太熟悉手机操作的长辈来说,每次打卡都是一次挑战。他们总是担心自己会操作失误,导致打卡失败。 为了解决这些烦恼,我开始思考是否可以通过编写一个全自动化脚本来实现钉钉打卡。经过一段时间的摸索和学习,我终于成功编写出了一个适用于苹果和安卓系统的钉钉打卡脚本。
2024-04-09 15:03 15KB 钉钉 钉钉打卡