unity一键注释日志和反注释日志
开发背景:游戏中日志也是很大的开销,虽然有些日志不打印但是毕竟有字符串的开销,甚至有字符串拼接的开销,有些还有装箱和拆箱的开销,比如Debug.Log(1) 这种
因此需要注释掉,当然还需要提供反注释的功能,需要的时候能立马找回来。
也就是说我们只需要打包的时候处理即可。
1.先检测代码中是否存在不规范代码,类似if(a > 1) Debug.Log("11") 这种代码如果注释掉日志会引起业务逻辑问题需要先找出来,类似的不限于if else for foreach while这些 所以尽可能的要做这个检测,有这类问题直接报错。然后才可以进行后续流程
2.注释Debug.Log或者UnityEngine.Debug.Log,但是有一个问题就是注释之后再注释的问题,所以在调用这个接口前需要调用反注释然后调用注释,这样就可以保证注释是没有问题的。
3.需要提供反注释的代码,主要是配合2的功能
注意事项:大家写代码的过程中想打日志就打日志这个没有限制,打包的时候会自动处理日志。
如果是特别的日志可以打错误日志或者警告,或者自定义日志那种日志不去处理即可。
未来规划是用代码分析给日志加标签,平常开发还是按unity的习惯来我绝对不改变大家的unity标准,在做扩展的时候尽量做到神不知鬼不觉,不让大家有额外操作不让大家有压力。
接着上面扩展 :如下图所示这种else日志不被允许

