Java - H.264视频编码

5

我正在尝试制作一个能够编解码h.264视频的程序,以便我可以编辑这个视频。如果我想用Java制作这个程序,有人能告诉我该怎么做吗?


最终FFMpeg可能会做到这一点,但我不是很确定。如果可以的话,肯定有一个Java封装器可用。 - evotopid
尝试使用Xuggler API。 - JuanZe
现在你提到了它,这似乎是一个比尝试使用JMF更好的想法。+1 - Andrew Thompson
Xuggler自2012年以来已经停止更新,被humble-video所取代。 - Matthieu
4个回答

11
您可以使用 JCodec (http://jcodec.org)。要解码视频序列,请执行以下操作:
int frameNumber = 10000;
FileChannelWrapper ch = null;
try {
    ch = NIOUtils.readableFileChannel(new File(path to mp4));
    FrameGrab frameGrab = new FrameGrab(ch);

    frameGrab.seek(frameNumber);
    Picture frame;
    for (int i = 0; (frame = frameGrab.getNativeFrame()) != null && i < 200; i++) {
        // Do something
    }
} finally {
    NIOUtils.closeQuietly(ch);
}

编码一个序列:

public class SequenceEncoder {
    private SeekableByteChannel ch;
    private Picture toEncode;
    private RgbToYuv420 transform;
    private H264Encoder encoder;
    private ArrayList<ByteBuffer> spsList;
    private ArrayList<ByteBuffer> ppsList;
    private CompressedTrack outTrack;
    private ByteBuffer _out;
    private int frameNo;
    private MP4Muxer muxer;

    public SequenceEncoder(File out) throws IOException {
        this.ch = NIOUtils.writableFileChannel(out);

        // Transform to convert between RGB and YUV
        transform = new RgbToYuv420(0, 0);

        // Muxer that will store the encoded frames
        muxer = new MP4Muxer(ch, Brand.MP4);

        // Add video track to muxer
        outTrack = muxer.addTrackForCompressed(TrackType.VIDEO, 25);

        // Allocate a buffer big enough to hold output frames
        _out = ByteBuffer.allocate(1920 * 1080 * 6);

        // Create an instance of encoder
        encoder = new H264Encoder();

        // Encoder extra data ( SPS, PPS ) to be stored in a special place of
        // MP4
        spsList = new ArrayList<ByteBuffer>();
        ppsList = new ArrayList<ByteBuffer>();

    }

    public void encodeImage(Picture bi) throws IOException {
        if (toEncode == null) {
            toEncode = Picture.create(bi.getWidth(), bi.getHeight(), ColorSpace.YUV420);
        }

        // Perform conversion ( RGB -> YUV )
        transform.transform(bi, toEncode);

        // Encode image into H.264 frame, the result is stored in '_out' buffer
        _out.clear();
        ByteBuffer result = encoder.encodeFrame(_out, toEncode);

        // Based on the frame above form correct MP4 packet
        spsList.clear();
        ppsList.clear();
        H264Utils.encodeMOVPacket(result, spsList, ppsList);

        // Add packet to video track
        outTrack.addFrame(new MP4Packet(result, frameNo, 25, 1, frameNo, true, null, frameNo, 0));

        frameNo++;
    }

    public void finish() throws IOException {
        // Push saved SPS/PPS to a special storage in MP4
        outTrack.addSampleEntry(H264Utils.createMOVSampleEntry(spsList, ppsList));

        // Write MP4 header and finalize recording
        muxer.writeHeader();
        NIOUtils.closeQuietly(ch);
    }
}

最后,要转换 Android 镜像的 TOFROM,请使用以下命令:

public static Picture fromBitmap(Bitmap src) {
    Picture dst = Picture.create((int)src.getWidth(), (int)src.getHeight(), RGB);
    fromBitmap(src, dst);
    return dst;
}

public static void fromBitmap(Bitmap src, Picture dst) {
    int[] dstData = dst.getPlaneData(0);
    int[] packed = new int[src.getWidth() * src.getHeight()];

    src.getPixels(packed, 0, src.getWidth(), 0, 0, src.getWidth(), src.getHeight());

    for (int i = 0, srcOff = 0, dstOff = 0; i < src.getHeight(); i++) {
        for (int j = 0; j < src.getWidth(); j++, srcOff++, dstOff += 3) {
            int rgb = packed[srcOff];
            dstData[dstOff]     = (rgb >> 16) & 0xff;
            dstData[dstOff + 1] = (rgb >> 8) & 0xff;
            dstData[dstOff + 2] = rgb & 0xff;
        }
    }
}

public static Bitmap toBitmap(Picture src) {
    Bitmap dst = Bitmap.create(pic.getWidth(), pic.getHeight(), ARGB_8888);
    toBitmap(src, dst);
    return dst;
}

public static void toBitmap(Picture src, Bitmap dst) {
    int[] srcData = src.getPlaneData(0);
    int[] packed = new int[src.getWidth() * src.getHeight()];

    for (int i = 0, dstOff = 0, srcOff = 0; i < src.getHeight(); i++) {
        for (int j = 0; j < src.getWidth(); j++, dstOff++, srcOff += 3) {
            packed[dstOff] = (srcData[srcOff] << 16) | (srcData[srcOff + 1] << 8) | srcData[srcOff + 2];
        }
    }
    dst.setPixels(packed, 0, src.getWidth(), 0, 0, src.getWidth(), src.getHeight());
}

最后解码后,您将得到YUV帧,为了将其转换为RGB帧,请执行以下操作:

Transform transform = ColorUtil.getTransform(pic.getColor(), ColorSpace.RGB);
Picture rgb = Picture.create(pic.getWidth(), pic.getHeight(), ColorSpace.RGB);
transform.transform(pic, rgb);

请确保您下载包含所有依赖项的 JAR 文件:http://jcodec.org/downloads/jcodec-0.1.3-uberjar.jar


谢谢你的回答。编码操作在计算上有多大的强度? - biggvsdiccvs

1

0

检查ffmpeg库和x264。编译并组合这些库。它具有非常好的视频编码/解码API。如果您只编译ffmpeg,则无法对h.264进行编码,只能进行解码。要对h264进行编码,您需要编译x264库。http://ubuntuforums.org/showthread.php?t=786095。相当不错的教程。 这是C API,但您可以使用本地方法调用它。

最好的问候,对我的英语表示抱歉。


0
获取或编写一个能够编码和解码所需格式的类。确保它是服务提供者接口。将其添加到应用程序的运行时类路径中。使用JMF的默认播放器进行播放。对于编辑方面不是很确定。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接