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

VSIX:C#项目 重命名所有标识符(Visual Studio扩展开发)

        出于某种目的(合法的,真的合法的,合同上明确指出可以这样做),我准备了一个重命名所有标识符的VS扩展,用来把一个C#库改头换面,在简单的测试项目上工作很满意,所有标识符都被准确替换。我还尝试用在C++项目上,问题就比较多了,因为VS并不能准确识别代码,这说明,C#比C++好用太多了。

        当然,最终合同被放弃了,所以这个东西也没派上用场,纯粹成了我的个人练习(因为没有人指派我做这个程序)。

        本文涉及的代码支持2017、2019和2022,为了稳妥起见,项目本身用不同版本的VS创建,实际代码则放在一个共享文件中,只需要在生成的框架之加入一句调用代码即可。

目录

一、创建项目框架

1.1 新建VS项目

1.2 添加命令

1.3 测试此框架代码

二、引入实际代码

2.1 添加项目文件

2.2 添加依赖项

三、修改代码

四、测试实际效果

4.1 创建测试项目

4.2 在VSIX项目打开测试项目

4.3 效果

五、代码


        

一、创建项目框架

1.1 新建VS项目

        以下均以VS2022社区版为例。

        项目类型过滤选择“扩展”,项目类型为“VSIX Project”。

1.2 添加命令

        创建以后在项目上右键,“添加”-“新建项”:

        “Command”就是一个菜单命令,会出现在VS的“工具”菜单下面。

        添加之后会看到增加了一个文件:Command1.cs,当然如果你改了命令名就是另外一个文件。

        文件不长,直接拉到最后,看最后一个方法的代码:

		/// <summary>/// This function is the callback used to execute the command when the menu item is clicked./// See the constructor to see how the menu item is associated with this function using/// OleMenuCommandService service and MenuCommand class./// </summary>/// <param name="sender">Event sender.</param>/// <param name="e">Event args.</param>private void Execute(object sender, EventArgs e){ThreadHelper.ThrowIfNotOnUIThread();string message = string.Format(CultureInfo.CurrentCulture, "Inside {0}.MenuItemCallback()", this.GetType().FullName);string title = "Command1";// Show a message box to prove we were hereVsShellUtilities.ShowMessageBox(this.package,message,title,OLEMSGICON.OLEMSGICON_INFO,OLEMSGBUTTON.OLEMSGBUTTON_OK,OLEMSGDEFBUTTON.OLEMSGDEFBUTTON_FIRST);}

        好简单的,猜也能猜到就是显式一个消息框。

1.3 测试此框架代码

        编译项目,应该没什么问题(全是开发工具生成的代码嘛)。

        调试或者直接运行不调试(“调试菜单”的“开始执行(不调试)”),会打开一个新的VS2022窗口,像普通VS一样,但是已经加载了扩展。

        选择项目,或者不选择项目直接进入。不选择项目直接进入(点击“继续但无需代码”):

        工具菜单下会出现“Invoke Command1”(图中还有另外一个相似菜单,是我的正式项目创建的),点击出现:

        这框架就算完成了,剩下的就是修改命令代码。

二、引入实际代码

2.1 添加项目文件

        在项目上右键,“添加”-“现有项”,找到实际代码文件添加进来,当然你也可以直接放在项目里面,但是因为VS扩展项目是依赖VS版本的,最好把通用部分独立出来。

        这个位置是在项目之外的,可以由多个项目共享。

2.2 添加依赖项

        文件加进来之后不能编译:

        因为缺少依赖项,在“项目”-“引用”上右键,“添加引用”:

        在“程序集”-“扩展”里面找到“Microsoft.VisualStudio.VCCodeModel”,选中,确定。

        然后程序就可以编译了。这个依赖项其实只和C++项目功能有关。

        如果发生奇怪错误:

