电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Android 錄制音視頻的完整代碼

瀏覽:80日期:2022-09-17 18:00:35

Android 錄制音視頻的完整代碼

打開(kāi)camera

private void openCamera(int position) { if (mCamera == null) { mCamera = Camera.open(position); int degree = getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 0 : 90; mCamera.setDisplayOrientation(degree); } }

camera默認(rèn)是橫屏的,所以我們要使用豎屏錄制要旋轉(zhuǎn)90度

int degree = getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 0 : 90; mCamera.setDisplayOrientation(degree);

camera預(yù)覽

我們要選擇一個(gè)與我們要顯示的SurfaceView大小比例最接近的一個(gè)camera預(yù)覽大小,這里要特別注意camera支持的寬高都是寬大于高。

所以就有了下面這段選擇代碼

private Size getBestCameraResolution(Camera.Parameters parameters, Size screenResolution) { float tmp = 0f; float mindiff = 100f; Log.e('yuanVideo', 'screen height=' + screenResolution.getHeight()); float width_d_height; if (screenResolution.getWidth() > screenResolution.getHeight()) { width_d_height = (float) screenResolution.getWidth() / (float) screenResolution.getHeight(); } else { width_d_height = (float) screenResolution.getHeight() / (float) screenResolution.getWidth(); } Log.e('yuanVideo', 'width_d_height=' + width_d_height); Camera.Size best = null; List<Camera.Size> supportedPreviewSizes = parameters.getSupportedPreviewSizes(); for (Camera.Size s : supportedPreviewSizes) { tmp = Math.abs(((float) s.width / (float) s.height) - width_d_height); Log.e('yuanVideo', 'support ratio=' + tmp); if (tmp < mindiff) { mindiff = tmp; best = s; } } Log.e('yuanVideo', 'best height=' + best.height); return new Size(best.width, best.height); }

初始化MediaRecorder

