手记

对apk 文件进行伪加密

进行字节码变换的工具,可运行的源代码。

其中的apk的目录,需要自己修改成,你的自己的目录。

[代码]xml代码:

?


package com.ljt.apk; import java.io.File;import java.io.FileInputStream;import java.io.FileOutputStream;import java.io.IOException; public class MainClass {    public static void main(String[] args) {        //包含所有需要加密的apk的父目录        File rootFile = new File("G:\\apk\\apks");        File[] listFiles = rootFile.listFiles();                 System.out.println("--------------------------开始执行加密,和文件字节码的破坏------------------------------------");                 for(int i = 0; i < listFiles.length; i++){            System.out.println("--------------------------对文件进行加密:"+listFiles[i]+"------------------------------------");            // 对压缩文件进行伪加密,            try {                new ApkUtilTool().ChangToEncryptedEntry(listFiles[i], new File("G:\\apk\\tempApk\\"+listFiles[i].getName()));            } catch (IOException e) {                e.printStackTrace();            }        }        System.out.println("--------------------加密完成------------------------");                 //加密完成后,的临时目录        File tempFile = new File("G:\\apk\\tempApk");        File[] tempFiles = tempFile.listFiles();        for(int i = 0; i < tempFiles.length; i++){            String name = tempFiles[i].getName();            System.out.println("--------------------破坏文件:"+name+"------------------------");            name = name.substring(0,name.length()-4);            //加密完成后的目录            File outFile = new File("G:\\apk\\encryptApk\\"+name);            outFile.mkdirs();            damageFile(tempFiles[i],outFile.getAbsolutePath()+"\\Android.apk");            tempFiles[i].delete();        }        System.out.println("-----------------------------全部完成---------------------------------");                 //测试去除伪加密//      try {//          new ApkUtilTool().FixEncryptedEntry("c:/Android.apk", "c:/Android1.apk");//      } catch (IOException e) {//          e.printStackTrace();//      }    }         /**     * 破坏文件     * @param inFile     * @param outFilePath     */    public static void damageFile(File inFile,String outFilePath){        try {            FileInputStream in = new FileInputStream(inFile);            FileOutputStream out = new FileOutputStream(outFilePath);            int read = 0;            byte[] readBuffer = new byte[512];                         long fileSize = inFile.length() - 512;            long count = 0;            while((read = in.read(readBuffer)) != -1){                if(count > fileSize){                    readBuffer[read-2] = (byte) (readBuffer[read-2] ^ 0xff);                    out.write(readBuffer,0,read);                }else{                                         out.write(readBuffer,0,read);                }                count += read;            }            in.close();            out.flush();            out.close();        } catch (Exception e) {            e.printStackTrace();        }    }}

?


package com.ljt.apk; import java.io.File;import java.io.FileInputStream;import java.io.FileOutputStream;import java.io.IOException;import java.nio.ByteBuffer;import java.nio.channels.FileChannel;import java.util.Arrays;import java.util.zip.ZipError;import com.ljt.apk.ZipConstants; public class ApkUtilTool {     private FileChannel ch; // channel to the zipfile    private FileChannel fc;     /**     * 修复zip伪加密状态的Entry     *      * @param inZip     * @param storeZip     * @throws IOException     */    private void FixEncryptedEntry(File inZip, File fixZip) throws IOException {        changEntry(inZip, fixZip, true);     }     /**     * 修复zip伪加密状态的Entry     *      * @param inZip     * @param storeZip     * @throws IOException     */    public void FixEncryptedEntry(String inZip, String fixZip) throws IOException {        FixEncryptedEntry(new File(inZip), new File(fixZip));    }     /**     * 修改zip的Entry为伪加密状态     *      * @param inZip     * @param storeZip     * @throws IOException     */    public void ChangToEncryptedEntry(String inZip, String storeZip) throws IOException {        ChangToEncryptedEntry(new File(inZip), new File(storeZip));    }    /**     * 修改zip的Entry为伪加密状态     *      * @param inZip     * @param storeZip     * @throws IOException     */    public void ChangToEncryptedEntry(File inZip, File storeZip) throws IOException {        changEntry(inZip, storeZip, false);    }      /**     * 更改zip的Entry为伪加密状态     *      * @param inZip     * @param storeZip     * @param fix     *            ture:修复伪加密 false:更改到伪加密     * @throws IOException     */    private void changEntry(File inZip, File storeZip, boolean fix) throws IOException {        FileInputStream fis = new FileInputStream(inZip);        FileOutputStream fos = new FileOutputStream(storeZip);         byte[] buf = new byte[10240];        int len;        while ((len = fis.read(buf)) != -1) {            fos.write(buf, 0, len);        }         ch = fis.getChannel();        fc = fos.getChannel();         changEntry(fix);         ch.close();        fc.close();         fis.close();        fos.close();    }     // Reads zip file central directory. Returns the file position of first    // CEN header, otherwise returns -1 if an error occured. If zip->msg != NULL    // then the error was a zip format error and zip->msg has the error text.    // Always pass in -1 for knownTotal; it's used for a recursive call.    private void changEntry(boolean fix) throws IOException {        END end = findEND();         if (end.cenlen > end.endpos)            zerror("invalid END header (bad central directory size)");        long cenpos = end.endpos - end.cenlen; // position of CEN table         // Get position of first local file (LOC) header, taking into        // account that there may be a stub prefixed to the zip file.        long locpos = cenpos - end.cenoff;        if (locpos < 0)            zerror("invalid END header (bad central directory offset)");         // read in the CEN and END        byte[] cen = new byte[(int) (end.cenlen + ZipConstants.ENDHDR)];        if (readFullyAt(cen, 0, cen.length, cenpos) != end.cenlen + ZipConstants.ENDHDR) {            zerror("read CEN tables failed");        }         int pos = 0;        int limit = cen.length - ZipConstants.ENDHDR;        while (pos < limit) {            if (ZipConstants.CENSIG(cen, pos) != ZipConstants.CENSIG)                zerror("invalid CEN header (bad signature)");            int method = ZipConstants.CENHOW(cen, pos);            int nlen = ZipConstants.CENNAM(cen, pos);            int elen = ZipConstants.CENEXT(cen, pos);            int clen = ZipConstants.CENCOM(cen, pos);             if (fix) {                if ((ZipConstants.CENFLG(cen, pos) & 1) != 0) {                    byte[] name = Arrays.copyOfRange(cen, pos + ZipConstants.CENHDR, pos + ZipConstants.CENHDR + nlen);                    System.out.println("Found the encrypted entry : " + new String(name) + ", fix...");                    // b[n] & 0xff) | ((b[n + 1] & 0xff) << 8                    cen[pos + 8] &= 0xFE;                    // cen[pos+8] ^= CEN***(cen, pos) % 2;                    // cen[pos+8] ^= cen[pos+8] % 2;                    // zerror("invalid CEN header (encrypted entry)");                }            } else {                if ((ZipConstants.CENFLG(cen, pos) & 1) == 0) {                    byte[] name = Arrays.copyOfRange(cen, pos + ZipConstants.CENHDR, pos + ZipConstants.CENHDR + nlen);                    System.out.println("Chang the entry : " + new String(name) + ", Encrypted...");                    // b[n] & 0xff) | ((b[n + 1] & 0xff) << 8                    cen[pos + 8] |= 0x1;                    // zerror("invalid CEN header (encrypted entry)");                }            }             if (method != ZipConstants.METHOD_STORED && method != ZipConstants.METHOD_DEFLATED)                zerror("invalid CEN header (unsupported compression method: " + method + ")");            if (pos + ZipConstants.CENHDR + nlen > limit)                zerror("invalid CEN header (bad header size)");             // skip ext and comment            pos += (ZipConstants.CENHDR + nlen + elen + clen);        }         writeFullyAt(cen, 0, cen.length, cenpos);         if (pos + ZipConstants.ENDHDR != cen.length) {            zerror("invalid CEN header (bad header size)");        }    }     // Reads len bytes of data from the specified offset into buf.    // Returns the total number of bytes read.    // Each/every byte read from here (except the cen, which is mapped).    final long readFullyAt(byte[] buf, int off, long len, long pos) throws IOException {        ByteBuffer bb = ByteBuffer.wrap(buf);        bb.position(off);        bb.limit((int) (off + len));        return readFullyAt(bb, pos);    }     private final long readFullyAt(ByteBuffer bb, long pos) throws IOException {        synchronized (ch) {            return ch.position(pos).read(bb);        }    }     final long writeFullyAt(byte[] buf, int off, long len, long pos) throws IOException {        ByteBuffer bb = ByteBuffer.wrap(buf);        bb.position(off);        bb.limit((int) (off + len));        return writeFullyAt(bb, pos);    }     private final long writeFullyAt(ByteBuffer bb, long pos) throws IOException {        synchronized (fc) {            return fc.position(pos).write(bb);        }    }     // Searches for end of central directory (END) header. The contents of    // the END header will be read and placed in endbuf. Returns the file    // position of the END header, otherwise returns -1 if the END header    // was not found or an error occurred.    private END findEND() throws IOException {        byte[] buf = new byte[ZipConstants.READBLOCKSZ];        long ziplen = ch.size();        long minHDR = (ziplen - ZipConstants.END_MAXLEN) > 0 ? ziplen - ZipConstants.END_MAXLEN : 0;        long minPos = minHDR - (buf.length - ZipConstants.ENDHDR);         for (long pos = ziplen - buf.length; pos >= minPos; pos -= (buf.length - ZipConstants.ENDHDR)) {            int off = 0;            if (pos < 0) {                // Pretend there are some NUL bytes before start of file                off = (int) -pos;                Arrays.fill(buf, 0, off, (byte) 0);            }            int len = buf.length - off;            if (readFullyAt(buf, off, len, pos + off) != len)                zerror("zip END header not found");             // Now scan the block backwards for END header signature            for (int i = buf.length - ZipConstants.ENDHDR; i >= 0; i--) {                if (buf[i + 0] == (byte) 'P' && buf[i + 1] == (byte) 'K' && buf[i + 2] == (byte) '\005' && buf[i + 3] == (byte) '\006' && (pos + i + ZipConstants.ENDHDR + ZipConstants.ENDCOM(buf, i) == ziplen)) {                    // Found END header                    buf = Arrays.copyOfRange(buf, i, i + ZipConstants.ENDHDR);                    END end = new END();                    end.endsub = ZipConstants.ENDSUB(buf);                    end.centot = ZipConstants.ENDTOT(buf);                    end.cenlen = ZipConstants.ENDSIZ(buf);                    end.cenoff = ZipConstants.ENDOFF(buf);                    end.comlen = ZipConstants.ENDCOM(buf);                    end.endpos = pos + i;                    if (end.cenlen == ZipConstants.ZIP64_MINVAL || end.cenoff == ZipConstants.ZIP64_MINVAL || end.centot == ZipConstants.ZIP64_MINVAL32) {                        // need to find the zip64 end;                        byte[] loc64 = new byte[ZipConstants.ZIP64_LOCHDR];                        if (readFullyAt(loc64, 0, loc64.length, end.endpos - ZipConstants.ZIP64_LOCHDR) != loc64.length) {                            return end;                        }                        long end64pos = ZipConstants.ZIP64_LOCOFF(loc64);                        byte[] end64buf = new byte[ZipConstants.ZIP64_ENDHDR];                        if (readFullyAt(end64buf, 0, end64buf.length, end64pos) != end64buf.length) {                            return end;                        }                        // end64 found, re-calcualte everything.                        end.cenlen = ZipConstants.ZIP64_ENDSIZ(end64buf);                        end.cenoff = ZipConstants.ZIP64_ENDOFF(end64buf);                        end.centot = (int) ZipConstants.ZIP64_ENDTOT(end64buf); // assume                                                                                // total                                                                                // <                                                                                // 2g                        end.endpos = end64pos;                    }                    return end;                }            }        }        zerror("zip END header not found");        return null; // make compiler happy    }     static void zerror(String msg) {        throw new ZipError(msg);    }     // End of central directory record    static class END {        int disknum;        int sdisknum;        int endsub; // endsub        int centot; // 4 bytes        long cenlen; // 4 bytes        long cenoff; // 4 bytes        int comlen; // comment length        byte[] comment;         /* members of Zip64 end of central directory locator */        int diskNum;        long endpos;        int disktot;         @Override        public String toString() {            return "disknum : " + disknum + "\n" + "sdisknum : " + sdisknum + "\n" + "endsub : " + endsub + "\n" + "centot : " + centot + "\n" + "cenlen : " + cenlen + "\n" + "cenoff : " + cenoff + "\n" + "comlen : " + comlen + "\n" + "diskNum : " + diskNum + "\n" + "endpos : " + endpos + "\n" + "disktot : " + disktot;        }    }}

?


package com.ljt.apk; class ZipConstants {    /*     * Compression methods     */    static final int METHOD_STORED     = 0;    static final int METHOD_DEFLATED   = 8;    static final int METHOD_DEFLATED64 = 9;    static final int METHOD_BZIP2      = 12;    static final int METHOD_LZMA       = 14;    static final int METHOD_LZ77       = 19;    static final int METHOD_AES        = 99;     /*     * General purpose big flag     */    static final int FLAG_ENCRYPTED  = 0x01;    static final int FLAG_DATADESCR  = 0x08;    // crc, size and csize in dd    static final int FLAG_EFS        = 0x800;   // If this bit is set the filename and                                                // comment fields for this file must be                                                // encoded using UTF-8.    /*     * Header signatures     */    static long LOCSIG = 0x04034b50L;   // "PK\003\004"    static long EXTSIG = 0x08074b50L;   // "PK\007\008"    static long CENSIG = 0x02014b50L;   // "PK\001\002"    static long ENDSIG = 0x06054b50L;   // "PK\005\006"     /*     * Header sizes in bytes (including signatures)     */    static final int LOCHDR = 30;       // LOC header size    static final int EXTHDR = 16;       // EXT header size    static final int CENHDR = 46;       // CEN header size    static final int ENDHDR = 22;       // END header size     /*     * Local file (LOC) header field offsets     */    static final int LOCVER = 4;        // version needed to extract    static final int LOCFLG = 6;        // general purpose bit flag    static final int LOCHOW = 8;        // compression method    static final int LOCTIM = 10;       // modification time    static final int LOCCRC = 14;       // uncompressed file crc-32 value    static final int LOCSIZ = 18;       // compressed size    static final int LOCLEN = 22;       // uncompressed size    static final int LOCNAM = 26;       // filename length    static final int LOCEXT = 28;       // extra field length     /*     * Extra local (EXT) header field offsets     */    static final int EXTCRC = 4;        // uncompressed file crc-32 value    static final int EXTSIZ = 8;        // compressed size    static final int EXTLEN = 12;       // uncompressed size     /*     * Central directory (CEN) header field offsets     */    static final int CENVEM = 4;        // version made by    static final int CENVER = 6;        // version needed to extract    static final int CENFLG = 8;        // encrypt, decrypt flags    static final int CENHOW = 10;       // compression method    static final int CENTIM = 12;       // modification time    static final int CENCRC = 16;       // uncompressed file crc-32 value    static final int CENSIZ = 20;       // compressed size    static final int CENLEN = 24;       // uncompressed size    static final int CENNAM = 28;       // filename length    static final int CENEXT = 30;       // extra field length    static final int CENCOM = 32;       // comment length    static final int CENDSK = 34;       // disk number start    static final int CENATT = 36;       // internal file attributes    static final int CENATX = 38;       // external file attributes    static final int CENOFF = 42;       // LOC header offset     /*     * End of central directory (END) header field offsets     */    static final int ENDSUB = 8;        // number of entries on this disk    static final int ENDTOT = 10;       // total number of entries    static final int ENDSIZ = 12;       // central directory size in bytes    static final int ENDOFF = 16;       // offset of first CEN header    static final int ENDCOM = 20;       // zip file comment length     /*     * ZIP64 constants     */    static final long ZIP64_ENDSIG = 0x06064b50L;  // "PK\006\006"    static final long ZIP64_LOCSIG = 0x07064b50L;  // "PK\006\007"    static final int  ZIP64_ENDHDR = 56;           // ZIP64 end header size    static final int  ZIP64_LOCHDR = 20;           // ZIP64 end loc header size    static final int  ZIP64_EXTHDR = 24;           // EXT header size    static final int  ZIP64_EXTID  = 0x0001;       // Extra field Zip64 header ID     static final int  ZIP64_MINVAL32 = 0xFFFF;    static final long ZIP64_MINVAL = 0xFFFFFFFFL;     /*     * Zip64 End of central directory (END) header field offsets     */    static final int  ZIP64_ENDLEN = 4;       // size of zip64 end of central dir    static final int  ZIP64_ENDVEM = 12;      // version made by    static final int  ZIP64_ENDVER = 14;      // version needed to extract    static final int  ZIP64_ENDNMD = 16;      // number of this disk    static final int  ZIP64_ENDDSK = 20;      // disk number of start    static final int  ZIP64_ENDTOD = 24;      // total number of entries on this disk    static final int  ZIP64_ENDTOT = 32;      // total number of entries    static final int  ZIP64_ENDSIZ = 40;      // central directory size in bytes    static final int  ZIP64_ENDOFF = 48;      // offset of first CEN header    static final int  ZIP64_ENDEXT = 56;      // zip64 extensible data sector     /*     * Zip64 End of central directory locator field offsets     */    static final int  ZIP64_LOCDSK = 4;       // disk number start    static final int  ZIP64_LOCOFF = 8;       // offset of zip64 end    static final int  ZIP64_LOCTOT = 16;      // total number of disks     /*     * Zip64 Extra local (EXT) header field offsets     */    static final int  ZIP64_EXTCRC = 4;       // uncompressed file crc-32 value    static final int  ZIP64_EXTSIZ = 8;       // compressed size, 8-byte    static final int  ZIP64_EXTLEN = 16;      // uncompressed size, 8-byte     /*     * Extra field header ID     */    static final int  EXTID_ZIP64 = 0x0001;      // ZIP64    static final int  EXTID_NTFS  = 0x000a;      // NTFS    static final int  EXTID_UNIX  = 0x000d;      // UNIX    static final int  EXTID_EFS   = 0x0017;      // Strong Encryption    static final int  EXTID_EXTT  = 0x5455;      // Info-ZIP Extended Timestamp     /*     * fields access methods     */    ///////////////////////////////////////////////////////    static final int CH(byte[] b, int n) {       return b[n] & 0xff;    }     static final int SH(byte[] b, int n) {        return (b[n] & 0xff) | ((b[n + 1] & 0xff) << 8);    }     static final long LG(byte[] b, int n) {        return ((SH(b, n)) | (SH(b, n + 2) << 16)) & 0xffffffffL;    }     static final long LL(byte[] b, int n) {        return (LG(b, n)) | (LG(b, n + 4) << 32);    }     static final long GETSIG(byte[] b) {        return LG(b, 0);    }     // local file (LOC) header fields    static final long LOCSIG(byte[] b) { return LG(b, 0); } // signature    static final int  LOCVER(byte[] b) { return SH(b, 4); } // version needed to extract    static final int  LOCFLG(byte[] b) { return SH(b, 6); } // general purpose bit flags    static final int  LOCHOW(byte[] b) { return SH(b, 8); } // compression method    static final long LOCTIM(byte[] b) { return LG(b, 10);} // modification time    static final long LOCCRC(byte[] b) { return LG(b, 14);} // crc of uncompressed data    static final long LOCSIZ(byte[] b) { return LG(b, 18);} // compressed data size    static final long LOCLEN(byte[] b) { return LG(b, 22);} // uncompressed data size    static final int  LOCNAM(byte[] b) { return SH(b, 26);} // filename length    static final int  LOCEXT(byte[] b) { return SH(b, 28);} // extra field length     // extra local (EXT) header fields    static final long EXTCRC(byte[] b) { return LG(b, 4);}  // crc of uncompressed data    static final long EXTSIZ(byte[] b) { return LG(b, 8);}  // compressed size    static final long EXTLEN(byte[] b) { return LG(b, 12);} // uncompressed size     // end of central directory header (END) fields    static final int  ENDSUB(byte[] b) { return SH(b, 8); }  // number of entries on this disk    static final int  ENDTOT(byte[] b) { return SH(b, 10);}  // total number of entries    static final long ENDSIZ(byte[] b) { return LG(b, 12);}  // central directory size    static final long ENDOFF(byte[] b) { return LG(b, 16);}  // central directory offset    static final int  ENDCOM(byte[] b) { return SH(b, 20);}  // size of zip file comment    static final int  ENDCOM(byte[] b, int off) { return SH(b, off + 20);}     // zip64 end of central directory recoder fields    static final long ZIP64_ENDTOD(byte[] b) { return LL(b, 24);}  // total number of entries on disk    static final long ZIP64_ENDTOT(byte[] b) { return LL(b, 32);}  // total number of entries    static final long ZIP64_ENDSIZ(byte[] b) { return LL(b, 40);}  // central directory size    static final long ZIP64_ENDOFF(byte[] b) { return LL(b, 48);}  // central directory offset    static final long ZIP64_LOCOFF(byte[] b) { return LL(b, 8);}   // zip64 end offset     // central directory header (CEN) fields    static final long CENSIG(byte[] b, int pos) { return LG(b, pos + 0); }    static final int  CENVEM(byte[] b, int pos) { return SH(b, pos + 4); }    static final int  CENVER(byte[] b, int pos) { return SH(b, pos + 6); }    static final int  CENFLG(byte[] b, int pos) { return SH(b, pos + 8); }    static final int  CENHOW(byte[] b, int pos) { return SH(b, pos + 10);}    static final long CENTIM(byte[] b, int pos) { return LG(b, pos + 12);}    static final long CENCRC(byte[] b, int pos) { return LG(b, pos + 16);}    static final long CENSIZ(byte[] b, int pos) { return LG(b, pos + 20);}    static final long CENLEN(byte[] b, int pos) { return LG(b, pos + 24);}    static final int  CENNAM(byte[] b, int pos) { return SH(b, pos + 28);}    static final int  CENEXT(byte[] b, int pos) { return SH(b, pos + 30);}    static final int  CENCOM(byte[] b, int pos) { return SH(b, pos + 32);}    static final int  CENDSK(byte[] b, int pos) { return SH(b, pos + 34);}    static final int  CENATT(byte[] b, int pos) { return SH(b, pos + 36);}    static final long CENATX(byte[] b, int pos) { return LG(b, pos + 38);}    static final long CENOFF(byte[] b, int pos) { return LG(b, pos + 42);}     /* The END header is followed by a variable length comment of size < 64k. */    static final long END_MAXLEN = 0xFFFF + ENDHDR;    static final int READBLOCKSZ = 128;}

对apk进行多一层的保护,使代码更安全可靠,很难被更改。

原文链接:http://www.apkbus.com/blog-708270-63607.html

0人推荐
随时随地看视频
慕课网APP

相关阅读

学习面试题Day07