android之圖片選擇器--pictureselector

推薦一個安卓圖片/視頻/文件選擇器。簡單好用。
不多廢話。直接上代碼:

首先,添加依賴:

	//圖片選擇器api 'io.github.lucksiege:pictureselector:v3.11.1'//圖片壓縮api 'io.github.lucksiege:compress:v3.11.1'//圖片裁剪api 'io.github.lucksiege:ucrop:v3.11.1'//自定義相機api 'io.github.lucksiege:camerax:v3.11.1'api 'com.github.bumptech.glide:glide:4.12.0'

這些依賴是必須的。其次是封裝一下,方便后續使用
首先,創建 VideoThumbListener.java:

public class VideoThumbListener implements OnVideoThumbnailEventListener {private Context context;public VideoThumbListener(Context context) {this.context = context;}@Overridepublic void onVideoThumbnail(Context context, String videoPath, OnKeyValueResultCallbackListener call) {Glide.with(context).asBitmap().sizeMultiplier(0.6F).load(videoPath).into(new CustomTarget<Bitmap>() {@Overridepublic void onResourceReady(@NonNull Bitmap resource, @Nullable Transition<? super Bitmap> transition) {ByteArrayOutputStream stream = new ByteArrayOutputStream();resource.compress(Bitmap.CompressFormat.JPEG, 60, stream);FileOutputStream fos = null;String result = null;try {File targetFile = new File(getVideoThumbnailDir(), "thumbnails_" + System.currentTimeMillis() + ".jpg");fos = new FileOutputStream(targetFile);fos.write(stream.toByteArray());fos.flush();result = targetFile.getAbsolutePath();} catch (IOException e) {e.printStackTrace();} finally {PictureFileUtils.close(fos);PictureFileUtils.close(stream);}if (call != null) {call.onCallback(videoPath, result);}}@Overridepublic void onLoadCleared(@Nullable Drawable placeholder) {if (call != null) {call.onCallback(videoPath, "");}}});}private String getVideoThumbnailDir() {File externalFilesDir = context.getExternalFilesDir("");File customFile = new File(externalFilesDir.getAbsolutePath(), "Thumbnail");if (!customFile.exists()) {customFile.mkdirs();}return customFile.getAbsolutePath() + File.separator;}
}

再次就是圖片加載器的封裝:GlideUtils:


public class GlideUtils implements ImageEngine {/*** 加載圖片** @param context   上下文* @param url       資源url* @param imageView 圖片承載控件*/@Overridepublic void loadImage(Context context, String url, ImageView imageView) {if (!ActivityCompatHelper.assertValidRequest(context)) {return;}Glide.with(context).load(url)// .error(R.mipmap.icon_photo_error).into(imageView);}/*** 加載圖片** @param context    上下文* @param url        資源url* @param imageView  圖片承載控件* @param errorImage 圖片加載失敗資源文件*/public void loadImage(Context context, String url, ImageView imageView, int errorImage) {if (!ActivityCompatHelper.assertValidRequest(context)) {return;}Glide.with(context).load(url).error(errorImage).into(imageView);}@Overridepublic void loadImage(Context context, ImageView imageView, String url, int maxWidth, int maxHeight) {if (!ActivityCompatHelper.assertValidRequest(context)) {return;}Glide.with(context).load(url)
//                .error(R.mipmap.ic_launcher).override(maxWidth, maxHeight).into(imageView);}/*** 加載相冊目錄封面** @param context   上下文* @param url       圖片路徑* @param imageView 承載圖片ImageView*/@Overridepublic void loadAlbumCover(Context context, String url, ImageView imageView) {if (!ActivityCompatHelper.assertValidRequest(context)) {return;}Glide.with(context).asBitmap().load(url)
//                .error(R.mipmap.ic_launcher).override(180, 180).sizeMultiplier(0.5f).transform(new CenterCrop(), new RoundedCorners(8))
//                .placeholder(R.mipmap.ic_launcher).into(imageView);}/*** 默認圖片** @param context* @param url* @param imageView*/public void loadVideoThumb(Context context, String url, ImageView imageView) {loadVideoThumb(context, url, 0, imageView);}public void loadVideoThumb(Context context, String url, int errorImage, ImageView imageView) {Glide.with(context).setDefaultRequestOptions(new RequestOptions().frame(1000000).centerCrop().error(errorImage).placeholder(errorImage)).load(url).into(imageView);}/*** 獲取視頻第一證* @param path 文件路徑* @return*/public Bitmap loadVideoThumb(String path) {MediaMetadataRetriever media = new MediaMetadataRetriever();media.setDataSource(path);return media.getFrameAtTime();}/*** 加載圖片列表圖片** @param context   上下文* @param url       圖片路徑* @param imageView 承載圖片ImageView*/@Overridepublic void loadGridImage(Context context, String url, ImageView imageView) {if (!ActivityCompatHelper.assertValidRequest(context)) {return;}Glide.with(context).load(url).override(200, 200).centerCrop()
//                .error(R.mipmap.ic_launcher)
//                .placeholder(R.mipmap.ic_launcher).into(imageView);}@Overridepublic void pauseRequests(Context context) {Glide.with(context).pauseRequests();}@Overridepublic void resumeRequests(Context context) {Glide.with(context).resumeRequests();}private GlideUtils() {}private static final class InstanceHolder {static final GlideUtils instance = new GlideUtils();}public static GlideUtils createGlideEngine() {return InstanceHolder.instance;}
}

最后。就是正題了,PictureUtils.java:

import android.Manifest;
import android.content.Context;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.drawable.Drawable;
import android.net.Uri;
import android.provider.MediaStore;
import android.widget.ImageView;import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;import com.bumptech.glide.Glide;
import com.bumptech.glide.request.target.CustomTarget;
import com.bumptech.glide.request.transition.Transition;
import com.luck.lib.camerax.SimpleCameraX;
import com.luck.picture.lib.basic.PictureSelector;
import com.luck.picture.lib.config.PictureMimeType;
import com.luck.picture.lib.config.SelectMimeType;
import com.luck.picture.lib.engine.CompressFileEngine;
import com.luck.picture.lib.engine.CropFileEngine;
import com.luck.picture.lib.engine.VideoPlayerEngine;
import com.luck.picture.lib.entity.LocalMedia;
import com.luck.picture.lib.entity.MediaExtraInfo;
import com.luck.picture.lib.interfaces.OnCameraInterceptListener;
import com.luck.picture.lib.interfaces.OnExternalPreviewEventListener;
import com.luck.picture.lib.interfaces.OnRecordAudioInterceptListener;
import com.luck.picture.lib.interfaces.OnResultCallbackListener;
import com.luck.picture.lib.permissions.PermissionChecker;
import com.luck.picture.lib.permissions.PermissionResultCallback;
import com.luck.picture.lib.utils.MediaUtils;
import com.luck.picture.lib.utils.ToastUtils;
import com.yalantis.ucrop.UCrop;
import com.yalantis.ucrop.UCropImageEngine;import java.io.File;
import java.util.ArrayList;import top.zibin.luban.Luban;
import top.zibin.luban.OnNewCompressListener;/*** @author: zzw* @Description:* @Date: 2022/9/21 21:37*/
public class PictureUtils {/*** 打開攝像頭 拍照** @param context* @param isRotateImage                   true 前置 false 后置* @param onPictureSelectorResultListener 回調*/public static void openCamera(Context context, boolean isRotateImage, OnPictureSelectorResultListener onPictureSelectorResultListener) {openCamera(context, SelectMimeType.ofImage(), isRotateImage, onPictureSelectorResultListener);}/*** 打開攝像頭 錄制視頻** @param context* @param isRotateImage                   true 前置 false 后置* @param onPictureSelectorResultListener 回調*/public static void openVideo(Context context, boolean isRotateImage, OnPictureSelectorResultListener onPictureSelectorResultListener) {openCamera(context, SelectMimeType.ofVideo(), isRotateImage, onPictureSelectorResultListener);}/*** 打開攝像頭** @param context                         上下文* @param onPictureSelectorResultListener 回調*/public static void openCamera(Context context, int openCamera, boolean isRotateImage, OnPictureSelectorResultListener onPictureSelectorResultListener) {PictureSelector.create(context).openCamera(openCamera).isCameraAroundState(isRotateImage).setVideoThumbnailListener(new VideoThumbListener(context)).setCompressEngine((CompressFileEngine) (context1, source, call) -> Luban.with(context1).load(source).ignoreBy(100).setCompressListener(new OnNewCompressListener() {@Overridepublic void onStart() {}@Overridepublic void onSuccess(String source, File compressFile) {if (call != null) {call.onCallback(source, compressFile.getAbsolutePath());}}@Overridepublic void onError(String source, Throwable e) {if (call != null) {call.onCallback(source, null);}}}).launch()).forResult(new OnResultCallbackListener<LocalMedia>() {@Overridepublic void onResult(ArrayList<LocalMedia> result) {onPictureSelectorResultListener.onResult(result);}@Overridepublic void onCancel() {}});}/*** 設置頭像** @param mContext* @param selectResult                    結果* @param onPictureSelectorResultListener 結果回調*/public static void createAvatar(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {create(mContext, SelectMimeType.ofImage(), selectResult, 1, 1, true, onPictureSelectorResultListener);}/*** 選擇單張圖片** @param mContext* @param selectResult                    結果* @param onPictureSelectorResultListener 結果回調*/public static void createImageMin(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {create(mContext, SelectMimeType.ofImage(), selectResult, 1, 1, false, onPictureSelectorResultListener);}/*** 選擇多張圖片** @param mContext* @param selectResult                    結果* @param selectMax                       最多選擇* @param onPictureSelectorResultListener 結果回調*/public static void createImageMax(Context mContext, int selectMax, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {create(mContext, SelectMimeType.ofImage(), selectResult, 1, selectMax, false, onPictureSelectorResultListener);}/*** 選擇單個視頻** @param mContext* @param selectResult                    結果* @param onPictureSelectorResultListener 結果回調*/public static void createVideo(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {create(mContext, SelectMimeType.ofVideo(), selectResult, 1, 1, false, onPictureSelectorResultListener);}/*** 選擇單個音頻** @param mContext* @param selectResult                    結果* @param onPictureSelectorResultListener 結果回調*/public static void createAudio(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {create(mContext, SelectMimeType.ofAudio(), selectResult, 1, 1, false, onPictureSelectorResultListener);}/*** 選擇 媒體 圖片 視頻 音頻** @param mContext* @param selectResult                    結果* @param onPictureSelectorResultListener 結果回調*/public static void createPicture(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {create(mContext, SelectMimeType.ofAll(), selectResult, 1, 1, false, onPictureSelectorResultListener);}/*** 默認設置** @param mContext* @param selectMimeType                  結果* @param selectResult                    結果* @param selectMin                       最少選擇* @param selectMax                       最多選擇* @param isCrop                          是否剪裁* @param onPictureSelectorResultListener 結果回到*/public static void create(Context mContext, int selectMimeType, ArrayList<LocalMedia> selectResult, int selectMin, int selectMax, boolean isCrop, OnPictureSelectorResultListener onPictureSelectorResultListener) {PictureSelector.create(mContext).openGallery(selectMimeType).setMaxSelectNum(selectMax).setCropEngine(getCropFileEngine(isCrop)).setMinSelectNum(selectMin).setFilterVideoMaxSecond(selectMimeType == SelectMimeType.ofVideo() ? 60 : 60 * 10).setFilterVideoMinSecond(5).setRecordVideoMaxSecond(selectMimeType == SelectMimeType.ofVideo() ? 60 : 60 * 10).setRecordVideoMinSecond(5).setFilterMaxFileSize(100 * 1024 * 1024).setCameraInterceptListener(new MeOnCameraInterceptListener()).isFilterSizeDuration(true).setSelectedData(selectResult).setRecordAudioInterceptListener(new MeOnRecordAudioInterceptListener()).setCompressEngine((CompressFileEngine) (context, source, call) -> Luban.with(context).load(source).ignoreBy(100).setCompressListener(new OnNewCompressListener() {@Overridepublic void onStart() {}@Overridepublic void onSuccess(String source, File compressFile) {if (call != null) {call.onCallback(source, compressFile.getAbsolutePath());}}@Overridepublic void onError(String source, Throwable e) {if (call != null) {call.onCallback(source, null);}}}).launch()).setImageEngine(GlideUtils.createGlideEngine()).forResult(new OnResultCallbackListener<LocalMedia>() {@Overridepublic void onResult(ArrayList<LocalMedia> result) {for (LocalMedia media : result) {if (media.getWidth() == 0 || media.getHeight() == 0) {if (PictureMimeType.isHasImage(media.getMimeType())) {MediaExtraInfo imageExtraInfo = MediaUtils.getImageSize(mContext, media.getPath());media.setWidth(imageExtraInfo.getWidth());media.setHeight(imageExtraInfo.getHeight());} else if (PictureMimeType.isHasVideo(media.getMimeType())) {MediaExtraInfo videoExtraInfo = MediaUtils.getVideoSize(mContext, media.getPath());media.setWidth(videoExtraInfo.getWidth());media.setHeight(videoExtraInfo.getHeight());}}
//                            LogUtils.e("文件名: " + media.getFileName() + "\n" +
//                                    "是否壓縮:" + media.isCompressed() + "\n" +
//                                    "壓縮路徑:" + media.getCompressPath() + "\n" +
//                                    "初始路徑:" + media.getPath() + "\n" +
//                                    "絕對路徑:" + media.getRealPath() + "\n" +
//                                    "是否裁剪:" + media.isCut() + "\n" +
//                                    "裁剪路徑:" + media.getCutPath() + "\n" +
//                                    "是否開啟原圖:" + media.isOriginal() + "\n" +
//                                    "原圖路徑:" + media.getOriginalPath() + "\n" +
//                                    "沙盒路徑:" + media.getSandboxPath() + "\n" +
//                                    "水印路徑:" + media.getWatermarkPath() + "\n" +
//                                    "視頻縮略圖:" + media.getVideoThumbnailPath() + "\n" +
//                                    "原始寬高: " + media.getWidth() + "x" + media.getHeight() + "\n" +
//                                    "裁剪寬高: " + media.getCropImageWidth() + "x" + media.getCropImageHeight() + "\n" +
//                                    "文件大小: " + PictureFileUtils.formatAccurateUnitFileSize(media.getSize()) + "\n" +
//                                    "文件大小: " + media.getSize() + "\n" +
//                                    "文件時長: " + media.getDuration()
//                            );}onPictureSelectorResultListener.onResult(result);}@Overridepublic void onCancel() {}});}/*** 查看圖片大圖** @param mContext* @param position* @param localMedia*/public static void openImage(Context mContext, int position, ArrayList<LocalMedia> localMedia) {PictureSelector.create(mContext).openPreview().isHidePreviewDownload(true).setImageEngine(GlideUtils.createGlideEngine()).setExternalPreviewEventListener(new OnExternalPreviewEventListener() {@Overridepublic void onPreviewDelete(int position) {}@Overridepublic boolean onLongPressDownload(Context context, LocalMedia media) {return false;}}).startActivityPreview(position, false, localMedia);}public static void openImage(Context mContext, int position, String imageUrl) {ArrayList<LocalMedia> localMedia = new ArrayList<>();for (String url : imageUrl.split(",")) {localMedia.add(LocalMedia.generateHttpAsLocalMedia(url));}PictureSelector.create(mContext).openPreview().setImageEngine(GlideUtils.createGlideEngine()).setExternalPreviewEventListener(new OnExternalPreviewEventListener() {@Overridepublic void onPreviewDelete(int position) {}@Overridepublic boolean onLongPressDownload(Context context, LocalMedia media) {return false;}}).startActivityPreview(position, false, localMedia);}/*** 預覽視頻** @param mContext* @param position* @param localMedia*/public static void openVideo(Context mContext, int position, ArrayList<LocalMedia> localMedia) {openVideo(mContext, position, localMedia, null);}/*** 預覽視頻** @param mContext* @param position* @param localMedia*/public static void openVideo(Context mContext, int position, ArrayList<LocalMedia> localMedia, VideoPlayerEngine videoPlayerEngine) {PictureSelector.create(mContext).openPreview().setImageEngine(GlideUtils.createGlideEngine()).setVideoPlayerEngine(videoPlayerEngine).isAutoVideoPlay(true).setExternalPreviewEventListener(new OnExternalPreviewEventListener() {@Overridepublic void onPreviewDelete(int position) {}@Overridepublic boolean onLongPressDownload(Context context, LocalMedia media) {return false;}}).startActivityPreview(position, false, localMedia);}/*** 裁剪引擎** @return*/private static ImageFileCropEngine getCropFileEngine(boolean isCrop) {return isCrop ? new ImageFileCropEngine() : null;}/*** 自定義裁剪*/private static class ImageFileCropEngine implements CropFileEngine {@Overridepublic void onStartCrop(Fragment fragment, Uri srcUri, Uri destinationUri, ArrayList<String> dataSource, int requestCode) {UCrop.Options options = buildOptions();UCrop uCrop = UCrop.of(srcUri, destinationUri, dataSource);uCrop.withOptions(options);uCrop.setImageEngine(new UCropImageEngine() {@Overridepublic void loadImage(Context context, String url, ImageView imageView) {Glide.with(context).load(url).override(180, 180).into(imageView);}@Overridepublic void loadImage(Context context, Uri url, int maxWidth, int maxHeight, OnCallbackListener<Bitmap> call) {Glide.with(context).asBitmap().load(url).override(maxWidth, maxHeight).into(new CustomTarget<Bitmap>() {@Overridepublic void onResourceReady(@NonNull Bitmap resource, @Nullable Transition<? super Bitmap> transition) {if (call != null) {call.onCall(resource);}}@Overridepublic void onLoadCleared(@Nullable Drawable placeholder) {if (call != null) {call.onCall(null);}}});}});uCrop.start(fragment.requireActivity(), fragment, requestCode);}}/*** 配制UCrop,可根據需求自我擴展** @return*/private static UCrop.Options buildOptions() {UCrop.Options options = new UCrop.Options();options.setHideBottomControls(true);options.setFreeStyleCropEnabled(true);options.setShowCropFrame(true);options.setShowCropGrid(false);options.setCircleDimmedLayer(false);options.withAspectRatio(1, 1);options.isCropDragSmoothToCenter(false);options.setMaxScaleMultiplier(100);return options;}public interface OnPictureSelectorResultListener {void onResult(ArrayList<LocalMedia> result);}/*** 自定義拍照*/private static class MeOnCameraInterceptListener implements OnCameraInterceptListener {@Overridepublic void openCamera(Fragment fragment, int cameraMode, int requestCode) {SimpleCameraX camera = SimpleCameraX.of();camera.isAutoRotation(true);camera.setCameraMode(cameraMode);camera.setVideoFrameRate(50);camera.setVideoBitRate(5 * 1024 * 1024);camera.isDisplayRecordChangeTime(true);camera.isManualFocusCameraPreview(true);camera.isZoomCameraPreview(true);camera.setImageEngine((context, url, imageView) -> Glide.with(context).load(url).into(imageView));camera.start(fragment.requireActivity(), fragment, requestCode);}}/*** 錄音回調事件*/private static class MeOnRecordAudioInterceptListener implements OnRecordAudioInterceptListener {@Overridepublic void onRecordAudio(Fragment fragment, int requestCode) {String[] recordAudio = {Manifest.permission.RECORD_AUDIO};if (PermissionChecker.isCheckSelfPermission(fragment.getContext(), recordAudio)) {startRecordSoundAction(fragment, requestCode);} else {PermissionChecker.getInstance().requestPermissions(fragment, new String[]{Manifest.permission.RECORD_AUDIO}, new PermissionResultCallback() {@Overridepublic void onGranted() {startRecordSoundAction(fragment, requestCode);}@Overridepublic void onDenied() {}});}}}/*** 啟動錄音意圖** @param fragment* @param requestCode*/private static void startRecordSoundAction(Fragment fragment, int requestCode) {Intent recordAudioIntent = new Intent(MediaStore.Audio.Media.RECORD_SOUND_ACTION);if (recordAudioIntent.resolveActivity(fragment.requireActivity().getPackageManager()) != null) {fragment.startActivityForResult(recordAudioIntent, requestCode);} else {ToastUtils.showToast(fragment.getContext(), "The system is missing a recording component");}}
}

至于封裝,到此就結束了。最后看一下用法,舉例::

打開相機:

PictureUtils.openCamera(this,false){val localMedia = it[0]
}

選擇X張照片:

PictureUtils.createImageMax(this, 9, arrayListOf<LocalMedia>()){it.forEach { section ->// section.realPath:獲取圖片路徑}
}

就簡單列舉兩個。剩下的都差不多。
使用的時候別忘記添加權限喲。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/166294.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/166294.shtml
英文地址,請注明出處:http://en.pswp.cn/news/166294.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Springboot3+vue3從0到1開發實戰項目(一)

一. 可以在本項目里面自由發揮拓展 二. 知識整合項目使用到的技術 后端開發 &#xff1a; Validation, Mybatis,Redis, Junit,SpringBoot3 &#xff0c;mysql&#xff0c;Swagger, JDK17 &#xff0c;項目部署 前端開發&#xff1a; Vue3&#xff0c;Vite&#xff0c;Router…

Java數組和集合

在Java中&#xff0c;數組和集合是兩個重要的概念&#xff0c;它們用于存儲和操作數據。本文將詳細介紹Java中的數組和集合&#xff0c;包括它們的定義、初始化、訪問和常見操作 一、數組&#xff08;Array&#xff09; 數組是一種用于存儲相同類型數據的容器&#xff0c;它可…

DNS的各種進階新玩法

你們好&#xff0c;我的網工朋友&#xff0c;今天和你聊聊DNS。 01 什么是DNS&#xff1f; mac地址誕生&#xff0c;可是太不容易記憶了&#xff0c;出現了簡化了IP形式&#xff0c;它被直接暴露給外網不說&#xff0c;還讓人類還是覺得比較麻煩&#xff0c;干脆用幾個字母算了…

【Git】一文教你學會 submodule 的增、刪、改、查

添加子模塊 $ git submodule add <url> <path>url 為想要添加的子模塊路徑path 為子模塊存放的本地路徑 示例&#xff0c;添加 r-tinymaix 為子模塊到主倉庫 ./sdk/packages/online-packages/r-tinymaix 路徑下&#xff0c;命令如下所示&#xff1a; $ git subm…

用自己熱愛的事賺錢,是多么的幸福

挖掘天賦可能有些困難&#xff0c;但挖掘愛好就簡單多啦&#xff01;最幸福的事情就是能用自己喜歡的事情賺錢。 我們要說的是一個博主&#xff0c;他非常喜歡騎自行車&#xff0c;雖然他的工作是在外貿公司做銷售&#xff0c;但每當有空時&#xff0c;他都會騎自行車。而且他…

Nginx同時支持Http和Https的配置詳解

當配置Nginx同時支持HTTP和HTTPS時&#xff0c;需要進行以下步驟&#xff1a; 安裝和配置SSL證書&#xff1a; 獲得SSL證書&#xff1a;從可信任的證書頒發機構&#xff08;CA&#xff09;或使用自簽名證書創建SSL證書。 將證書和私鑰保存到服務器&#xff1a;將SSL證書和私鑰…

spring 的事務隔離;Spring框架的事務管理的優點

文章目錄 說一下 spring 的事務隔離&#xff1f;Spring框架的事務管理有哪些優點&#xff1f;你更傾向用哪種事務管理類型&#xff1f; 聊一聊spring事務的隔離&#xff0c;事務的隔離對于一個系統來說也是非常重要的&#xff0c;直接上干貨&#xff01;&#xff01;&#xff0…

Python與設計模式--享元模式

10-Python與設計模式–享元模式 一、網上咖啡選購平臺 假設有一個網上咖啡選購平臺&#xff0c;客戶可以在該平臺上下訂單訂購咖啡&#xff0c;平臺會根據用戶位置進行 線下配送。假設其咖啡對象構造如下&#xff1a; class Coffee:name price 0def __init__(self,name):se…

Go iota簡介

當聲明枚舉類型或定義一組相關常量時&#xff0c;Go語言中的iota關鍵字可以幫助我們簡化代碼并自動生成遞增的值。本文檔將詳細介紹iota的用法和行為。 iota關鍵字 iota是Go語言中的一個預定義標識符&#xff0c;它用于創建自增的無類型整數常量。iota的行為類似于一個計數器…

數據庫基礎入門 — SQL排序與分頁

我是南城余&#xff01;阿里云開發者平臺專家博士證書獲得者&#xff01; 歡迎關注我的博客&#xff01;一同成長&#xff01; 一名從事運維開發的worker&#xff0c;記錄分享學習。 專注于AI&#xff0c;運維開發&#xff0c;windows Linux 系統領域的分享&#xff01; 本…

[深度理解] 重啟 Splunk Search Head Cluster

1: 背景: 關于釋放Splunk search head 的job 運行壓力:splunk search head cluster 要重啟的話,怎么辦? 答案是:splunk rolling-restart shcluster-members Initiate a rolling restart from the command line Invoke the splunk rolling-restart command from any me…

3款免費次數多且功能又強大的國產AI繪畫工具

hi&#xff0c;同學們&#xff0c;本期是我們第55 期 AI工具教程 最近兩個月&#xff0c;國內很多AI繪畫軟件被關停&#xff0c;國外絕大部分AI繪畫工具費用不低&#xff0c;因此 這兩天我 重新整理 國產 AI繪畫 工具 &#xff0c; 最終 篩選了 3款功能強大&#xf…

LeeCode前端算法基礎100題(3)- N皇后

一、問題詳情&#xff1a; 按照國際象棋的規則&#xff0c;皇后可以攻擊與之處在同一行或同一列或同一斜線上的棋子。 n 皇后問題 研究的是如何將 n 個皇后放置在 nn 的棋盤上&#xff0c;并且使皇后彼此之間不能相互攻擊。 給你一個整數 n &#xff0c;返回所有不同的 n 皇后…

虛擬機系列:vmware和Oracle VM VirtualBox虛擬機的區別,簡述哪一個更適合我?以及相互轉換

一. VMware和Oracle VM VirtualBox虛擬機的區別主要體現在以下幾個方面: 首先兩種軟件的安裝使用教程如下: VMware ESXI 安裝使用教程 Oracle VM VirtualBox安裝使用教程 商業模式:VMware是一家商業公司,而Oracle VM VirtualBox是開源軟件; 功能:VMware擁有更多的功能和…

Leetcode200. 島嶼數量

Every day a Leetcode 題目來源&#xff1a;200. 島嶼數量 解法1&#xff1a;深度優先搜索 設目前指針指向一個島嶼中的某一點 (i, j)&#xff0c;尋找包括此點的島嶼邊界。 從 (i, j) 向此點的上下左右 (i1,j)&#xff0c;(i-1,j)&#xff0c;(i,j1)&#xff0c;(i,j-1) …

“圓柱-計算公式“技術支持網址

該軟件可以計算圓柱的底面圓周長、底面積、側面積和體積。 您在使用中有遇到任何問題都可以和我們聯系。我們會在第一時間回復您。 郵箱地址&#xff1a;elmo30zeongmail.com 謝謝&#xff01;

如何將本地websocket發布至公網并實現遠程訪問?

本地websocket服務端暴露至公網訪問【cpolar內網穿透】 文章目錄 本地websocket服務端暴露至公網訪問【cpolar內網穿透】1. Java 服務端demo環境2. 在pom文件引入第三包封裝的netty框架maven坐標3. 創建服務端,以接口模式調用,方便外部調用4. 啟動服務,出現以下信息表示啟動成功…

VR云游:讓旅游產業插上數字化翅膀,打造地方名片

自多地入冬降溫以來&#xff0c;泡溫泉成了許多人周末度假的選擇&#xff0c;在氣溫持續走低的趨勢下&#xff0c;溫泉游也迎來了旺季&#xff1b;但是依舊有些地區溫度依舊溫暖&#xff0c;例如南京的梧桐美景也吸引了不少游客前去打卡&#xff0c;大家穿著漢服與金黃的樹葉合…

【AI考證筆記】NO.1人工智能的基礎概念

以下部分內容來自于百度智能云人才認證培訓講義&#xff0c;騰訊等也有人工智能類似的講義&#xff0c;限時免費&#xff0c;也就是不報考&#xff0c;也能系統學習&#xff0c;課程做的都是不錯的。有感興趣的朋友&#xff0c;可以去檢索學習。 本系列是學習筆記&#xff0c;…

6個常用的聚類評價指標

評估聚類結果的有效性&#xff0c;即聚類評估或驗證&#xff0c;對于聚類應用程序的成功至關重要。它可以確保聚類算法在數據中識別出有意義的聚類&#xff0c;還可以用來確定哪種聚類算法最適合特定的數據集和任務&#xff0c;并調優這些算法的超參數(例如k-means中的聚類數量…