private boolean prepareMediaRecorder() { // 創(chuàng)建MediaPlayer對(duì)象 mCamera.unlock(); mRecorder = new MediaRecorder(); mRecorder.reset(); mRecorder.setCamera(mCamera); // 設(shè)置從麥克風(fēng)采集聲音(或來(lái)自錄像機(jī)的聲音AudioSource.CAMCORDER) mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC); // 設(shè)置從攝像頭采集圖像 mRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA); Log.e('yuanProfile', 'QUALITY_LOW=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW)); Log.e('yuanProfile', 'QUALITY_HIGH=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH)); Log.e('yuanProfile', 'QUALITY_QCIF=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QCIF)); Log.e('yuanProfile', 'QUALITY_480P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P)); Log.e('yuanProfile', 'QUALITY_720P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P)); Log.e('yuanProfile', 'QUALITY_1080P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P)); Log.e('yuanProfile', 'QUALITY_QVGA=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QVGA)); Log.e('yuanProfile', 'QUALITY_2160P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2160P)); Log.e('yuanProfile', 'QUALITY_VGA=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_VGA)); Log.e('yuanProfile', 'QUALITY_4KDCI=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_4KDCI)); Log.e('yuanProfile', 'QUALITY_QHD=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QHD)); Log.e('yuanProfile', 'QUALITY_2K=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2K)); if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_720P)); } else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_1080P)); } else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_480P)); } else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_LOW)); } else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH)); } else { return false; } // mTempList.add(mCurrentTempRecordData); mRecorder.setOutputFile(mCurPath); mRecorder.setPreviewDisplay(activtityVideoRecordBinding.sView.getHolder().getSurface()); // ① int degree; if(getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE){ int degree ; if (cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT) { degree = 270; } else { degree = 90; } mRecorder.setOrientationHint(degree); } try { mRecorder.prepare(); } catch (Exception e) { e.printStackTrace(); return false; } return true; }

這里也要設(shè)置視頻的旋轉(zhuǎn)參數(shù)

if(getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE){ int degree ; if (cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT) { degree = 270; } else { degree = 90; } mRecorder.setOrientationHint(degree); }

下面是完整的代碼

package com.yuanxuzhen.ffmpeg; import android.app.Activity;import android.content.Context;import android.content.res.Configuration;import android.graphics.PixelFormat;import android.hardware.Camera;import android.media.CamcorderProfile;import android.media.MediaRecorder;import android.os.Build;import android.os.Bundle;import android.util.DisplayMetrics;import android.util.Log;import android.util.Size;import android.view.SurfaceHolder;import android.view.View;import android.view.Window;import android.view.WindowManager; import androidx.annotation.NonNull;import androidx.annotation.Nullable;import androidx.annotation.RequiresApi; import com.yuanxuzhen.ffmpeg.databinding.ActivtityVideoRecordBinding; import java.io.File;import java.io.IOException;import java.util.List;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors; public class VideoRecordActivity extends Activity { ActivtityVideoRecordBinding activtityVideoRecordBinding; MediaRecorder mRecorder; private boolean isRecording = false; private int cameraPosition = Camera.CameraInfo.CAMERA_FACING_FRONT;//0代表前置攝像頭,1代表后置攝像頭 private Camera mCamera; private Camera.Parameters mParameters; private String mCurPath = null; private VideoTempRecordData mCurrentTempRecordData = null; @Override protected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);requestWindowFeature(Window.FEATURE_NO_TITLE);getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN);getWindow().setFormat(PixelFormat.TRANSLUCENT);mCurPath = DirUtil.getCacheDir(this) + File.separator + 'out.mp4';activtityVideoRecordBinding = ActivtityVideoRecordBinding.inflate(getLayoutInflater());setContentView(activtityVideoRecordBinding.getRoot());activtityVideoRecordBinding.sView.getHolder().setKeepScreenOn(true);activtityVideoRecordBinding.sView.getHolder().addCallback(new SurfaceHolder.Callback() { @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) @Override public void surfaceCreated(@NonNull SurfaceHolder holder) {openPreView(); } @Override public void surfaceChanged(SurfaceHolder surfaceHolder, int format, int width, int height) {Log.e('yuanVideo', 'surfaceChanged height=' + height); } @Override public void surfaceDestroyed(@NonNull SurfaceHolder holder) { }}); activtityVideoRecordBinding.recordOrStop.setText('開(kāi)始');activtityVideoRecordBinding.recordOrStop.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {if (isRecording) { Log.d('TAG', '停止錄像'); stopRecord();} else { startRecord();} }});activtityVideoRecordBinding.change.setOnClickListener(new View.OnClickListener() { @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) @Override public void onClick(View v) {if(isRecording){ return;}releaseCamera();cameraPosition = cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT ? Camera.CameraInfo.CAMERA_FACING_BACK : Camera.CameraInfo.CAMERA_FACING_FRONT;openCamera(cameraPosition);openPreView(); }}); } /** * 1.打開(kāi)相機(jī) */ private void openCamera(int position) {if (mCamera == null) { mCamera = Camera.open(position); int degree = getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 0 : 90; mCamera.setDisplayOrientation(degree);} } /** * initCameraAndSurfaceViewHolder初始化hoder后 * 2.設(shè)置預(yù)覽功能 */ @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) private void openPreView() {try { if (mCamera != null) {mParameters = mCamera.getParameters();mCamera.setPreviewDisplay(activtityVideoRecordBinding.sView.getHolder());Size screenPoint = getScreenMetrics(VideoRecordActivity.this);Size bestPreviewSize = getBestCameraResolution(mCamera.getParameters(), screenPoint);mParameters.setPreviewSize(bestPreviewSize.getWidth(), bestPreviewSize.getHeight());mCamera.setParameters(mParameters);mCamera.startPreview();mCamera.autoFocus(new Camera.AutoFocusCallback() { @Override public void onAutoFocus(boolean success, Camera camera) {Log.e('yuanVideo', 'autoFocus success=' + success); }}); mCamera.setPreviewCallback(new Camera.PreviewCallback() { @Override public void onPreviewFrame(byte[] data, Camera camera) {Log.i('TAG', '獲取預(yù)覽幀...');Log.d('TAG', '預(yù)覽幀大小:' + String.valueOf(data.length)); }}); }} catch (IOException e) { e.printStackTrace();} } private boolean prepareMediaRecorder() {// 創(chuàng)建MediaPlayer對(duì)象mCamera.unlock();mRecorder = new MediaRecorder();mRecorder.reset();mRecorder.setCamera(mCamera); // 設(shè)置從麥克風(fēng)采集聲音(或來(lái)自錄像機(jī)的聲音AudioSource.CAMCORDER)mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);// 設(shè)置從攝像頭采集圖像mRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);Log.e('yuanProfile', 'QUALITY_LOW=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW));Log.e('yuanProfile', 'QUALITY_HIGH=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH));Log.e('yuanProfile', 'QUALITY_QCIF=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QCIF));Log.e('yuanProfile', 'QUALITY_480P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P));Log.e('yuanProfile', 'QUALITY_720P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P));Log.e('yuanProfile', 'QUALITY_1080P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P));Log.e('yuanProfile', 'QUALITY_QVGA=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QVGA));Log.e('yuanProfile', 'QUALITY_2160P=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2160P));Log.e('yuanProfile', 'QUALITY_VGA=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_VGA));Log.e('yuanProfile', 'QUALITY_4KDCI=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_4KDCI));Log.e('yuanProfile', 'QUALITY_QHD=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_QHD));Log.e('yuanProfile', 'QUALITY_2K=' + CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_2K)); if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_720P)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_720P));} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_480P)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_1080P));} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_1080P)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_480P));} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_LOW)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_LOW));} else if (CamcorderProfile.hasProfile(CamcorderProfile.QUALITY_HIGH)) { mRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH));} else { return false;}//mTempList.add(mCurrentTempRecordData);mRecorder.setOutputFile(mCurPath);mRecorder.setPreviewDisplay(activtityVideoRecordBinding.sView.getHolder().getSurface()); // ① if(getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE){ int degree ; if (cameraPosition == Camera.CameraInfo.CAMERA_FACING_FRONT) {degree = 270; } else {degree = 90; } mRecorder.setOrientationHint(degree);} try { mRecorder.prepare();} catch (Exception e) { e.printStackTrace(); return false;}return true; } private void startRecord() {if (prepareMediaRecorder()) { mRecorder.start(); isRecording = true; activtityVideoRecordBinding.recordOrStop.setText('停止');} else { releaseMediaRecorder(); isRecording = false; activtityVideoRecordBinding.recordOrStop.setText('開(kāi)始');} } private void stopRecord() {if (mRecorder == null) { return;}mRecorder.stop();releaseMediaRecorder();isRecording = false;activtityVideoRecordBinding.recordOrStop.setText('開(kāi)始'); } @Nullable @Override public CharSequence onCreateDescription() {return super.onCreateDescription(); } @Override protected void onDestroy() {releaseCamera();releaseMediaRecorder();super.onDestroy(); } /** * 釋放相機(jī)資源 */ private void releaseCamera() {if (mCamera != null) { mCamera.setPreviewCallback(null); mCamera.stopPreview(); mCamera.release(); mCamera = null;} } private void releaseMediaRecorder() {if (mRecorder != null) { mRecorder.reset(); mRecorder.release(); mRecorder = null; mCamera.lock();} } @Override protected void onResume() {super.onResume();openCamera(cameraPosition); } @Override protected void onPause() {super.onPause();releaseMediaRecorder();releaseCamera(); } /** * 獲取最佳預(yù)覽大小 * * @param parameters 相機(jī)參數(shù) * @param screenResolution 屏幕寬高 * @return */ @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) private Size getBestCameraResolution(Camera.Parameters parameters, Size screenResolution) {float tmp = 0f; float mindiff = 100f;Log.e('yuanVideo', 'screen height=' + screenResolution.getHeight()); float width_d_height;if (screenResolution.getWidth() > screenResolution.getHeight()) { width_d_height = (float) screenResolution.getWidth() / (float) screenResolution.getHeight();} else { width_d_height = (float) screenResolution.getHeight() / (float) screenResolution.getWidth();}Log.e('yuanVideo', 'width_d_height=' + width_d_height); Camera.Size best = null; List<Camera.Size> supportedPreviewSizes = parameters.getSupportedPreviewSizes();for (Camera.Size s : supportedPreviewSizes) { tmp = Math.abs(((float) s.width / (float) s.height) - width_d_height); Log.e('yuanVideo', 'support ratio=' + tmp); if (tmp < mindiff) {mindiff = tmp; best = s; } }Log.e('yuanVideo', 'best height=' + best.height); return new Size(best.width, best.height); } /** * 獲取屏幕寬度和高度,單位為px * * @param context * @return */ @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) public static Size getScreenMetrics(Context context) {DisplayMetrics dm = context.getResources().getDisplayMetrics(); int w_screen = dm.widthPixels; int h_screen = dm.heightPixels; return new Size(w_screen, h_screen); } }

布局

<?xml version='1.0' encoding='utf-8'?><RelativeLayout xmlns:android='http://schemas.android.com/apk/res/android' android:orientation='vertical' android:layout_width='match_parent' android:layout_height='match_parent'> <!-- 顯示視頻預(yù)覽的SurfaceView --> <com.yuanxuzhen.ffmpeg.ResizeAbleSurfaceViewandroid: android:layout_width='match_parent'android:layout_height='match_parent'android:layout_centerInParent='true'/> <RelativeLayoutandroid:layout_width='match_parent'android:layout_height='wrap_content'android:layout_alignParentTop='true'><TextView android: android:layout_width='wrap_content' android:layout_height='wrap_content' android:text='0秒' android:layout_centerInParent='true' android:textColor='@color/white' /><Button android: android:layout_width='wrap_content' android:layout_height='66dp' android:text='切換攝像頭' android:layout_alignParentEnd='true' /> </RelativeLayout> <LinearLayoutandroid:orientation='horizontal'android:layout_width='wrap_content'android:layout_height='wrap_content'android:gravity='center_horizontal'android:layout_alignParentBottom='true'android:layout_centerHorizontal='true'><Button android: android:layout_width='66dp' android:layout_height='66dp' android:text='錄制' /><Button android: android:layout_width='66dp' android:layout_height='66dp' android:text='保存' /> </LinearLayout></RelativeLayout>

package com.yuanxuzhen.ffmpeg; import android.content.Context;import android.util.AttributeSet;import android.view.SurfaceView; public class ResizeAbleSurfaceView extends SurfaceView { private int mWidth = -1; private int mHeight = -1; public ResizeAbleSurfaceView(Context context) {super(context); } public ResizeAbleSurfaceView(Context context, AttributeSet attrs) {super(context, attrs); } public ResizeAbleSurfaceView(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {if (-1 == mWidth || -1 == mHeight) { super.onMeasure(widthMeasureSpec, heightMeasureSpec);}else { setMeasuredDimension(mWidth, mHeight);} } public void resize(int width, int height) {mWidth = width;mHeight = height;getHolder().setFixedSize(width, height);requestLayout();invalidate(); } }

package com.yuanxuzhen.ffmpeg; import android.content.Context;import android.os.Environment; import java.io.File; public class DirUtil { public static final String WEBVIEW_CACHE = '.webviewCache'; public static final String IMAGE_PATH = 'image'; public static final String DOWNLOAD_PATH = 'download'; public static final String VIDEO_PATH = '.video'; public static final String NET_PATH = '.net'; //image public static String getImageDir(Context context) {return getCacheDir(context) + File.separator + IMAGE_PATH; } //webview public static String getWebviewCache(Context context) {return getCacheDir(context) + File.separator + WEBVIEW_CACHE; } //download public static String getDownloadDir(Context context) {return getCacheDir(context) + File.separator + DOWNLOAD_PATH; } //video public static String getVideoPath(Context context) {return getCacheDir(context) + File.separator + VIDEO_PATH; } //net public static String getNetPath(Context context) {return getCacheDir(context) + File.separator + NET_PATH; } public static String getCacheDir(Context context) { if (context == null) { return '';}String path = null;if (context.getExternalCacheDir() != null&& (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())|| !Environment.isExternalStorageRemovable())) { //外部存儲(chǔ)可用 path = context.getExternalCacheDir().getPath();} else { //內(nèi)部存儲(chǔ)不可用 path = context.getCacheDir().getPath();}return path; } }

以上就是Android 錄制音視頻的詳細(xì)內(nèi)容,更多關(guān)于Android 錄制音視頻的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Android
相關(guān)文章:
主站蜘蛛池模板: 飞飞影视_热门电影在线观看_影视大全 | 色谱柱-淋洗液罐-巴罗克试剂槽-巴氏吸管-5ml样品瓶-SBS液氮冻存管-上海希言科学仪器有限公司 | 南京泽朗生物科技有限公司-液体饮料代加工_果汁饮料代加工_固体饮料代加工 | 石英砂矿石色选机_履带辣椒色选机_X光异物检测机-合肥幼狮光电科技 | 依维柯自动挡房车,自行式国产改装房车,小型房车价格,中国十大房车品牌_南京拓锐斯特房车 - 南京拓锐斯特房车 | 北京森语科技有限公司-模型制作专家-展览展示-沙盘模型设计制作-多媒体模型软硬件开发-三维地理信息交互沙盘 | 液氮罐_液氮容器_自增压液氮罐-北京君方科仪科技发展有限公司 | 小程序开发公司_APP开发多少钱_软件开发定制_微信小程序制作_客户销售管理软件-济南小溪畅流网络科技有限公司 | 钢制拖链生产厂家-全封闭钢制拖链-能源钢铝拖链-工程塑料拖链-河北汉洋机械制造有限公司 | 小威小说网 - 新小威小说网 - 小威小说网小说搜索引擎 | 全自动实验室洗瓶机,移液管|培养皿|进样瓶清洗机,清洗剂-广州摩特伟希尔机械设备有限责任公司 | 欧必特空气能-商用空气能热水工程,空气能热水器,超低温空气源热泵生产厂家-湖南欧必特空气能公司 | 「阿尔法设计官网」工业设计_产品设计_产品外观设计 深圳工业设计公司 | 电动打包机_气动打包机_钢带捆扎机_废纸打包机_手动捆扎机 | 飞行者联盟-飞机模拟机_无人机_低空经济_航空技术交流平台 | 东莞市海宝机械有限公司-不锈钢分选机-硅胶橡胶-生活垃圾-涡电流-静电-金属-矿石分选机 | 微信聊天记录恢复_手机短信删除怎么恢复_通讯录恢复软件下载-快易数据恢复 | 金现代信息产业股份有限公司--数字化解决方案供应商 | 雷达液位计_超声波风速风向仪_雨量传感器_辐射传感器-山东风途物联网 | 「安徽双凯」自动售货机-无人售货机-成人用品-自动饮料食品零食售货机 | 海外仓系统|国际货代系统|退货换标系统|WMS仓储系统|海豚云 | 空冷器|空气冷却器|空水冷却器-无锡赛迪森机械有限公司[官网] | 工程管道/塑料管材/pvc排水管/ppr给水管/pe双壁波纹管等品牌管材批发厂家-河南洁尔康建材 | 巨野电机维修-水泵维修-巨野县飞宇机电维修有限公司 | 不锈钢复合板|钛复合板|金属复合板|南钢集团安徽金元素复合材料有限公司-官网 | 钢木实验台-全钢实验台-化验室通风柜-实验室装修厂家-杭州博扬实验设备 | 知网论文检测系统入口_论文查重免费查重_中国知网论文查询_学术不端检测系统 | 中医治疗皮肤病_潍坊银康医院「山东」重症皮肤病救治平台 | 集装箱箱号识别_自重载重图像识别_铁路车号自动识别_OCR图像识别 | 环保袋,无纺布袋,无纺布打孔袋,保温袋,环保袋定制,环保袋厂家,环雅包装-十七年环保袋定制厂家 | 小威小说网 - 新小威小说网 - 小威小说网小说搜索引擎 | 全自动包衣机-无菌分装隔离器-浙江迦南科技股份有限公司 | 砂石生产线_石料生产线设备_制砂生产线设备价格_生产厂家-河南中誉鼎力智能装备有限公司 | 铸铝门厂家,别墅大门庭院大门,别墅铸铝门铜门[十大品牌厂家]军强门业 | 污水提升器,污水提升泵,污水提升装置-德国泽德(zehnder)水泵系统有限公司 | 月嫂_保姆_育婴_催乳_母婴护理_产后康复_养老护理-吉祥到家家政 硫酸亚铁-聚合硫酸铁-除氟除磷剂-复合碳源-污水处理药剂厂家—长隆科技 | 传动滚筒_厂家-淄博海恒机械制造厂 | 卡诺亚轻高定官网_卧室系统_整家定制_定制家居_高端定制_全屋定制加盟_定制家具加盟_定制衣柜加盟 | 北京乾茂兴业科技发展有限公司 | 数显水浴恒温振荡器-分液漏斗萃取振荡器-常州市凯航仪器有限公司 | 蚂蚁分类信息系统 - PHP同城分类信息系统 - MayiCMS |