严重性	代码	说明	项目	文件	行	禁止显示状态
错误		CreatePkgDef : error : ArgumentException: No Visual Studio registration attribute found in this assembly.
The assembly should contain an instance of the attribute 'Microsoft.VisualStudio.Shell.RegistrationAttribute' defined in assembly 'Microsoft.VisualStudio.Shell.Framework' version '17.0.0.0' 在 Microsoft.VisualStudio.Tools.CreatePkgDef.ProcessAssembly(String fileName, Hive hive, PkgDefContext context, Boolean register, RegistrationMode mode) 位置 D:\a\_work\1\s\src\product\vssdk\tools\CreatePkgDef\CreatePkgDef.cs:行号 383在 Microsoft.VisualStudio.Tools.CreatePkgDef.DoCreatePkgDef(InputArguments inputArguments) 位置 D:\a\_work\1\s\src\product\vssdk\tools\CreatePkgDef\CreatePkgDef.cs:行号 202在 Microsoft.VisualStudio.Tools.CreatePkgDef.Main(String[] arguments) 位置 D:\a\_work\1\s\src\product\vssdk\tools\CreatePkgDef\CreatePkgDef.cs:行号 91	VSIXProject1			

        不要尝试任何解决方案,删除刚才添加的东西也没用,整个过程删掉重来。这可能是VS的BUG。

三、修改代码

        现在我们可以将代码引入,在Command1.cs里面添加如下内容:

//文件头添加对共享代码的引用
using VSIXProjectShare;
using Task = System.Threading.Tasks.Task;//在类里面添加变量,就近放在构造函数前面好了private CommandShare commandshare;//构造函数最后加上这一句commandshare = new CommandShare(this.package);//Execute最后加一句commandshare.Execute();

        最终的Command1.cs是这样的(四处修改在里面已经注明):

using Microsoft.VisualStudio.Shell;
using Microsoft.VisualStudio.Shell.Interop;
using System;
using System.ComponentModel.Design;
using System.Globalization;
using System.Threading;
using System.Threading.Tasks;
using VSIXProjectShare;//第一处修改
using Task = System.Threading.Tasks.Task;namespace VSIXProject1
{/// <summary>/// Command handler/// </summary>internal sealed class Command1{/// <summary>/// Command ID./// </summary>public const int CommandId = 0x0100;/// <summary>/// Command menu group (command set GUID)./// </summary>public static readonly Guid CommandSet = new Guid("16dcb30d-2f74-4781-bde4-c21c60716ac8");/// <summary>/// VS Package that provides this command, not null./// </summary>private readonly AsyncPackage package;private CommandShare commandshare;//第二处修改/// <summary>/// Initializes a new instance of the <see cref="Command1"/> class./// Adds our command handlers for menu (commands must exist in the command table file)/// </summary>/// <param name="package">Owner package, not null.</param>/// <param name="commandService">Command service to add command to, not null.</param>private Command1(AsyncPackage package, OleMenuCommandService commandService){this.package = package ?? throw new ArgumentNullException(nameof(package));commandService = commandService ?? throw new ArgumentNullException(nameof(commandService));var menuCommandID = new CommandID(CommandSet, CommandId);var menuItem = new MenuCommand(this.Execute, menuCommandID);commandService.AddCommand(menuItem);commandshare = new CommandShare(this.package);//第三处修改}/// <summary>/// Gets the instance of the command./// </summary>public static Command1 Instance{get;private set;}/// <summary>/// Gets the service provider from the owner package./// </summary>private Microsoft.VisualStudio.Shell.IAsyncServiceProvider ServiceProvider{get{return this.package;}}/// <summary>/// Initializes the singleton instance of the command./// </summary>/// <param name="package">Owner package, not null.</param>public static async Task InitializeAsync(AsyncPackage package){// Switch to the main thread - the call to AddCommand in Command1's constructor requires// the UI thread.await ThreadHelper.JoinableTaskFactory.SwitchToMainThreadAsync(package.DisposalToken);OleMenuCommandService commandService = await package.GetServiceAsync(typeof(IMenuCommandService)) as OleMenuCommandService;Instance = new Command1(package, commandService);}/// <summary>/// This function is the callback used to execute the command when the menu item is clicked./// See the constructor to see how the menu item is associated with this function using/// OleMenuCommandService service and MenuCommand class./// </summary>/// <param name="sender">Event sender.</param>/// <param name="e">Event args.</param>private void Execute(object sender, EventArgs e){ThreadHelper.ThrowIfNotOnUIThread();string message = string.Format(CultureInfo.CurrentCulture, "Inside {0}.MenuItemCallback()", this.GetType().FullName);string title = "Command1-d";//修改这里以确认版本// Show a message box to prove we were hereVsShellUtilities.ShowMessageBox(this.package,message,title,OLEMSGICON.OLEMSGICON_INFO,OLEMSGBUTTON.OLEMSGBUTTON_OK,OLEMSGDEFBUTTON.OLEMSGDEFBUTTON_FIRST);commandshare.Execute();//第四处修改}}
}

        然后编译执行(仍然用“继续但无需代码”):

        看,版本正确,这是一个小技巧,debug发生困惑的时候先确认版本,不要编译失败执行旧版本。

        这就对了,因为共享代码需要项目来操作。这说明共享代码也正确进去了。后面就可以测试实际效果,共享代码放在本文最后。

