程式碼混淆———Proguard使用最新,最全教程,親自試驗
最近公司有一個專案,是外包專案,由於對方也有技術人員,出於技術上的保密,需要對class檔案進行二次處理,於是網上找了好久,只發現Proguard是用的最廣泛而且網上資料最多的。由於不是純JAVA專案,而是WEB專案,涉及到大量的配置檔案,所以用這個工具稍顯吃力,於是開始研究這玩意,花了好長一段時間,重複試驗了N次,終於整出來了,下面總結一下我的經驗。。
首先我介紹下我要混淆的專案框架是jeecg+easyui+spring(包含xml配置檔案,導致部分class檔案不能直接混淆)。下面開始說詳細的操作步奏:
1)將web專案的src目錄的java檔案打包,只需要選擇java檔案即可,其他配置檔案什麼的都不用選擇,如圖
2)到http://proguard.sourceforge.net/下載proguard,目前我下載並使用的是proguard5.1(注:本人下載頻道也有proguard5.1)。
3)解壓proguard5.1,執行 bin目錄下的proguardgui.bat
然後會彈出如下圖所示視窗
4)點選左邊“input/output”選單,然後點選右邊的“Add input”按鈕,新增需要混淆的jar包,我這裡是test.jar,然後點選“add output”,選擇輸出的路徑和包名。
5)下面開始新增支援庫,這個地方很重要,很多同學剛開始使用這個工具的時候就是這裡老是出問題。
點選右邊的“add”。說明一下,這裡最好把你的eclipse裡java project裡的libraries所有Library的jar包,包含web專案lib下面的包,jdk中jre下面的包和servlet.jar包等copy到一個目錄,然後在這裡加入這些jar包。系統預設會帶上rt.jar,這裡我們可以先remove掉,然後到jre下面copy所有的包。
6)點選“shrinking”,設定成如圖所示。
7)點選“obfuscation”,設定如圖所示
8)點選“optimization”設定如圖所示
9)點選information,設定如圖所示,注意選擇jdk版本(Target
10) 點選“process”,再點選“save configuration”,在彈出的對話方塊中,輸入要儲存的配置檔名稱(這裡我的是1111111.pro),最後點選“儲存”.
11) 設定基本完成,關掉proguard視窗,找到剛剛儲存的配置檔案,開始手動修改部分配置。
以下是我的配置檔案,經測試通過,手動修改的部分為紅色字型
-injars Test\test.jar -outjars Test\test--.jar -libraryjars 'D:\jdk1.6.0_45\jre\lib\rt.jar' -libraryjars hunxiao\a\activation-1.1.jar -libraryjars hunxiao\a\activiti-cxf-5.10.jar -libraryjars hunxiao\a\activiti-engine-5.10.jar -libraryjars hunxiao\a\activiti-spring-5.10.jar -libraryjars hunxiao\a\alt-rt.jar -libraryjars hunxiao\a\alt-string.jar -libraryjars hunxiao\a\aopalliance-1.0.jar -libraryjars hunxiao\a\c3p0-0.9.1.2.jar -libraryjars hunxiao\a\charsets.jar -libraryjars hunxiao\a\commons-beanutils-1.9.1.jar -libraryjars hunxiao\a\commons-codec-1.9.jar -libraryjars hunxiao\a\commons-collections-3.2.1.jar -libraryjars hunxiao\a\commons-digester-1.7.jar -libraryjars hunxiao\a\commons-digester3-3.2.jar -libraryjars hunxiao\a\commons-io-2.0.1.jar -libraryjars hunxiao\a\commons-lang3-3.3.jar -libraryjars hunxiao\a\commons-logging-1.1.3.jar -libraryjars hunxiao\a\cos-26Dec2008.jar -libraryjars hunxiao\a\deploy.jar -libraryjars hunxiao\a\dom4j-1.6.1.jar -libraryjars hunxiao\a\druid-0.2.6.jar -libraryjars hunxiao\a\edtftpj.jar -libraryjars hunxiao\a\ehcache-core-2.5.2.jar -libraryjars hunxiao\a\fastjson-1.2.0.jar -libraryjars hunxiao\a\fprzMock.jar -libraryjars hunxiao\a\freemarker-2.3.16.jar -libraryjars hunxiao\a\groovy-all-1.5.5.jar -libraryjars hunxiao\a\guava-16.0.1.jar -libraryjars hunxiao\a\hessian-4.0.7.jar -libraryjars hunxiao\a\itext-2.1.7.jar -libraryjars hunxiao\a\iTextAsian-2.1.jar -libraryjars hunxiao\a\jasperreports-3.7.4.jar -libraryjars hunxiao\a\javaws.jar -libraryjars hunxiao\a\javax.servlet.jsp.jstl-1.2.0.v201105211821.jar -libraryjars hunxiao\a\jce.jar -libraryjars hunxiao\a\jfinal-1.6-bin-with-src.jar -libraryjars hunxiao\a\jfinal-ext-eu.jar -libraryjars hunxiao\a\jna-4.1.0.jar -libraryjars hunxiao\a\jna-platform-4.1.0.jar -libraryjars hunxiao\a\joda-time-2.1.jar -libraryjars hunxiao\a\joor-0.9.3.jar -libraryjars hunxiao\a\jsse.jar -libraryjars hunxiao\a\jxls-core-0.9.9.jar -libraryjars hunxiao\a\kaptcha-2.3.2.jar -libraryjars hunxiao\a\log4j-1.2.16.jar -libraryjars hunxiao\a\management-agent.jar -libraryjars hunxiao\a\mybatis-3.1.1.jar -libraryjars hunxiao\a\mysql-connector-java-5.1.20-bin.jar -libraryjars hunxiao\a\ojdbc6.jar -libraryjars hunxiao\a\org.apache.taglibs.standard.glassfish-1.2.0.v201112081803.jar -libraryjars hunxiao\a\org.springframework.aop-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.asm-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.aspects-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.beans-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.context-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.context.support-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.core-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.expression-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.instrument-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.instrument.tomcat-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.jdbc-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.jms-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.org.apache.commons.logging-1.1.1.jar -libraryjars hunxiao\a\org.springframework.orm-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.oxm-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.test-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.transaction-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.web-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.web.portlet-3.1.1.RELEASE.jar -libraryjars hunxiao\a\org.springframework.web.servlet-3.1.1.RELEASE.jar -libraryjars hunxiao\a\plugin.jar -libraryjars hunxiao\a\poi-3.9.jar -libraryjars hunxiao\a\quartz-1.8.6.jar -libraryjars hunxiao\a\resources.jar -libraryjars hunxiao\a\rt.jar -libraryjars hunxiao\a\servlet-api.jar -libraryjars hunxiao\a\shiro-all-1.2.3.jar -libraryjars hunxiao\a\slf4j-api-1.6.1.jar -libraryjars hunxiao\a\slf4j-log4j12-1.6.1.jar -libraryjars hunxiao\a\sqlite-jdbc-3.7.2.jar -libraryjars hunxiao\a\ssosdk-2.0-SNAPSHOT.jar -libraryjars hunxiao\a\TaxWsBean.jar -target 1.6 -dontshrink -useuniqueclassmembernames -keeppackagenames -keepattributes Exceptions,InnerClasses,Signature,Deprecated,SourceFile,LineNumberTable,LocalVariable*Table,*Annotation*,Synthetic,EnclosingMethod -keepparameternames #保留單個類 -keep public class net.easyunion.common.shiro.ShiroDbRealm -keep public class net.easyunion.common.filters.SetCharacterEncodingFilter -keep public class net.easyunion.common.queue.MakeQueue #保留所有控制類(如果是SSH三大框架,由於頁面發出請求到struts核心攔截器攔截之後,找到配置檔案,配置檔案必須對應action裡面的類和方法,這裡就不能混淆類和方法,所以所有的action類包括裡面的方法都不需要混淆,按照如下方式設定就行,保留所有的Action類名和方法名) -keep public class net.easyunion.app.invoice.controller.* {*;} -keep public class net.easyunion.app.sysseting.controller.* {*;} -keep public class net.easyunion.app.system.controller.* {*;} -keep public class net.easyunion.app.system.model.* {*;} -keep public class net.easyunion.app.supplier.controller.* {*;} -keep public class net.easyunion.common.controller.* {*;} -keep public class net.easyunion.app.Config # Keep names - Native method names. Keep all native class/method names. -keepclasseswithmembers,includedescriptorclasses,allowshrinking class * { native <methods>; } # Keep names - _class method names. Keep all .class method names. This may be # useful for libraries that will be obfuscated again with different obfuscators. -keepclassmembers,allowshrinking class * { java.lang.Class class$(java.lang.String); java.lang.Class class$(java.lang.String,boolean); } # Remove - System method calls. Remove all invocations of System # methods without side effects whose return values are not used. -assumenosideeffects public class java.lang.System { public static long currentTimeMillis(); static java.lang.Class getCallerClass(); public static int identityHashCode(java.lang.Object); public static java.lang.SecurityManager getSecurityManager(); public static java.util.Properties getProperties(); public static java.lang.String getProperty(java.lang.String); public static java.lang.String getenv(java.lang.String); public static java.lang.String mapLibraryName(java.lang.String); public static java.lang.String getProperty(java.lang.String,java.lang.String); } # Remove - Math method calls. Remove all invocations of Math # methods without side effects whose return values are not used. -assumenosideeffects public class java.lang.Math { public static double sin(double); public static double cos(double); public static double tan(double); public static double asin(double); public static double acos(double); public static double atan(double); public static double toRadians(double); public static double toDegrees(double); public static double exp(double); public static double log(double); public static double log10(double); public static double sqrt(double); public static double cbrt(double); public static double IEEEremainder(double,double); public static double ceil(double); public static double floor(double); public static double rint(double); public static double atan2(double,double); public static double pow(double,double); public static int round(float); public static long round(double); public static double random(); public static int abs(int); public static long abs(long); public static float abs(float); public static double abs(double); public static int max(int,int); public static long max(long,long); public static float max(float,float); public static double max(double,double); public static int min(int,int); public static long min(long,long); public static float min(float,float); public static double min(double,double); public static double ulp(double); public static float ulp(float); public static double signum(double); public static float signum(float); public static double sinh(double); public static double cosh(double); public static double tanh(double); public static double hypot(double,double); public static double expm1(double); public static double log1p(double); } # Remove - Number method calls. Remove all invocations of Number # methods without side effects whose return values are not used. -assumenosideeffects public class java.lang.* extends java.lang.Number { public static java.lang.String toString(byte); public static java.lang.Byte valueOf(byte); public static byte parseByte(java.lang.String); public static byte parseByte(java.lang.String,int); public static java.lang.Byte valueOf(java.lang.String,int); public static java.lang.Byte valueOf(java.lang.String); public static java.lang.Byte decode(java.lang.String); public int compareTo(java.lang.Byte); public static java.lang.String toString(short); public static short parseShort(java.lang.String); public static short parseShort(java.lang.String,int); public static java.lang.Short valueOf(java.lang.String,int); public static java.lang.Short valueOf(java.lang.String); public static java.lang.Short valueOf(short); public static java.lang.Short decode(java.lang.String); public static short reverseBytes(short); public int compareTo(java.lang.Short); public static java.lang.String toString(int,int); public static java.lang.String toHexString(int); public static java.lang.String toOctalString(int); public static java.lang.String toBinaryString(int); public static java.lang.String toString(int); public static int parseInt(java.lang.String,int); public static int parseInt(java.lang.String); public static java.lang.Integer valueOf(java.lang.String,int); public static java.lang.Integer valueOf(java.lang.String); public static java.lang.Integer valueOf(int); public static java.lang.Integer getInteger(java.lang.String); public static java.lang.Integer getInteger(java.lang.String,int); public static java.lang.Integer getInteger(java.lang.String,java.lang.Integer); public static java.lang.Integer decode(java.lang.String); public static int highestOneBit(int); public static int lowestOneBit(int); public static int numberOfLeadingZeros(int); public static int numberOfTrailingZeros(int); public static int bitCount(int); public static int rotateLeft(int,int); public static int rotateRight(int,int); public static int reverse(int); public static int signum(int); public static int reverseBytes(int); public int compareTo(java.lang.Integer); public static java.lang.String toString(long,int); public static java.lang.String toHexString(long); public static java.lang.String toOctalString(long); public static java.lang.String toBinaryString(long); public static java.lang.String toString(long); public static long parseLong(java.lang.String,int); public static long parseLong(java.lang.String); public static java.lang.Long valueOf(java.lang.String,int); public static java.lang.Long valueOf(java.lang.String); public static java.lang.Long valueOf(long); public static java.lang.Long decode(java.lang.String); public static java.lang.Long getLong(java.lang.String); public static java.lang.Long getLong(java.lang.String,long); public static java.lang.Long getLong(java.lang.String,java.lang.Long); public static long highestOneBit(long); public static long lowestOneBit(long); public static int numberOfLeadingZeros(long); public static int numberOfTrailingZeros(long); public static int bitCount(long); public static long rotateLeft(long,int); public static long rotateRight(long,int); public static long reverse(long); public static int signum(long); public static long reverseBytes(long); public int compareTo(java.lang.Long); public static java.lang.String toString(float); public static java.lang.String toHexString(float); public static java.lang.Float valueOf(java.lang.String); public static java.lang.Float valueOf(float); public static float parseFloat(java.lang.String); public static boolean isNaN(float); public static boolean isInfinite(float); public static int floatToIntBits(float); public static int floatToRawIntBits(float); public static float intBitsToFloat(int); public static int compare(float,float); public boolean isNaN(); public boolean isInfinite(); public int compareTo(java.lang.Float); public static java.lang.String toString(double); public static java.lang.String toHexString(double); public static java.lang.Double valueOf(java.lang.String); public static java.lang.Double valueOf(double); public static double parseDouble(java.lang.String); public static boolean isNaN(double); public static boolean isInfinite(double); public static long doubleToLongBits(double); public static long doubleToRawLongBits(double); public static double longBitsToDouble(long); public static int compare(double,double); public boolean isNaN(); public boolean isInfinite(); public int compareTo(java.lang.Double); public byte byteValue(); public short shortValue(); public int intValue(); public long longValue(); public float floatValue(); public double doubleValue(); public int compareTo(java.lang.Object); public boolean equals(java.lang.Object); public int hashCode(); public java.lang.String toString(); } # Remove - String method calls. Remove all invocations of String # methods without side effects whose return values are not used. -assumenosideeffects public class java.lang.String { public static java.lang.String copyValueOf(char[]); public static java.lang.String copyValueOf(char[],int,int); public static java.lang.String valueOf(boolean); public static java.lang.String valueOf(char); public static java.lang.String valueOf(char[]); public static java.lang.String valueOf(char[],int,int); public static java.lang.String valueOf(double); public static java.lang.String valueOf(float); public static java.lang.String valueOf(int); public static java.lang.String valueOf(java.lang.Object); public static java.lang.String valueOf(long); public boolean contentEquals(java.lang.StringBuffer); public boolean endsWith(java.lang.String); public boolean equalsIgnoreCase(java.lang.String); public boolean equals(java.lang.Object); public boolean matches(java.lang.String); public boolean regionMatches(boolean,int,java.lang.String,int,int); public boolean regionMatches(int,java.lang.String,int,int); public boolean startsWith(java.lang.String); public boolean startsWith(java.lang.String,int); public byte[] getBytes(); public byte[] getBytes(java.lang.String); public char charAt(int); public char[] toCharArray(); public int compareToIgnoreCase(java.lang.String); public int compareTo(java.lang.Object); public int compareTo(java.lang.String); public int hashCode(); public int indexOf(int); public int indexOf(int,int); public int indexOf(java.lang.String); public int indexOf(java.lang.String,int); public int lastIndexOf(int); public int lastIndexOf(int,int); public int lastIndexOf(java.lang.String); public int lastIndexOf(java.lang.String,int); public int length(); public java.lang.CharSequence subSequence(int,int); public java.lang.String concat(java.lang.String); public java.lang.String replaceAll(java.lang.String,java.lang.String); public java.lang.String replace(char,char); public java.lang.String replaceFirst(java.lang.String,java.lang.String); public java.lang.String[] split(java.lang.String); public java.lang.String[] split(java.lang.String,int); public java.lang.String substring(int); public java.lang.String substring(int,int); public java.lang.String toLowerCase(); public java.lang.String toLowerCase(java.util.Locale); public java.lang.String toString(); public java.lang.String toUpperCase(); public java.lang.String toUpperCase(java.util.Locale); public java.lang.String trim(); } # Remove - StringBuffer method calls. Remove all invocations of StringBuffer # methods without side effects whose return values are not used. -assumenosideeffects public class java.lang.StringBuffer { public java.lang.String toString(); public char charAt(int); public int capacity(); public int codePointAt(int); public int codePointBefore(int); public int indexOf(java.lang.String,int); public int lastIndexOf(java.lang.String); public int lastIndexOf(java.lang.String,int); public int length(); public java.lang.String substring(int); public java.lang.String substring(int,int); } # Remove - StringBuilder method calls. Remove all invocations of StringBuilder # methods without side effects whose return values are not used. -assumenosideeffects public class java.lang.StringBuilder { public java.lang.String toString(); public char charAt(int); public int capacity(); public int codePointAt(int); public int codePointBefore(int); public int indexOf(java.lang.String,int); public int lastIndexOf(java.lang.String); public int lastIndexOf(java.lang.String,int); public int length();
相關推薦程式碼混淆———Proguard使用最新,最全教程,親自試驗最近公司有一個專案,是外包專案,由於對方也有技術人員,出於技術上的保密,需要對class檔案進行二次處理,於是網上找了好久,只發現Proguard是用的最廣泛而且網上資料最多的。由於不是純JAVA專案,而是WEB專案,涉及到大量的配置檔案,所以用這個工具稍顯吃力,於是開始 Proguard使用最新,最全教程,親自試驗cti sna 選擇 關鍵字 blog [] 運行 include 命名 最近公司有一個項目,是外包項目,由於對方也有技術人員,出於技術上的保密,需要對class文件進行二次處理,於是網上找了好久,只發現Proguard是用的最廣泛而且網上資料最多的。由於不是純JAVA項目 【maven】史上最全教程,看了必懂一、為什麼使用Maven這樣的構建工具【why】 一個專案就是一個工程 如果專案非常龐大,就不適合使用package來劃分模 史上最全的資料結構視訊教程分享-絕對是史上最全的,共30個!!資料結構視訊教程下載地址 出自我是碼農 以下資料結構視訊教程是我多年收集的,因為在百度網盤上分享整個教程很快就會被delete,所以我只好花費大量功夫對單個視訊進行一個一個的分享,這樣才能長時間保留下來,為了學習,麻煩些也值得了!現在毫無保留的免費共享給大家,與君共勉! 個人認為資料結構 資料結構視訊教程-絕對是史上最全的,共30個!!史上最全的資料結構視訊教程打包下載地址 本文出自出自我是碼農,轉載請註明出處,謝謝! 以下資料結構視訊教程是我多年收集的,因為在百度網盤上分享整個教程很快就會被delete,所以我只好花費大量功夫對單個視訊進行一個一個的分享,這樣才能長時間保留下來,為了學習,麻煩些也值得了!現在毫無保留的免費共 JQData安裝 | 最貼心教程,安裝JQData全靠這篇指南轉載: https://www.joinquant.com/post/13936 首先,JQData是基於python的一個數據包,所以安裝JQData的第一步是安裝Python (沒有接觸過python或者python基礎不好的小夥伴,可以關注聚寬量化課堂的python講堂進行python 這是我見過最牛逼,最全面的Beautiful Soup 4.2 教程!沒有之一進群:125240963 即可獲取數十套PDF!Beautiful Soup 是一個可以從HTML或XML檔案中提取資料的Python庫.它能夠通過你喜歡的轉換器實現慣用的文件導航,查詢,修改文件的方式.Beautiful Soup會幫你節省數小時甚至數天的工作時間.這 Linux下設定終端解析度,最全的VGA程式碼和解析度對照表一般安裝完linux後預設的kernel給的tty解析度是非常有限的,解決的方法就是給kernel傳遞VGA引數。對應的要修改的配置檔案為 /etc/grub.conf,(或者 /boot/grub/grub.conf)就是在grub.conf啟動列表的kernel最後新增vga引數 vga=xxxx, CentOS7.4安裝php7.2+mysql8.0+nginx1.14+redis4.0以及相關擴充套件,最全最仔細教程!!!!!CentOS7.4安裝php7.0+mysql8.0+nginx1.14+redis4.0以及相關擴充套件 安裝nginx1.14 在終端執行下面兩條指令 將nginx1.14源增加到CentOS7.4中 wget http://nginx.org 第212天:15種CSS居中的方式,最全了pre block transform tran post -i 時間 ica for CSS居中是前端工程師經常要面對的問題,也是基本技能之一。今天有時間把CSS居中的方案整理了一下,目前包括水平居中,垂直居中及水平垂直居中方案共15種。如有漏掉的,還會陸續的補充進來,算 嵌入式題庫最全的,去公司面試都會出現大寫字母 地方 video 很快 不能 ear per 作用 機制 現在嵌入式發展這麽快,很多人開始踏上嵌入式學習之路,據市場統計,一般畢業或者找工作的一些人,在面試公司的時候,都會在這個地方卡殼,那就是面試題,很多人都是面試的很好,但是在做面試題的時候,手下的功夫不夠深, NGINX源碼安裝配置詳解(./configure),最全解析unzip roo without rpc服務 所有 googl 版本 並且 大文件 NGINX ./configure詳解 在"./configure"配置中,"--with"表示啟用模塊,也就是說這些模塊在編譯時不會自動構建&qu SpringCloudStream最全教程,包括配置檔案描述Spring Cloud Stream 知識整理 概念 使用方法 概念 1. 釋出/訂閱 簡單的講就是一種生產者,消費者模式。釋出者是生產,將輸出釋出到資料中心,訂閱者是消費者,訂閱自己感興趣的資料。當有資料到達資料中心時,就把資料傳送給對應的訂閱者。 2. 消費組 零基礎自學Python,最全學習路線+參考資料!今天給大家分享一位前輩整理的一個Python web學習路線。這位前輩由於有程式設計基礎,所以採用了自學Python的方式。學完後主要做後端開發。希望對你有所啟發。 整理的一個 python web 學習路線,這基本就是筆者自學後做後端的學習路線。 入門基礎 程式語言: Python thinkphp 5 自動生成模組,最簡單方式,一句程式碼直接在專案入口檔案中加上最後一句就夠了 可以不依賴自動生成檔案,直接使用預設目錄生成模組,例如: // 定義應用目錄 define('APP_PATH', __DIR__ . '/../application/'); // 載入框架引 Docker最全教程——從理論到實戰(一)容器是應用走向雲端之後必然的發展趨勢,因此筆者非常樂於和大家分享我們這段時間對容器的理解、心得和實踐。 本篇教程持續編寫了2個星期左右,只是為了大家更好地瞭解、理解和消化這個技術,能夠搭上這波車。 你可以和我們一起討論,我們希望能夠多多交流,多多分享。 如果覺得不錯,請多多點贊,你們的支援是我們前進的最 Docker最全教程——從理論到實戰(三)往期連結: https://www.cnblogs.com/codelove/p/10030439.html https://www.cnblogs.com/codelove/p/10036608.html 寫在前面 容器是應用走向雲端之後必然的發展趨勢,因此筆者非常樂 Docker最全教程——從理論到實戰(四)變量 參加 屬於 當我 web服務 隔離 方便 當前 context 往期內容鏈接 https://www.cnblogs.com/codelove/p/10030439.html https://www.cnblogs.com/codelove/p/10036608. 還不會vs快捷鍵的快進來,最全vs快捷鍵,非常常用,和我都是小的白的一定要看看ctrl±(shift+ctrl±):移動游標到上次位置或相反,比如定位一個函式,轉到函式定義後想回到函式使用處,則用ctrl±,若又想回到函式定義處則可以按shift+ctrl± F12:Go to Definition,到變數或函式定義的地方,如變數宣告處, macOS Mojave 搶先更新,最全的更新介紹都在這裡了2018年9月25日,果粉們期待已久的macOS最新版本 macOS Mojave 正式版已經可以在 App Store 直接進行更新。macOS Mojave 帶來了各項精心設計的新功能,下面就讓小馬給大家詳細介紹一下有哪些新功能吧。 深色模式:大作,大為奪目。 深 |