X Tutup
import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.util.Arrays; import java.nio.charset.Charset; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; import java.util.concurrent.locks.ReentrantLock; public class IPExt { public static void main(String[] args) { IPExt.load("H:\\loveapp\\codebase\\17mon\\17monipdb.datx"); System.out.println(Arrays.toString(IPExt.find("8.8.8.8"))); System.out.println(Arrays.toString(IPExt.find("118.28.8.8"))); System.out.println(Arrays.toString(IPExt.find("255.255.255.255"))); } public static boolean enableFileWatch = false; private static int offset; private static int[] index = new int[65536]; private static ByteBuffer dataBuffer; private static ByteBuffer indexBuffer; private static Long lastModifyTime = 0L; private static File ipFile ; private static ReentrantLock lock = new ReentrantLock(); public static void load(String filename) { ipFile = new File(filename); load(); if (enableFileWatch) { watch(); } } public static void load(String filename, boolean strict) throws Exception { ipFile = new File(filename); if (strict) { int contentLength = Long.valueOf(ipFile.length()).intValue(); if (contentLength < 512 * 1024) { throw new Exception("ip data file error."); } } load(); if (enableFileWatch) { watch(); } } public static String[] find(String ip) { String[] ips = ip.split("\\."); int prefix_value = (Integer.valueOf(ips[0]) * 256 + Integer.valueOf(ips[1])); long ip2long_value = ip2long(ip); int start = index[prefix_value]; int max_comp_len = offset - 262144 - 4; long tmpInt; long index_offset = -1; int index_length = -1; byte b = 0; for (start = start * 9 + 262144; start < max_comp_len; start += 9) { tmpInt = int2long(indexBuffer.getInt(start)); if (tmpInt >= ip2long_value) { index_offset = bytesToLong(b, indexBuffer.get(start + 6), indexBuffer.get(start + 5), indexBuffer.get(start + 4)); index_length = ((0xFF & indexBuffer.get(start + 7)) << 8) + (0xFF & indexBuffer.get(start + 8)); break; } } byte[] areaBytes; lock.lock(); try { dataBuffer.position(offset + (int) index_offset - 262144); areaBytes = new byte[index_length]; dataBuffer.get(areaBytes, 0, index_length); } finally { lock.unlock(); } return new String(areaBytes, Charset.forName("UTF-8")).split("\t", -1); } private static void watch() { Executors.newScheduledThreadPool(1).scheduleAtFixedRate(new Runnable() { @Override public void run() { long time = ipFile.lastModified(); if (time > lastModifyTime) { load(); } } }, 1000L, 5000L, TimeUnit.MILLISECONDS); } private static void load() { lastModifyTime = ipFile.lastModified(); lock.lock(); try { dataBuffer = ByteBuffer.wrap(getBytesByFile(ipFile)); dataBuffer.position(0); offset = dataBuffer.getInt(); // indexLength byte[] indexBytes = new byte[offset]; dataBuffer.get(indexBytes, 0, offset - 4); indexBuffer = ByteBuffer.wrap(indexBytes); indexBuffer.order(ByteOrder.LITTLE_ENDIAN); for (int i = 0; i < 256; i++) { for (int j = 0; j < 256; j++) { index[i * 256 + j] = indexBuffer.getInt(); } } indexBuffer.order(ByteOrder.BIG_ENDIAN); } finally { lock.unlock(); } } private static byte[] getBytesByFile(File file) { FileInputStream fin = null; byte[] bs = new byte[new Long(file.length()).intValue()]; try { fin = new FileInputStream(file); int readBytesLength = 0; int i; while ((i = fin.available()) > 0) { fin.read(bs, readBytesLength, i); readBytesLength += i; } } catch (IOException ioe) { ioe.printStackTrace(); } finally { try { if (fin != null) { fin.close(); } } catch (IOException e){ e.printStackTrace(); } } return bs; } private static long bytesToLong(byte a, byte b, byte c, byte d) { return int2long((((a & 0xff) << 24) | ((b & 0xff) << 16) | ((c & 0xff) << 8) | (d & 0xff))); } private static int str2Ip(String ip) { String[] ss = ip.split("\\."); int a, b, c, d; a = Integer.parseInt(ss[0]); b = Integer.parseInt(ss[1]); c = Integer.parseInt(ss[2]); d = Integer.parseInt(ss[3]); return (a << 24) | (b << 16) | (c << 8) | d; } private static long ip2long(String ip) { return int2long(str2Ip(ip)); } private static long int2long(int i) { long l = i & 0x7fffffffL; if (i < 0) { l |= 0x080000000L; } return l; } }
X Tutup