那么如何进行csharp的语法分析呢 ,我们建一个c#的控制台程序,使用nuget安装3个包
Microsoft.CodeAnalysis.CSharp 和 Microsoft.CodeAnalysis.CSharp.Workspaces 和Microsoft.CodeAnalysis
分析c#代码检测是否存在if else for while foreach 后直接接Debug.Log或者UnityEngine.Debug.Log完整控制台程序如下
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using Microsoft.CodeAnalysis.CSharp.Syntax;
namespace cs_test
{
internal class Program
{
public static long getCurrentTimeMillis()
{
long milliseconds = System.DateTime.Now.Ticks / System.TimeSpan.TicksPerMillisecond;
return milliseconds;
}
static long firstTime = 0;
const float LogWritetimeOutTime = 2f;
static List<string> strList = new List<string>();
static void SetTimer(float time = LogWritetimeOutTime)
{
System.Timers.Timer timer = new System.Timers.Timer();
timer.Interval = time;
timer.AutoReset = false;
timer.Enabled = true;
timer.Elapsed += (a, b) =>
{
using (var fs = new FileStream(logPath, FileMode.Append,FileAccess.Write,FileShare.Write))
{
using (var sw = new StreamWriter(fs, Encoding.UTF8))
{
lock (strList)
{
foreach (var item in strList)
{
sw.WriteLine(item);
}
strList.Clear();
}
}
}
};
}
static void Log(string str,bool useFileSystem = true)
{
if(useFileSystem && isLogToFile)
{
lock (strList)
{
strList.Add(str);
}
if (firstTime == 0)
{
firstTime = getCurrentTimeMillis();
SetTimer(LogWritetimeOutTime);
return;
}
if (getCurrentTimeMillis() - firstTime > LogWritetimeOutTime)
{
firstTime = 0;
SetTimer(LogWritetimeOutTime);
}
}
else
{
Console.WriteLine(str);
}
}
public class LogStatementWalker : CSharpSyntaxWalker
{
public List<Issue> Issues { get; } = new List<Issue>();
public override void VisitIfStatement(IfStatementSyntax node)
{
CheckBlockStatement(node.Statement, node.GetLocation());
if (node.Else != null)
{
CheckElseStatement(node.Else);
}
base.VisitIfStatement(node);
}
public override void VisitForStatement(ForStatementSyntax node)
{
CheckBlockStatement(node.Statement, node.GetLocation());
base.VisitForStatement(node);
}
public override void VisitForEachStatement(ForEachStatementSyntax node)
{
CheckBlockStatement(node.Statement, node.GetLocation());
base.VisitForEachStatement(node);
}
public override void VisitWhileStatement(WhileStatementSyntax node)
{
CheckBlockStatement(node.Statement, node.GetLocation());
base.VisitWhileStatement(node);
}
private void CheckElseStatement(ElseClauseSyntax elseClause)
{
if (elseClause.Statement is IfStatementSyntax)
{
// Skip checking if it's an "else if" statement
return;
}
CheckBlockStatement(elseClause.Statement, elseClause.GetLocation());
}
private void CheckBlockStatement(StatementSyntax statement, Location location)
{
if (statement is BlockSyntax)
{
return;
}
var logStatements = statement.DescendantNodes()
.OfType<InvocationExpressionSyntax>()
.Where(inv => inv.Expression is MemberAccessExpressionSyntax memberAccess &&
(memberAccess.Name.ToString() == "Log" || memberAccess.Name.ToString() == "Debug.Log"))
.ToList();
foreach (var logStatement in logStatements)
{
var lineSpan = logStatement.GetLocation().GetLineSpan();
Issues.Add(new Issue
{
LineNumber = lineSpan.StartLinePosition.Line + 1,
Message = $"Debug.Log or UnityEngine.Debug.Log found in a single-line {statement.Kind()} statement without braces."
});
}
}
}
public class Issue
{
public int LineNumber { get; set; }
public string Message { get; set; }
}
static string unityRoot;
static string logPath = "D:/tempLog.txt";
static bool isLogToFile = true;
static void Main(string[] args)
{
List<string> pathList = new List<string>();
if (args == null || args.Length == 0)
{
var path = System.Environment.CurrentDirectory;
path = path.Substring(0, path.IndexOf("tools"));
path = path.Replace("\\", "/");
args = new string[]
{
path + "Assets/HotUpdate"
};
}
unityRoot = args[0].Substring(0, args[0].IndexOf("Assets"));
logPath = unityRoot + "tools/tempLog.txt";
if(File.Exists(logPath))
{
File.Delete(logPath);
}
pathList.AddRange(args);
isLogToFile = bool.Parse(pathList[pathList.Count - 1]);
pathList.RemoveAt(pathList.Count - 1);
List<string> allCsharpFiles = new List<string>();
foreach(var path_item in pathList)
{
allCsharpFiles.AddRange(Directory.GetFiles(path_item, "*.cs", SearchOption.AllDirectories));
}
bool hasInsue = false;
strList.Clear();
foreach(var item in allCsharpFiles)
{
string sourceCode = File.ReadAllText(item);
SyntaxTree tree = CSharpSyntaxTree.ParseText(sourceCode);
CompilationUnitSyntax root = tree.GetCompilationUnitRoot();
var diagnostics = tree.GetDiagnostics();
if (diagnostics.Any(d => d.Severity == DiagnosticSeverity.Error))
{
Log("Syntax errors found in the source code.");
return;
}
var walker = new LogStatementWalker();
walker.Visit(root);
foreach (var issue in walker.Issues)
{
hasInsue = true;
var itemName = item.Substring(item.IndexOf("Assets"));
Log($"请使用括号封装你的日志! at line {issue.LineNumber}: {itemName}");
}
}
if(!hasInsue)
{
Log("未发现不规范代码");
}
}
/// <summary>
/// 检测判断if语句后面是否带花括号,由于在for循环中经常会遇到if不带花括号的情况,故这个检测暂时放弃 不然
/// 会有非常多的代码提示
/// 这里的代码暂时封存
/// </summary>
/// <param name="args"></param>
static void Main2(string[] args)
{
if(File.Exists(logPath))
{
File.Delete(logPath);
}
File.Create(logPath).Dispose();
string directoryPath = "D:\\qiangsheng_wx\\Assets\\HotUpdate"; // 替换为你的目录路径
// 获取目录中的所有 .cs 文件
string[] files = Directory.GetFiles(directoryPath, "*.cs", SearchOption.AllDirectories);
foreach (string filePath in files)
{
Log($"Processing file: {filePath}");
string code = File.ReadAllText(filePath);
SyntaxTree tree = CSharpSyntaxTree.ParseText(code);
var root = tree.GetRoot();
var walker = new IfElseSyntaxWalker();
walker.Visit(root);
}
Console.ReadLine();
}
class IfElseSyntaxWalker : CSharpSyntaxWalker
{
public override void VisitIfStatement(IfStatementSyntax node)
{
base.VisitIfStatement(node);
// 检查 if 语句是否有花括号
if (node.Statement is BlockSyntax) { }
else
{
var str = $"警告: if 语句 at line {node.GetLocation().GetLineSpan().StartLinePosition.Line + 1} does not use 花括号";
Log(str);
}
// 检查 else 语句是否有花括号
if (node.Else != null)
{
if (node.Else.Statement is BlockSyntax) { }
else
{
var str = $"警告: else 语句 at line {node.Else.GetLocation().GetLineSpan().StartLinePosition.Line + 1} does not use 花括号";
Log(str);
}
}
}
}
}
}
控制台程序做完之后还需要对接到unity项目中,我是直接将微信api的代码分析回调日志直接写入了某个文件,让unity调用这个exe文件,你可以封装成bat将参数传给exe执行,unity那边读取文件将文本内容Debug.LogError输出即可。
上诉工作做完了之后就可以提供注释和反注释的代码接口了,将下述代码封装到你的编辑器工具类中或者某个编辑器类下的对象中
private const string LogPattern = @"(?s)(Debug\.Log\s*\(.*?\);|UnityEngine\.Debug\.Log\s*\(.*?\);)";
private const string unLogPattern = @"(?s)/\*(\s*Debug\.Log\s*\(.*?\);\s*|UnityEngine\.Debug\.Log\s*\(.*?\);\s*)\*/";
public static void ForbidLog(string filePath)
{
string fileContent = File.ReadAllText(filePath,Encoding.UTF8);
// 正则表达式匹配 Debug.Log 调用及其相关代码
string pattern = LogPattern;
string replacement = @"/*$1*/";
if(!Regex.IsMatch(fileContent, pattern))
{
return;
}
string newContent = Regex.Replace(fileContent, pattern, replacement);
using (StreamWriter writer = new StreamWriter(filePath, false, new UTF8Encoding(false)))
{
writer.Write(newContent);
}
}
public static void UnForbidLog(string filePath)
{
string fileContent = File.ReadAllText(filePath, Encoding.UTF8);
// 正则表达式匹配被注释的 Debug.Log 调用及其相关代码
string pattern = unLogPattern;
string replacement = @"$1";
if (!Regex.IsMatch(fileContent, pattern))
{
return;
}
string newContent = Regex.Replace(fileContent, pattern, replacement);
using (StreamWriter writer = new StreamWriter(filePath, false, new UTF8Encoding(false)))
{
writer.Write(newContent);
}
}
[MenuItem("Tools/日志/注释所有日志", priority = -1999)]
public static void ReMoveLog()
{
var listFolder = new List<string>();
listFolder.Add(System.Environment.CurrentDirectory + "/Assets/HotUpdate");
listFolder.Add(System.Environment.CurrentDirectory + "/Assets/SDKTool");
listFolder.Add(System.Environment.CurrentDirectory + "/Assets/DevWork");
var allCsharpFiles = new List<string>();
foreach(string folder in listFolder)
{
allCsharpFiles.AddRange(Directory.GetFiles(folder, "*.cs",SearchOption.AllDirectories));
}
string[] filters = new string[]
{
"InitCtrl.cs",
"Global.cs",
"LoginPanel.cs",
"LoginCode.cs",
"ProductModel.cs",
};
foreach(string csFile in allCsharpFiles)
{
var filename = Path.GetFileName(csFile);
if(filters.Contains(filename))
{
Debug.Log("跳过过滤文件" + csFile);
continue;
}
ForbidLog(csFile);
}
allCsharpFiles.Clear();
listFolder.Clear();
listFolder = null;
allCsharpFiles = null;
AssetDatabase.SaveAssets();
AssetDatabase.Refresh();
}
[MenuItem("Tools/日志/反注释所有日志", priority = -1999)]
public static void UnReMoveLog()
{
var listFolder = new List<string>();
listFolder.Add(System.Environment.CurrentDirectory + "/Assets/HotUpdate");
listFolder.Add(System.Environment.CurrentDirectory + "/Assets/SDKTool");
listFolder.Add(System.Environment.CurrentDirectory + "/Assets/DevWork");
var allCsharpFiles = new List<string>();
foreach (string folder in listFolder)
{
allCsharpFiles.AddRange(Directory.GetFiles(folder, "*.cs", SearchOption.AllDirectories));
}
foreach (string csFile in allCsharpFiles)
{
UnForbidLog(csFile);
}
allCsharpFiles.Clear();
listFolder.Clear();
listFolder = null;
allCsharpFiles = null;
AssetDatabase.SaveAssets();
AssetDatabase.Refresh();
}
运行截图:

相关文章:
unity一键注释日志和反注释日志
开发背景:游戏中日志也是很大的开销,虽然有些日志不打印但是毕竟有字符串的开销,甚至有字符串拼接的开销,有些还有装箱和拆箱的开销,比如Debug.Log(1) 这种 因此需要注释掉,当然还需要提供反注释的功能&am…...
VBA数据库解决方案第十五讲:Recordset集合中单个数据的精确处理
《VBA数据库解决方案》教程(版权10090845)是我推出的第二套教程,目前已经是第二版修订了。这套教程定位于中级,是学完字典后的另一个专题讲解。数据库是数据处理的利器,教程中详细介绍了利用ADO连接ACCDB和EXCEL的方法…...
甄选范文“论软件需求管理”,软考高级论文,系统架构设计师论文
论文真题 软件需求管理是一个对系统需求变更了解和控制的过程。需求管理过程与需求开发过程相互关联,初始需求导出的同时就要形成需求管理规划,一旦启动了软件开发过程,需求管理活动就紧密相伴。 需求管理过程中主要包含变更控制、版本控制、需求跟踪和需求状态跟踪等4项活…...
Android Studio Dolphin 中Gradle下载慢的解决方法
我用的版本Android Studio Dolphin | 2021.3.1 Patch 1 1.Gradle自身的版本下载慢 解决办法:修改gradle\wrapper\gradle-wrapper.properties中的distributionUrl 将https\://services.gradle.org/distributions为https\://mirrors.cloud.tencent.com/gradle dis…...
Excel实现省-市-区/县级联
数据准备 准备省份-城市映射数据,如下: 新建sheet页,命名为:省-市数据源,然后准备数据,如下所示: 准备城市-区|县映射数据,如下: 新建sheet页,命名为&#x…...
【优化代码结构】函数的参数归一化
某些封装的函数,其参数具有多样性,会导致函数中会增加非常多的分支,比如下面这个 format 函数有如下几种参数方式,其中 formatter 会有很多种情况 date:日期对象formatter: ‘date’:格式化日期…...
CSS中height设置100vh和100%的区别
文章目录 CSS中height设置100vh和100%的区别一、引言二、高度设置的区别1、100%1.1、父元素高度固定1.2、父元素高度未定义 2、100vh2.1、视口高度2.2、不受父元素限制 三、总结 CSS中height设置100vh和100%的区别 一、引言 在前端开发中,我们经常需要设置元素的高…...
红米k60至尊版工程固件 MTK芯片 资源预览 刷写说明 与nv损坏修复去除电阻图示
红米k60至尊版机型代码为:corot。 搭载了联发科天玑9200+处理器。此固件mtk引导为MT6985。博文将简单说明此固件的一些特点与刷写注意事项。对于NV损坏的机型。展示修改校验电阻的图示。方便改写参数等 通过博文了解 1💝💝💝-----此机型工程固件的资源刷写注意事项 2…...
QEMU使用Qemu-Guest-Agent传输文件、执行指令等
简介 之前介绍过qemu传输文件,使用的挂载 / samba方式 :Qemu和宿主机不使用外网进行文件传输。 这是一种方式,这里还有另一种方式:使用Qemu-Guest-Agent,后面简称qga。 官网介绍:https://www.qemu.org/d…...
【漏洞复现】金和OA C6 GeneralXmlhttpPage.aspx Sql注入漏洞
免责声明: 本文旨在提供有关特定漏洞的信息,以帮助用户了解潜在风险。发布此信息旨在促进网络安全意识和技术进步,并非出于恶意。读者应理解,利用本文提到的漏洞或进行相关测试可能违反法律或服务协议。未经授权访问系统、网络或应用程序可能导致法律责任或严重后果…...
复数表示的电场
Exm加是复振幅,这是用复数表示电场,并提取只与空间有关的项复振幅就是复数表示电场,且把与空间xyz有关的量提取出来 经过验证实数E0cos(wtδx)对t求导,等于E0e^j(wtδx)对t求导再取实部 实数表示电磁波cos…...
常用快捷键整理
用加粗标注的是我个人使用时常用的,其实这个全凭个人喜好,大家可以熟悉一下自己喜欢的,都多试试,把觉得有用的记一下,多使用,后续写代码效率就会提高一些) 常用 VS 运行调试程序快捷键 编译 . 编译程序&a…...
【Transformer】长距离依赖
在自然语言处理(NLP)中,长距离依赖(Long-Range Dependencies)指的是在文本中相隔较远的两个或多个元素之间的依赖关系。这些依赖关系可以是语法上的,也可以是语义上的。例如,在句子中࿰…...
Git傻傻分不清楚(下)
进入Idea编译器 File -> New -> Project from Version Control -> URL (这个路径是要拉取项目的Github路径哦~) 设置成maven项目...
golang学习笔记27-反射【重要】
本节也是GO核心部分,很重要。包括基本类型的反射,结构体类型的反射,类别方法Kind(),修改变量的值。 目录 一、概念,基本类型的反射二、结构体类型的反射三、类别方法Kind()四、修改变量的值 一、概念,基本…...
利用Puppeteer-Har记录与分析网页抓取中的性能数据
引言 在现代网页抓取中,性能数据的记录与分析是优化抓取效率和质量的重要环节。本文将介绍如何利用Puppeteer-Har工具记录与分析网页抓取中的性能数据,并通过实例展示如何实现这一过程。 Puppeteer-Har简介 Puppeteer是一个Node.js库,提供…...
YOLOv5改进系列(1)——添加CBAM注意力机制
一、如何理解注意力机制 假设你正在阅读一本书,同时有人在你旁边说话。当你听到某些关键字时,比如“你的名字”或者“你感兴趣的话题”,你会自动把注意力从书上转移到他们的谈话上,尽管你并没有完全忽略书本的内容。这就是注意力机…...
无头单向非循环java版的模拟实现
【本节目标】 1.ArrayList的缺陷 2.链表 1. ArrayList的缺陷 上节课已经熟悉了 ArrayList 的使用,并且进行了简单模拟实现。通过源码知道, ArrayList 底层使用数组来存储元素: public class ArrayList<E> extends AbstractList<…...
Bert Score-文本相似性评估
Bert Score Bert Score 是基于BERT模型的一种方法。它通过计算两个句子在BERT模型中的嵌入编码之间的余弦相似度来评估它们的相似度。BERTScore考虑了上下文信息和语义信息,因此能够更准确地衡量句子之间的相似度。 安装 pip install bert-score 使用例子 一个…...
Pyenv管理Python版本,conda之外的另一套python版本管理解决方案
简介 Pyenv 是一个 python 解释器管理工具,可以对计算机中的多个 python 版本进行管理和切换。为什么要用 pyenv 管理python呢,用过的 python 人都知道,python 虽然是易用而强大的编程语言,但是 python 解释器却有多个版本&#…...
多模态2025:技术路线“神仙打架”,视频生成冲上云霄
文|魏琳华 编|王一粟 一场大会,聚集了中国多模态大模型的“半壁江山”。 智源大会2025为期两天的论坛中,汇集了学界、创业公司和大厂等三方的热门选手,关于多模态的集中讨论达到了前所未有的热度。其中,…...
设计模式和设计原则回顾
设计模式和设计原则回顾 23种设计模式是设计原则的完美体现,设计原则设计原则是设计模式的理论基石, 设计模式 在经典的设计模式分类中(如《设计模式:可复用面向对象软件的基础》一书中),总共有23种设计模式,分为三大类: 一、创建型模式(5种) 1. 单例模式(Sing…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别
UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...
iPhone密码忘记了办?iPhoneUnlocker,iPhone解锁工具Aiseesoft iPhone Unlocker 高级注册版分享
平时用 iPhone 的时候,难免会碰到解锁的麻烦事。比如密码忘了、人脸识别 / 指纹识别突然不灵,或者买了二手 iPhone 却被原来的 iCloud 账号锁住,这时候就需要靠谱的解锁工具来帮忙了。Aiseesoft iPhone Unlocker 就是专门解决这些问题的软件&…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
当仓库学会“思考”,物流的终极形态正在诞生 想象这样的场景: 凌晨3点,某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径;AI视觉系统在0.1秒内扫描包裹信息;数字孪生平台正模拟次日峰值流量压力…...
MySQL账号权限管理指南:安全创建账户与精细授权技巧
在MySQL数据库管理中,合理创建用户账号并分配精确权限是保障数据安全的核心环节。直接使用root账号进行所有操作不仅危险且难以审计操作行为。今天我们来全面解析MySQL账号创建与权限分配的专业方法。 一、为何需要创建独立账号? 最小权限原则…...
NPOI操作EXCEL文件 ——CAD C# 二次开发
缺点:dll.版本容易加载错误。CAD加载插件时,没有加载所有类库。插件运行过程中用到某个类库,会从CAD的安装目录找,找不到就报错了。 【方案2】让CAD在加载过程中把类库加载到内存 【方案3】是发现缺少了哪个库,就用插件程序加载进…...
MySQL 索引底层结构揭秘:B-Tree 与 B+Tree 的区别与应用
文章目录 一、背景知识:什么是 B-Tree 和 BTree? B-Tree(平衡多路查找树) BTree(B-Tree 的变种) 二、结构对比:一张图看懂 三、为什么 MySQL InnoDB 选择 BTree? 1. 范围查询更快 2…...