四、测试实际效果

4.1 创建测试项目

        创建一个C#项目,一个对话框好了:

        一个对话框,有个静态文本,窗口初始化设置了一下文本。

4.2 在VSIX项目打开测试项目

        先创建好这个项目,然后回到VSIX项目,调试或运行,在新打开的VS启动时选择新建的这个项目,打开后是和普通VS一样操作的,只不过多了扩展菜单项。

        现在从工具菜单执行我们的命令,运行时会在输出窗口输出内容,最后会得到一个消息框:

        这就表示正确完成,提示信息是共享代码最后的版本,就是日期和时间。

        输出窗口输出如下:

        注意:此时修改的文件还没保存,要点击“全部保存”来保存文件。

        然后我们看看效果如何,现重新编译程序确认测试项目是正常的。

4.3 效果

        看看代码变成了什么样:

        Form1.Designer.cs就不贴了。

        看看文件比较:

五、代码

        共享代码在此,文件名CommandShare.cs:

using System;
using System.ComponentModel.Design;
using System.Globalization;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.VisualStudio.Shell;
using Microsoft.VisualStudio.Shell.Interop;
using Task = System.Threading.Tasks.Task;using EnvDTE;
using EnvDTE80;
using Microsoft.VisualStudio;
using Microsoft.VisualStudio.VCCodeModel;
using System.IO;
using Microsoft.Internal.VisualStudio.PlatformUI;namespace VSIXProjectShare
{public sealed class CommandShare{private readonly AsyncPackage package;enum ProjectType { VC,CSharp,OTHER};//项目类型ProjectType projectType;private Random r ;//随机数private string new_name_title;//新名称标题private long count = 0;//顺序编号public CommandShare(AsyncPackage _package){package = _package;Log("初始化插件");r = new Random();new_name_title = "_ASDFGHJKL_" + r.Next().ToString() + "_";}//显示消息对话框private void ShowMessageBox(string title, string message){VsShellUtilities.ShowMessageBox(package,message,title,OLEMSGICON.OLEMSGICON_INFO,OLEMSGBUTTON.OLEMSGBUTTON_OK,OLEMSGDEFBUTTON.OLEMSGDEFBUTTON_FIRST);}//输出日志private void Log(string msg){Log(0,msg);}private void Log(int level,string msg){ThreadHelper.ThrowIfNotOnUIThread();IVsOutputWindowPane pane = (IVsOutputWindowPane)Package.GetGlobalService(typeof(SVsGeneralOutputWindowPane));int tmp = pane.Activate();if (VSConstants.S_OK != tmp){ShowMessageBox("注意", "未能激活输出窗口 " + tmp.ToString());}for (int i = 0; i < level; ++i){pane.OutputStringThreadSafe("    ");}pane.OutputStringThreadSafe(msg + "\r\n");}private void AddFunction_myToString(int level, VCCodeElements codeElements){ThreadHelper.ThrowIfNotOnUIThread();IVsOutputWindowPane pane = (IVsOutputWindowPane)Package.GetGlobalService(typeof(SVsGeneralOutputWindowPane));string px = new string(' ', level * 4);string px2 = new string(' ', 4);px = level.ToString() + px;string fun_name = "myToString";string fun_type = "stringstream &";vsCMFunction fun_kind = vsCMFunction.vsCMFunctionFunction | vsCMFunction.vsCMFunctionConstant;foreach (VCCodeElement element in codeElements){Log(px + "Kind " + element.Kind.ToString() + " Name " + element.Name);if (0 != element.Children.Count){AddFunction_myToString(level + 1, element.Children as VCCodeElements);}if (element.Kind == vsCMElement.vsCMElementClass || element.Kind == vsCMElement.vsCMElementStruct){VCCodeElement Found = null;string bodytext = "";//基类必须首先处理foreach (VCCodeElement chileren in element.Children){if (chileren.Kind == vsCMElement.vsCMElementVCBase){Log(px + px2 + "基类 " + chileren.Name);bodytext += "\t\t " + chileren.Name + "::myToString(ss) << \" \";\r\n";}}foreach (VCCodeElement chileren in element.Children){Log(px + px2 + chileren.Name + " Kind " + chileren.Kind);if (chileren.Kind == vsCMElement.vsCMElementVariable){VCCodeVariable variable = (VCCodeVariable)chileren;Log(px + px2 + "变量 Name " + variable.Name + " TypeString " + variable.TypeString+ " StartPoint " + variable.StartPoint.Line + " " + variable.StartPoint.LineCharOffset+ " EndPoint " + variable.EndPoint.Line + " " + variable.EndPoint.LineCharOffset);if (variable.TypeString.EndsWith(")")){bodytext += "\t\t ss << \"函数指针 " + variable.Name + " \" << " + variable.Name + " << \" \";\r\n";}else if (variable.TypeString.EndsWith("]")){bodytext += "\t\t ss << \"数组 " + variable.Name + " \" << " + variable.Name + " << \" \";\r\n";}else{bodytext += "\t\t Template_" + fun_name + "(" + variable.Name + ", ss) << \" \";\r\n";}}else if (chileren.Kind == vsCMElement.vsCMElementFunction){if (chileren.Name == fun_name){Found = chileren;Log(px + px2 + fun_name + " 已存在,重新创建");}}}VCCodeFunction codeFunction;if (element.Kind == vsCMElement.vsCMElementClass){VCCodeClass codeClass = (VCCodeClass)element;codeClass.RemoveMember(Found);codeFunction = (VCCodeFunction)codeClass.AddFunction(fun_name, fun_kind, fun_type, -1, vsCMAccess.vsCMAccessPublic);}else{VCCodeStruct codeClass = (VCCodeStruct)element;codeClass.RemoveMember(Found);codeFunction = (VCCodeFunction)codeClass.AddFunction(fun_name, fun_kind, fun_type, -1, vsCMAccess.vsCMAccessPublic);}codeFunction.AddParameter("ss", "stringstream &");codeFunction.Comment = "自动生成的代码";bodytext += "\t\t return ss;";codeFunction.BodyText = bodytext;}}}private void CSharp_Rename(int level, CodeElements codeElements){ThreadHelper.ThrowIfNotOnUIThread();foreach (CodeElement _element in codeElements){Log(level, "Kind " + _element.Kind.ToString());string name = "未知";//Name属性不是每个都有if (_element.Kind == vsCMElement.vsCMElementImportStmt){name = "vsCMElementImportStmt";}else{name = _element.Name;//这个竟然不是每个都支持}CodeElement2 element = (CodeElement2)_element;Log(level, "Kind " + element.Kind.ToString() + " Name " + name + " type " + element.GetType().ToString());//处理子项if (0 != element.Children.Count){CSharp_Rename(level + 1, element.Children);}bool skip = false;//是否需要跳过//检查是否已经处理过if (name.StartsWith(new_name_title)){skip = true;}if (element.Kind == vsCMElement.vsCMElementVariable){CodeVariable variable = (CodeVariable)element;Log(level + 1, "变量 Name " + variable.Name+ " StartPoint " + variable.StartPoint.Line + " " + variable.StartPoint.LineCharOffset+ " EndPoint " + variable.EndPoint.Line + " " + variable.EndPoint.LineCharOffset);}else if (element.Kind == vsCMElement.vsCMElementFunction){Log(level + 1, "函数 " + name);if (name.Equals("Main")){Log(level + 1, "Main函数(跳过) " + name);skip = true;}if (name.Equals("Dispose")){Log(level + 1, "Dispose函数(跳过) " + name);skip = true;}}else if (element.Kind == vsCMElement.vsCMElementNamespace){Log(level + 1, "命名空间 " + name);//skip = true;}else if (element.Kind == vsCMElement.vsCMElementAttribute){Log(level + 1, "属性(跳过) " + name);skip = true;}else if (element.Kind == vsCMElement.vsCMElementImportStmt){Log(level + 1, "导入语句(跳过) " + name);skip = true;}else if (element.Kind == vsCMElement.vsCMElementOther){Log(level + 1, "vsCMElementOther(跳过) " + name);skip = true;}if (!skip){Log(level, "重命名 " + name + "(" + element.Kind.ToString() + ") 为 " + new_name_title + count.ToString());element.RenameSymbol(new_name_title + count.ToString());count++;Log(level, "重命名完成");}}}private void ProcessProjectItem(int level, ProjectItem projectItem){ThreadHelper.ThrowIfNotOnUIThread();//项目下的筛选器Log(level, "===============目录:" + projectItem.Name + " 项目子项FileCount:" + projectItem.FileCount.ToString());for (short i = 0; i < projectItem.FileCount; i++){Log(3, "文件名:" + projectItem.FileNames[i]);}if (projectType == ProjectType.CSharp && projectItem.Name == "Properties"){Log(3, "C#项目忽略属性目录");return;}if (null != projectItem.FileCodeModel){String language = "未知语言";switch (projectItem.FileCodeModel.Language){case CodeModelLanguageConstants.vsCMLanguageVC:language = "VC";AddFunction_myToString(5, projectItem.FileCodeModel.CodeElements as VCCodeElements);break;case CodeModelLanguageConstants.vsCMLanguageIDL:language = "IDL";Log(3, "未支持的语言 " + language);break;case CodeModelLanguageConstants.vsCMLanguageVB:language = "VB";Log(3, "未支持的语言 " + language);break;case CodeModelLanguageConstants.vsCMLanguageMC:language = "MC";Log(3, "未支持的语言 " + language);break;case CodeModelLanguageConstants.vsCMLanguageCSharp:language = "CSharp";Log(3, "语言 " + language);if (null == projectItem) Log(3, "语言1" + language);if (null == projectItem.FileCodeModel) Log(3, "语言 2" + language);if (null == projectItem.FileCodeModel.CodeElements) Log(3, "语言 3" + language);Log(3, "语言 " + language);CSharp_Rename(5, projectItem.FileCodeModel.CodeElements);break;}}foreach (ProjectItem current_project_item_item in projectItem.ProjectItems){ProcessProjectItem(level + 1, current_project_item_item);}}public void Execute(){ThreadHelper.ThrowIfNotOnUIThread();string message = string.Format(CultureInfo.CurrentCulture, "Inside {0}.MenuItemCallback()", this.GetType().FullName);string title = "Command1 2023-04-20 1720";// Show a message box to prove we were here//VsShellUtilities.ShowMessageBox(//    this.package,//    message,//    title,//    OLEMSGICON.OLEMSGICON_INFO,//    OLEMSGBUTTON.OLEMSGBUTTON_OK,//    OLEMSGDEFBUTTON.OLEMSGDEFBUTTON_FIRST);Log(title);try{DTE2 dte = (DTE2)Package.GetGlobalService(typeof(SDTE));Log("DTE:" + dte.Version);Log("DTE:" + dte.Name);Log("DTE:" + dte.Edition);Log("DTE:" + dte.Mode);var solution = dte.Solution;var SolutionName = Path.GetFileName(solution.FullName);     //解决方案名称var SolutionPath = Path.GetDirectoryName(solution.FullName);//解决方案路径Log("解决方案:" + solution.ToString());Log("解决方案FileName:" + solution.FileName);Log("解决方案FullName:" + solution.FullName);Log("解决方案GetFileName:" + SolutionName);Log("解决方案GetDirectoryName:" + SolutionPath);Log("解决方案Count:" + solution.Count);Log("解决方案Projects.Count:" + solution.Projects.Count);foreach (Project current_project in solution.Projects){//解决方案下的项目Log(1, "--------------------------Language:" + current_project.CodeModel.Language);if (current_project.CodeModel.Language == "{B5E9BD34-6D3E-4B5D-925E-8A43B79820B4}"){projectType = ProjectType.CSharp;}else if (current_project.CodeModel.Language == "{B5E9BD32-6D3E-4B5D-925E-8A43B79820B4}"){projectType = ProjectType.VC;}else{projectType = ProjectType.OTHER;}Log(1, "--------------------------项目:" + current_project.Name + " 类型 " + projectType + " 项目子项个数:" + current_project.ProjectItems.Count.ToString());foreach (ProjectItem current_project_item in current_project.ProjectItems){ProcessProjectItem(2, current_project_item);}}ShowMessageBox(title, "操作完成");}catch (Exception ex){ShowMessageBox("", ex.Message);}}}
}

        这个代码对C#项目执行CSharp_Rename,对C++项目则执行AddFunction_myToString,功能是给所有结构添加toString函数,尚不完善,所以无视即可(我挺希望别人能把这个功能做出来)。

        代码分析看这里:VSIX:C#项目 重命名所有标识符(Visual Studio扩展开发)代码详解-CSDN博客。

(这里是结束)

相关文章:

VSIX:C#项目 重命名所有标识符(Visual Studio扩展开发)

出于某种目的&#xff08;合法的&#xff0c;真的合法的&#xff0c;合同上明确指出可以这样做&#xff09;&#xff0c;我准备了一个重命名所有标识符的VS扩展&#xff0c;用来把一个C#库改头换面&#xff0c;在简单的测试项目上工作很满意&#xff0c;所有标识符都被准确替换…...

【CSDN 每日一练 ★★☆】【动态规划】最小路径和

【CSDN 每日一练 ★★☆】【动态规划】最小路径和 动态规划 题目 给定一个包含非负整数的 m x n 网格 grid &#xff0c;请找出一条从左上角到右下角的路径&#xff0c;使得路径上的数字总和为最小。 说明&#xff1a;每次只能向下或者向右移动一步。 示例 示例 1&#x…...

前端学习之webpack的使用

概述 webpack是一个流行的前端项目构建工具&#xff08;打包工具&#xff09;&#xff0c;可以解决当前web开发中所面临的问题。 webpack提供了友好的模块化支持&#xff0c;以及代码压缩混淆、处理js兼容问题、性能优化等强大的功能&#xff0c;从而让程序员把工作重心放到具…...

【java学习—十一】泛型(1)

文章目录 1. 为什么要有泛型Generic2. 泛型怎么用2.1. 泛型类2.2. 泛型接口2.3. 泛型方法 3. 泛型通配符3.1. 通配符3.2. 有限制的通配符 1. 为什么要有泛型Generic 泛型&#xff0c;JDK1.5新加入的&#xff0c;解决数据类型的安全性问题&#xff0c;其主要原理是在类声明时通过…...

CN考研真题知识点二轮归纳(4)

持续更新&#xff0c;上期目录&#xff1a; CN考研真题知识点二轮归纳&#xff08;4&#xff09;https://blog.csdn.net/jsl123x/article/details/134135134?spm1001.2014.3001.5501 1.既可以扩展网段又是二层的设备 网段一般指一个计算机网络中使用同一物理层设备&#xff…...

ROS学习笔记(4):ROS架构和通讯机制

前提 前4篇文章以及帮助大家快速入门ROS了&#xff0c;而从第5篇开始我们会更加注重知识积累。同时我强烈建议配合B站大学的视频一起服用。 1.ROS架构三层次&#xff1a; 1.基于Linux系统的OS层&#xff1b; 2.实现ROS核心通信机制以及众多机器人开发库的中间层&#xff1b…...

深度新闻稿件怎么写?新闻稿怎么写得有深度?

深度新闻稿件&#xff0c;顾名思义&#xff0c;是对新闻事件进行深入挖掘和分析的稿件。它不仅仅是对事件的简单报道&#xff0c;更注重对事件背后的社会现象、原因、影响等方面进行深度剖析&#xff0c;从而使读者能够全面、深入地了解事件。这种稿件要求作者具备较高的新闻敏…...

百度智能云千帆大模型平台黑客马拉松报名开启!

比赛简介 创造是生成式 AI 的核心。无论是智能导购带来的线上购物体验升级&#xff0c;还是主图生成带来的素材生产效率提升&#xff0c;又或是游戏场景的快速设置、智能 NPC 的全新交互、数字广告的精准推荐和个性化定制&#xff0c;亦或者是为学生提供更符合真实的口语练习环…...

数据库 | 看这一篇就够了!最全MySQL数据库知识框架!

大家好&#xff01; 作为一名程序员&#xff0c;每天和各种各样的“数据库”打交道&#xff0c;已经成为我们的日常。当然&#xff0c;立志成为一名超级架构师的我&#xff0c;肯定要精通这项技能。咳咳&#xff01;不过饭还是要一口一口吃的&#xff0c;“数据库”这个内容实在…...

Android 控件背景实现发光效果

主要实现的那种光晕效果&#xff1a;中间亮&#xff0c;四周逐渐变淡的。 这边有三种发光效果&#xff0c;先上效果图。 第一种、圆形发光体 实现代码&#xff1a;新建shape_light.xml&#xff0c;导入以下代码。使用时&#xff0c;直接给view设置为background。 <?xml …...

安全狗亮相厦门市工信领域数据安全宣贯培训会

10月31日&#xff0c;厦门市工业和信息化局&#xff08;市大数据管理局&#xff09;顺利举办厦门市工信领域数据安全宣贯培训。 作为国内云原生安全领导厂商&#xff0c;安全狗以厦门市工业领域数据安全管理支撑单位身份受邀出席此次会议。 据悉&#xff0c;此次活动旨在贯彻…...

最长回文子串

问题 给你一个字符串 s&#xff0c;找到 s 中最长的回文子串。 如果字符串的反序与原始字符串相同&#xff0c;则该字符串称为回文字符串。 示例 1&#xff1a; 输入&#xff1a;s "babad" 输出&#xff1a;"bab" 解释&#xff1a;"aba" 同…...

从瀑布模式到水母模式:ChatGPT引领软件研发的革新之路

ChatGPT引领软件研发的革新之路 概述操作建议本书优势 内容简介作者简介专家推荐读者对象目录直播预告写在末尾&#xff1a; 主页传送门&#xff1a;&#x1f4c0; 传送 概述 计算机技术的发展和互联网的普及&#xff0c;使信息处理和传输变得更加高效&#xff0c;极大地改变了…...

一种使用wireshark快速分析抓包文件amr音频流的思路方法

解决方案&#xff1a; 1. 使用wireshark过滤amr,并导出原始数据文件&#xff1b; 2.使用ue的二进制编辑模式&#xff0c;编辑该文件&#xff0c;添加amr头&#xff0c;6个字节数据“#!AMR”&#xff0c;字节数据为 23 21 41 4D 52 0A 3.修正格式&#xff1a;通过抓包发现&#…...

银河麒麟x86版、银河麒麟arm版操作系统编译zlmediakit

脚本 # 安装依赖 gcc-c.x86_64 这个不加的话会有问题 sudo yum -y install gcc gcc-c libssl-dev libsdl-dev libavcodec-dev libavutil-dev ffmpeg git openssl-devel gcc-c.x86_64mkdir -p /home/zenglg cd /home/zenglg git clone --depth 1 https://gitee.com/xia-chu…...

InnoDB - 双写机制

双写机制用于提高数据持久性和可靠性。 双写机制的核心思想是&#xff0c;将写操作先写入一个临时缓冲区&#xff0c;然后再写入实际的数据文件。这个临时缓冲区通常是固定大小的内存缓冲区&#xff0c;称为双写缓冲。这个机制的主要目的是避免数据文件在写入时出现损坏或数据…...

【蓝桥杯选拔赛真题08】C++最大值最小值平均值 青少年组蓝桥杯C++选拔赛真题 STEMA比赛真题解析

目录 C/C++最大值最小值平均值 一、题目要求 1、编程实现 2、输入输出 二、算法分析</...

软考高级系统架构设计师系列之:系统开发基础知识、项目管理、信息安全和网络安全、计算机网络章节选择题详解

软考高级系统架构设计师系列之:系统开发基础知识、项目管理、信息安全和网络安全、计算机网络章节选择题详解 一、产品配置二、需求管理三、需求跟踪四、软件生命周期五、RUP六、耦合与内聚七、软件文档八、软件需求九、软件活动十、项目时间管理十一、需求管理十二、项目范围…...

0基础学习PyFlink——时间滑动窗口(Sliding Time Windows)

在《0基础学习PyFlink——时间滚动窗口(Tumbling Time Windows)》我们介绍了不会有重复数据的时间滚动窗口。本节我们将介绍存在重复计算数据的时间滑动窗口。 关于滑动窗口&#xff0c;可以先看下《0基础学习PyFlink——个数滑动窗口&#xff08;Sliding Count Windows&#x…...

API安全之《大话:API的前世今生》

写在前面&#xff1a;本文结合API使用的业界现状&#xff0c;系统性地阐述API的基本概念、发展历史、表现形式等基础内容&#xff0c;主要包含以下内容&#xff1a; 1.什么是API 2.API的发展历史 3.现代API常用消息格式 4.top N 互联网企业API 使用现状 当前的世界是一个信…...

DeepSeek 赋能智慧能源:微电网优化调度的智能革新路径

目录 一、智慧能源微电网优化调度概述1.1 智慧能源微电网概念1.2 优化调度的重要性1.3 目前面临的挑战 二、DeepSeek 技术探秘2.1 DeepSeek 技术原理2.2 DeepSeek 独特优势2.3 DeepSeek 在 AI 领域地位 三、DeepSeek 在微电网优化调度中的应用剖析3.1 数据处理与分析3.2 预测与…...

MFC内存泄露

1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...

【网络安全产品大调研系列】2. 体验漏洞扫描

前言 2023 年漏洞扫描服务市场规模预计为 3.06&#xff08;十亿美元&#xff09;。漏洞扫描服务市场行业预计将从 2024 年的 3.48&#xff08;十亿美元&#xff09;增长到 2032 年的 9.54&#xff08;十亿美元&#xff09;。预测期内漏洞扫描服务市场 CAGR&#xff08;增长率&…...

CentOS下的分布式内存计算Spark环境部署

一、Spark 核心架构与应用场景 1.1 分布式计算引擎的核心优势 Spark 是基于内存的分布式计算框架&#xff0c;相比 MapReduce 具有以下核心优势&#xff1a; 内存计算&#xff1a;数据可常驻内存&#xff0c;迭代计算性能提升 10-100 倍&#xff08;文档段落&#xff1a;3-79…...

蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练

前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1)&#xff1a;从基础到实战的深度解析-CSDN博客&#xff0c;但实际面试中&#xff0c;企业更关注候选人对复杂场景的应对能力&#xff08;如多设备并发扫描、低功耗与高发现率的平衡&#xff09;和前沿技术的…...

