Java を使用してファイルの MD5 チェックサムを取得しようとしています。本当に驚きましたが、ファイルの MD5 チェックサムを取得する方法を示すものを見つけることができませんでした。
それはどのように行われますか?
入力ストリーム デコレータ がjava.security.DigestInputStream
あるので、通常どおりに入力ストリームを使用しながらダイジェストを計算できます。データを余分に渡す必要はありません。
MessageDigest md = MessageDigest.getInstance("MD5");
try (InputStream is = Files.newInputStream(Paths.get("file.txt"));
DigestInputStream dis = new DigestInputStream(is, md))
{
/* Read decorated stream (dis) to EOF as normal... */
}
byte[] digest = md.digest();
Apache Commons CodecライブラリのDigestUtilsを使用します。
try (InputStream is = Files.newInputStream(Paths.get("file.zip"))) {
String md5 = org.apache.commons.codec.digest.DigestUtils.md5Hex(is);
}
Real の Java-How-toにMessageDigestクラスを使用した例があります。
CRC32 と SHA-1 を使用した例については、そのページを確認してください。
import java.io.*;
import java.security.MessageDigest;
public class MD5Checksum {
public static byte[] createChecksum(String filename) throws Exception {
InputStream fis = new FileInputStream(filename);
byte[] buffer = new byte[1024];
MessageDigest complete = MessageDigest.getInstance("MD5");
int numRead;
do {
numRead = fis.read(buffer);
if (numRead > 0) {
complete.update(buffer, 0, numRead);
}
} while (numRead != -1);
fis.close();
return complete.digest();
}
// see this How-to for a faster way to convert
// a byte array to a HEX string
public static String getMD5Checksum(String filename) throws Exception {
byte[] b = createChecksum(filename);
String result = "";
for (int i=0; i < b.length; i++) {
result += Integer.toString( ( b[i] & 0xff ) + 0x100, 16).substring( 1 );
}
return result;
}
public static void main(String args[]) {
try {
System.out.println(getMD5Checksum("apache-tomcat-5.5.17.exe"));
// output :
// 0bb2827c5eacf570b6064e24e0e6653b
// ref :
// http://www.apache.org/dist/
// tomcat/tomcat-5/v5.5.17/bin
// /apache-tomcat-5.5.17.exe.MD5
// 0bb2827c5eacf570b6064e24e0e6653b *apache-tomcat-5.5.17.exe
}
catch (Exception e) {
e.printStackTrace();
}
}
}
com.google.common.hash API は以下を提供します。
ユーザー ガイド ( IO の説明、ハッシュの説明) をお読みください。
ユースケースFiles.hash()
では、ファイルのダイジェスト値を計算して返します。
たとえば、sha-1ダイジェスト計算 (SHA-1 を MD5 に変更して MD5 ダイジェストを取得)
HashCode hc = Files.asByteSource(file).hash(Hashing.sha1());
"SHA-1: " + hc.toString();
CRC32はMD5よりもはるかに高速であるため、暗号化された安全なチェックサムが必要ない場合はCRC32を使用してください。また、パスワードを使用して、 BCRYPT、 SCRYPT 、またはSHA-256を代わりに使用するために、 MD5をパスワードなどの保存に使用しないでください。
ハッシュによる長期的な保護のために、 マークル署名スキームがセキュリティを強化し、欧州委員会が後援するポスト量子暗号研究グループは、量子コンピューターに対する長期的な保護のためにこの暗号化の使用を推奨しています (参照)。
crc32は他のものよりも衝突率が高いことに注意してください。
nio2 (Java 7+) を使用し、外部ライブラリを使用しない場合:
byte[] b = Files.readAllBytes(Paths.get("/path/to/file"));
byte[] hash = MessageDigest.getInstance("MD5").digest(b);
結果を予想されるチェックサムと比較するには:
String expected = "2252290BC44BEAD16AA1BF89948472E8";
String actual = DatatypeConverter.printHexBinary(hash);
System.out.println(expected.equalsIgnoreCase(actual) ? "MATCH" : "NO MATCH");
Guavaは、JDK で提供されるさまざまなハッシュ API よりもはるかにユーザーフレンドリーな、新しい一貫したハッシュ API を提供するようになりました。ハッシュの説明を参照してください。ファイルの場合、MD5 サム、CRC32 (バージョン 14.0 以降)、またはその他の多くのハッシュを簡単に取得できます。
HashCode md5 = Files.hash(file, Hashing.md5());
byte[] md5Bytes = md5.asBytes();
String md5Hex = md5.toString();
HashCode crc32 = Files.hash(file, Hashing.crc32());
int crc32Int = crc32.asInt();
// the Checksum API returns a long, but it's padded with 0s for 32-bit CRC
// this is the value you would get if using that API directly
long checksumResult = crc32.padToLong();
Ok。追加する必要がありました。すでにSpringとApacheCommonsに依存している、またはそれを追加することを計画している人のための1行の実装:
DigestUtils.md5DigestAsHex(FileUtils.readFileToByteArray(file))
およびApacheコモンズのみのオプション(クレジット@duleshi):
DigestUtils.md5Hex(FileUtils.readFileToByteArray(file))
これが誰かを助けることを願っています。
Java 7 を使用したサードパーティ ライブラリを使用しないシンプルなアプローチ
String path = "your complete file path";
MessageDigest md = MessageDigest.getInstance("MD5");
md.update(Files.readAllBytes(Paths.get(path)));
byte[] digest = md.digest();
このバイト配列を印刷する必要がある場合。以下のように使用
System.out.println(Arrays.toString(digest));
このダイジェストから 16 進文字列が必要な場合。以下のように使用
String digestInHex = DatatypeConverter.printHexBinary(digest).toUpperCase();
System.out.println(digestInHex);
DatatypeConverter は javax.xml.bind.DatatypeConverter です。
私は最近、動的な文字列に対してこれを行う必要がありMessageDigest
、さまざまな方法でハッシュを表すことができます。md5sumコマンドで取得するようにファイルの署名を取得するには、次のような操作を行う必要がありました。
try {
String s = "TEST STRING";
MessageDigest md5 = MessageDigest.getInstance("MD5");
md5.update(s.getBytes(),0,s.length());
String signature = new BigInteger(1,md5.digest()).toString(16);
System.out.println("Signature: "+signature);
} catch (final NoSuchAlgorithmException e) {
e.printStackTrace();
}
これは明らかにファイルに対して具体的にそれを行う方法についてのあなたの質問に答えません、上記の答えはそれをうまく扱います。ほとんどのアプリケーションが表示するように合計を取得するのに多くの時間を費やしましたが、同じ問題が発生する可能性があると思いました。
public static void main(String[] args) throws Exception {
MessageDigest md = MessageDigest.getInstance("MD5");
FileInputStream fis = new FileInputStream("c:\\apache\\cxf.jar");
byte[] dataBytes = new byte[1024];
int nread = 0;
while ((nread = fis.read(dataBytes)) != -1) {
md.update(dataBytes, 0, nread);
};
byte[] mdbytes = md.digest();
StringBuffer sb = new StringBuffer();
for (int i = 0; i < mdbytes.length; i++) {
sb.append(Integer.toString((mdbytes[i] & 0xff) + 0x100, 16).substring(1));
}
System.out.println("Digest(in hex format):: " + sb.toString());
}
または、詳細情報を 入手できます http://www.asjava.com/core-java/java-md5-example/
以前の投稿で上記のコードに似たコードを使用していました
...
String signature = new BigInteger(1,md5.digest()).toString(16);
...
ただし、BigInteger.toString()
先頭のゼロが切り捨てられるため、ここでの使用には注意してください... (たとえば、 try s = "27"
、チェックサムは である必要があります"02e74f10e0327ad868d138f2b4fdd6f0"
)
私は Apache Commons Codec を使用するという提案に賛成し、独自のコードをそれに置き換えました。
public static String MD5Hash(String toHash) throws RuntimeException {
try{
return String.format("%032x", // produces lower case 32 char wide hexa left-padded with 0
new BigInteger(1, // handles large POSITIVE numbers
MessageDigest.getInstance("MD5").digest(toHash.getBytes())));
}
catch (NoSuchAlgorithmException e) {
// do whatever seems relevant
}
}
以下は、File をパラメーターとして受け取るように Sunil のコードをラップする単純な関数です。この関数には外部ライブラリは必要ありませんが、Java 7 が必要です。
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import javax.xml.bind.DatatypeConverter;
public class Checksum {
/**
* Generates an MD5 checksum as a String.
* @param file The file that is being checksummed.
* @return Hex string of the checksum value.
* @throws NoSuchAlgorithmException
* @throws IOException
*/
public static String generate(File file) throws NoSuchAlgorithmException,IOException {
MessageDigest messageDigest = MessageDigest.getInstance("MD5");
messageDigest.update(Files.readAllBytes(file.toPath()));
byte[] hash = messageDigest.digest();
return DatatypeConverter.printHexBinary(hash).toUpperCase();
}
public static void main(String argv[]) throws NoSuchAlgorithmException, IOException {
File file = new File("/Users/foo.bar/Documents/file.jar");
String hex = Checksum.generate(file);
System.out.printf("hex=%s\n", hex);
}
}
出力例:
hex=B117DD0C3CBBD009AC4EF65B6D75C97B
public String checksum(File file) {
try {
InputStream fin = new FileInputStream(file);
java.security.MessageDigest md5er =
MessageDigest.getInstance("MD5");
byte[] buffer = new byte[1024];
int read;
do {
read = fin.read(buffer);
if (read > 0)
md5er.update(buffer, 0, read);
} while (read != -1);
fin.close();
byte[] digest = md5er.digest();
if (digest == null)
return null;
String strDigest = "0x";
for (int i = 0; i < digest.length; i++) {
strDigest += Integer.toString((digest[i] & 0xff)
+ 0x100, 16).substring(1).toUpperCase();
}
return strDigest;
} catch (Exception e) {
return null;
}
}
結果は Linux の md5sum ユーティリティと同じです。
別の実装: Java での高速 MD5 実装
String hash = MD5.asHex(MD5.getHash(new File(filename)));
ANTを使用してビルドしている場合、これは非常に簡単です。build.xmlに以下を追加します。
<checksum file="${jarFile}" todir="${toDir}"/>
ここで、jarFileはMD5を生成するJARであり、toDirはMD5ファイルを配置するディレクトリです。
Google guava は新しい API を提供します。以下のものを見つけてください:
public static HashCode hash(File file,
HashFunction hashFunction)
throws IOException
Computes the hash code of the file using hashFunction.
Parameters:
file - the file to read
hashFunction - the hash function to use to hash the data
Returns:
the HashCode of all of the bytes in the file
Throws:
IOException - if an I/O error occurs
Since:
12.0
public static String getMd5OfFile(String filePath)
{
String returnVal = "";
try
{
InputStream input = new FileInputStream(filePath);
byte[] buffer = new byte[1024];
MessageDigest md5Hash = MessageDigest.getInstance("MD5");
int numRead = 0;
while (numRead != -1)
{
numRead = input.read(buffer);
if (numRead > 0)
{
md5Hash.update(buffer, 0, numRead);
}
}
input.close();
byte [] md5Bytes = md5Hash.digest();
for (int i=0; i < md5Bytes.length; i++)
{
returnVal += Integer.toString( ( md5Bytes[i] & 0xff ) + 0x100, 16).substring( 1 );
}
}
catch(Throwable t) {t.printStackTrace();}
return returnVal.toUpperCase();
}