2.Vue编写一个app

1.src中重要的组成 1.1main.ts // 引入createApp用于创建应用 import { createApp } from "vue"; // 引用App根组件 import App from ./App.vue;createApp(App).mount(#app)1.2 App.vue 其中要写三种标签 <template> <!--html--> </template>…...

Cinnamon修改面板小工具图标

Cinnamon开始菜单-CSDN博客 设置模块都是做好的&#xff0c;比GNOME简单得多&#xff01; 在 applet.js 里增加 const Settings imports.ui.settings;this.settings new Settings.AppletSettings(this, HTYMenusonichy, instance_id); this.settings.bind(menu-icon, menu…...

数据链路层的主要功能是什么

数据链路层&#xff08;OSI模型第2层&#xff09;的核心功能是在相邻网络节点&#xff08;如交换机、主机&#xff09;间提供可靠的数据帧传输服务&#xff0c;主要职责包括&#xff1a; &#x1f511; 核心功能详解&#xff1a; 帧封装与解封装 封装&#xff1a; 将网络层下发…...

DBAPI如何优雅的获取单条数据

API如何优雅的获取单条数据 案例一 对于查询类API&#xff0c;查询的是单条数据&#xff0c;比如根据主键ID查询用户信息&#xff0c;sql如下&#xff1a; select id, name, age from user where id #{id}API默认返回的数据格式是多条的&#xff0c;如下&#xff1a; {&qu…...

智能仓储的未来:自动化、AI与数据分析如何重塑物流中心

当仓库学会“思考”&#xff0c;物流的终极形态正在诞生 想象这样的场景&#xff1a; 凌晨3点&#xff0c;某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径&#xff1b;AI视觉系统在0.1秒内扫描包裹信息&#xff1b;数字孪生平台正模拟次日峰值流量压力…...