使用腾讯云上传图片并添加水印。
最编程
2024-03-17 10:52:08
...
controller层
@ApiOperation("上传")
@PostMapping("/upload")
//@RequiresRoles("admin")
public Result upload(
@ApiParam(value = "文件", required = true) @RequestParam("file") MultipartFile multipartFile,
@ApiParam(value = "是否加水印:0-不加,1-加")@RequestParam(defaultValue = "0") Integer isWaterMark
) {
Result result = new Result();
//将文件上传到腾讯云
try {
if (isWaterMark == 1) {
//加水印
multipartFile = WaterMarkUtils.addWorkMarkToMutipartFile(multipartFile, "找房快");
}
//上传腾讯云,返回随机文件名字
File file = MultipartFileUtil.multipartFileToFile(multipartFile);
String filename = TengXunOssUtil.uploadFile(file);
filename = TengXunOssUtil.URL_PREFIX + filename;//拼接访问路径和文件名
//删除本地临时文件
MultipartFileUtil.delteTempFile(file);
result.setCode(200);
result.setResult(filename);
result.setMessage("上传成功");
} catch (Exception e) {
log.info(e.getMessage(), e);
result.setCode(500);
result.setMessage("上传失败");
}
return result;
}
图片上传工具
import org.springframework.mock.web.MockMultipartFile;
import org.springframework.web.multipart.MultipartFile;
import javax.imageio.ImageIO;
import javax.imageio.stream.ImageOutputStream;
import javax.swing.*;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;
/**
* @author
* @version 1.0
* @date 2021/1/6 15:40
*/
public class WaterMarkUtils {
/**
* 加图片水印 : 没有测试
*
* @param bufImg --BufferedImage 用来画图的宽高跟需要加水印的图片一样的空白图
* @param img --需要加水印的图片
* @param markImg --水印图片
* @param width --水印图片宽
* @param height --水印图片高
* @param x --水印相对于底片的x轴坐标(PS:左上角为(0,0))
* @param y --水印相对于底片的y轴坐标(PS:左上角为(0,0))
*/
public static void markPic(BufferedImage bufImg, Image img, Image markImg, int width, int height, int x, int y) {
//取到画笔
Graphics2D g = bufImg.createGraphics();
//画底片
g.drawImage(img, 0, 0, bufImg.getWidth(), bufImg.getHeight(), null);
//画水印位置
g.drawImage(markImg, x, y, width, height, null);
g.dispose();
}
/**
* 直接给multipartFile加上图片水印再进行保存图片的操作方便省事
*
* @param multipartFile
* 文件上传的对象
* @param markImg
* 水印文件的路径 如果是相对路径请使用相对路径new Image的方法,此处用的是url
* @return
* @throws IOException
* @author 高永强
* @version 2018年11月30日 上午11:15:56
*/
public static MultipartFile addPicMarkToMutipartFile(MultipartFile multipartFile,
String markImg) throws IOException {
// 获取图片文件名 xxx.png xxx
String originFileName = multipartFile.getOriginalFilename();
// 获取原图片后缀 png
int lastSplit = originFileName.lastIndexOf(".");
String suffix = originFileName.substring(lastSplit + 1);
// 获取图片原始信息
String dOriginFileName = multipartFile.getOriginalFilename();
String dContentType = multipartFile.getContentType();
// 是图片且不是gif才加水印
if (!suffix.equalsIgnoreCase("gif") && dContentType.contains("image")) {
// 获取水印图片
InputStream inputImg = multipartFile.getInputStream();
Image img = ImageIO.read(inputImg);
URL url = new URL(markImg);
// 创建url连接;
HttpURLConnection urlconn = (HttpURLConnection) url.openConnection();
urlconn.connect();
InputStream inputStream = urlconn.getInputStream();
Image mark = ImageIO.read(inputStream);
// 加图片水印
int imgWidth = img.getWidth(null);
int imgHeight = img.getHeight(null);
int markWidth = mark.getWidth(null);
int markHeight = mark.getHeight(null);
BufferedImage bufImg = new BufferedImage(imgWidth, imgHeight,
BufferedImage.TYPE_INT_RGB);
//水印的相对位置 ps:这里是右下角 水印宽为底片的四分之一 位置自己可以调整
markPic(bufImg, img, mark, imgWidth / 4, (imgWidth * markHeight) / (4 * markWidth),
imgWidth - imgWidth / 4, imgHeight - (imgWidth * markHeight) / (4 * markWidth));
ByteArrayOutputStream bs = new ByteArrayOutputStream();
ImageOutputStream imOut = ImageIO.createImageOutputStream(bs);
ImageIO.write(bufImg, suffix, imOut);
InputStream is = new ByteArrayInputStream(bs.toByteArray());
// 加水印后的文件上传
multipartFile = new MockMultipartFile(dOriginFileName, dOriginFileName, dContentType, is);
}
//返回加了水印的上传对象
return multipartFile;
}
/**
* 加文字水印
* @param bufImg --BufferedImage 用来画图的宽高跟需要加水印的图片一样的空白图
* @param img --需要加水印的图片
* @param text --水印文字
* @param font --字体
* @param degree --旋转角度
* @param color --颜色
* @param x --水印相对于底片的x轴坐标(PS:左上角为(0,0))
* @param y --水印相对于底片的y轴坐标(PS:左上角为(0,0))
* @author 高永强
* @version 2018年11月30日 上午11:39:57
*/
public static void markWord(BufferedImage bufImg, Image img, String text, Font font, Color color,Integer degree, int x, int y) {
//取到画笔
Graphics2D g = bufImg.createGraphics();
//画底片
g.drawImage(img, 0, 0, bufImg.getWidth(), bufImg.getHeight(), null);
g.setColor(color);//设置水印颜色
//g.setFont(font);//设置字体.设置字体后台linux系统中无法打印,故注释掉
float alpha = 0.5f;//设置水印透明度 默认为1.0 值越小颜色越浅
g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, alpha));//设置水印文字透明度
if (null != degree) {
g.rotate(Math.toRadians(degree),(double)bufImg.getWidth(),(double)bufImg.getHeight());//设置水印旋转
}
//位置
g.drawString(text, x, y);
JLabel label = new JLabel(text);
FontMetrics metrics = label.getFontMetrics(font);
int width = metrics.stringWidth(label.getText());//文字水印的宽
int srcImgHeight = bufImg.getHeight();
int srcImgWidth = bufImg.getWidth();
int rowsNumber = srcImgHeight/width+srcImgHeight%width;// 图片的高 除以 文字水印的宽 打印的行数(以文字水印的宽为间隔)
int columnsNumber = srcImgWidth/width+srcImgWidth%width;//图片的宽 除以 文字水印的宽 每行打印的列数(以文字水印的宽为间隔)
//防止图片太小而文字水印太长,所以至少打印一次
if(rowsNumber < 1){
rowsNumber = 1;
}
if(columnsNumber < 1){
columnsNumber = 1;
}
for(int j=0;j<rowsNumber;j++){
for(int i=0;i<columnsNumber;i++){
g.drawString(text, i*width + j*width, -i*width + j*width);//画出水印,并设置水印位置
}
}
g.dispose();// 释放资源
}
/**
* 直接给multipartFile加上文字水印再进行保存图片的操作方便省事
*
* @param multipartFile
* 文件上传的对象
* @param markImg
* 水印文件的路径 如果是相对路径请使用相对路径new Image的方法,此处用的是url
* @return
* @throws IOException
* @author 高永强
* @version 2018年11月30日 上午11:15:56
*/
public static MultipartFile addWorkMarkToMutipartFile(MultipartFile multipartFile,
String word) throws IOException {
// 获取图片文件名 xxx.png xxx
String originFileName = multipartFile.getOriginalFilename();
// 获取原图片后缀 png
int lastSplit = originFileName.lastIndexOf(".");
String suffix = originFileName.substring(lastSplit + 1);
// 获取图片原始信息
String dOriginFileName = multipartFile.getOriginalFilename();
String dContentType = multipartFile.getContentType();
// 是图片且不是gif才加水印
if (!suffix.equalsIgnoreCase("gif") && dContentType.contains("image")) {
// 获取水印图片
InputStream inputImg = multipartFile.getInputStream();
Image img = ImageIO.read(inputImg);
// 加图片水印
int imgWidth = img.getWidth(null);
int imgHeight = img.getHeight(null);
BufferedImage bufImg = new BufferedImage(imgWidth, imgHeight,
BufferedImage.TYPE_INT_RGB);
//设置字体
Font font = new Font("宋体", Font.PLAIN, 14);
//调用画文字水印的方法
markWord(bufImg, img, word, font , Color.white,-45, imgWidth/8, imgHeight/8);
ByteArrayOutputStream bs = new ByteArrayOutputStream();
ImageOutputStream imOut = ImageIO.createImageOutputStream(bs);
ImageIO.write(bufImg, suffix, imOut);
InputStream is = new ByteArrayInputStream(bs.toByteArray());
// 加水印后的文件上传
multipartFile = new MockMultipartFile(dOriginFileName, dOriginFileName, dContentType,
is);
}
//返回加了水印的上传对象
return multipartFile;
}
}
MultipartFile 转 File工具
import org.springframework.web.multipart.MultipartFile;
import java.io.*;
/**
* @author 卿通胜
* @version 1.0
* @date 2020/9/11 9:41
*/
public class MultipartFileUtil {
/**
* MultipartFile 转 File
*
* @param file
* @throws Exception
*/
public static File multipartFileToFile(MultipartFile file){
File toFile = null;
if (file.equals("") || file.getSize() <= 0) {
file = null;
} else {
InputStream ins = null;
try {
ins = file.getInputStream();
toFile = new File(file.getOriginalFilename());
inputStreamToFile(ins, toFile);
ins.close();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (ins != null) {
try {
ins.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
return toFile;
}
//获取流文件
private static void inputStreamToFile(InputStream ins, File file) {
OutputStream os = null;
try {
os = new FileOutputStream(file);
int bytesRead = 0;
byte[] buffer = new byte[8192];
while ((bytesRead = ins.read(buffer, 0, 8192)) != -1) {
os.write(buffer, 0, bytesRead);
}
} catch (Exception e) {
e.printStackTrace();
}finally {
//释放资源
if (ins != null) {
try {
ins.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (os != null) {
try {
os.flush();
os.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
/**
* 删除本地临时文件:
* @param file
*/
public static void delteTempFile(File file) {
if (file != null) {
File del = new File(file.toURI());
del.delete();
}
}
}
文件上传依赖
<!--腾讯云oss-->
<dependency>
<groupId>com.qcloud</groupId>
<artifactId>cos_api</artifactId>
<version>5.6.24</version>
</dependency>
腾讯云文件上传
import com.qcloud.cos.COSClient;
import com.qcloud.cos.ClientConfig;
import com.qcloud.cos.auth.BasicCOSCredentials;
import com.qcloud.cos.auth.COSCredentials;
import com.qcloud.cos.exception.CosClientException;
import com.qcloud.cos.exception.CosServiceException;
import com.qcloud.cos.model.*;
import com.qcloud.cos.region.Region;
import lombok.extern.slf4j.Slf4j;
import me.zhyd.oauth.utils.UuidUtils;
import java.io.File;
import java.util.List;
/**
* 腾讯云oss
*
* @author 卿通胜
* @version 1.0
* @date 2020/9/8 15:29
*/
@Slf4j
public class TengXunOssUtil {
public static final String URL_PREFIX = "https://findhouse-1302106834.cos.ap-chongqing.myqcloud.com/";
// 1 初始化用户身份信息(secretId, secretKey)。
private static String secretId = "填写腾讯云cos密钥id";
private static String secretKey = "填写腾讯云cos密钥key";
private static COSCredentials cred = new BasicCOSCredentials(secretId, secretKey);
private static String bucket = "填写存储桶名称"; //存储桶名称,格式:BucketName-APPID
// 2 设置 bucket 的区域, COS 地域的简称请参照 https://cloud.tencent.com/document/product/436/6224
// clientConfig 中包含了设置 region, https(默认 http), 超时, 代理等 set 方法, 使用可参见源码或者常见问题 Java SDK 部分。
private static Region region = new Region("ap-chongqing");
private static ClientConfig clientConfig = new ClientConfig(region);
// 3 生成 cos 客户端。
private static COSClient cosClient = new COSClient(cred, clientConfig);
/**
* 查询存储桶列表
*
* @return
*/
public static List<Bucket> getBuckets() {
List<Bucket> buckets = cosClient.listBuckets();
for (Bucket bucketElement : buckets) {
String bucketName = bucketElement.getName();
String bucketLocation = bucketElement.getLocation();
}
return buckets;
}
/**
* 上传文件
*
* @param localFile
* @return 文件名称
*/
public static String uploadFile(File localFile) {
// 指定要上传到 COS 上对象键
String filename = localFile.getName();
String suffix = filename.substring(filename.lastIndexOf("."));//文件后缀
filename = UuidUtils.getUUID() + suffix;
PutObjectRequest putObjectRequest = new PutObjectRequest(bucket, filename, localFile);
PutObjectResult putObjectResult = cosClient.putObject(putObjectRequest);
return filename;
}
/**
* 删除文件
*
* @param fileName 文件名称
*/
public static void deletefile(String fileName) throws CosClientException, CosServiceException {
// 生成cos客户端
COSClient cosclient = new COSClient(cred, clientConfig);
//处理图片名称
if (fileName.contains(TengXunOssUtil.URL_PREFIX)) {
//如果文件名中包含了url前缀,则去掉
fileName = fileName.replace(TengXunOssUtil.URL_PREFIX, "");
}
// 指定要删除的 bucket 和路径
cosclient.deleteObject(bucket, fileName);
// 关闭客户端(关闭后台线程)
cosclient.shutdown();
}
/**
*
* @describe 上传文件的方法
* @methods uploadObject 方法名
* @param fileUrl 上传文件地址
* @param fileKey 文件对象名称
* @parameter @return 对象列表
*/
public PutObjectResult uploadObject(String fileUrl, String fileKey) {
try {
// 指定要上传的文件
File localFile = new File(fileUrl);
// fileKey 指定要上传到 COS 上对象键
PutObjectRequest putObjectRequest = new PutObjectRequest(bucket, fileKey, localFile);
PutObjectResult putObjectResult = cosClient.putObject(putObjectRequest);
return putObjectResult;
} catch (CosServiceException serverException) {
log.error(serverException.getErrorMessage());
throw new RuntimeException("上传文件平台Server异常"+serverException.getErrorMessage());
} catch (CosClientException clientException) {
log.error(clientException.getMessage());
throw new RuntimeException("上传文件平台Client异常"+clientException.getMessage());
}
}
//测试
/*public static void main(String [] args){
try {
File localFile = new File("F:\\找房快.jpg");
uploadFile(localFile);
}catch (Exception e){
e.printStackTrace();
}
}*/
/**
* 测试删除
* @param args
*/
/*public static void main(String[] args) {
deletefile("clipboard.png");
}*/
}
图片效果
解决上传图片到服务器 水印文字显示框框不显示文字的问题 ,java本地中文正确显示,服务器变成口口口
去此处: https://blog.****.net/lhq186/article/details/98621405
下一篇: 腾讯云 COS 上传文件--JAVA 版
推荐阅读
-
golang]为微信小程序动态生成二维码的实践:golang 为小程序生成二维码图片,并通过 s3 协议上传到对象存储桶 | 腾讯云 cos
-
使用腾讯云上传图片并添加水印。
-
腾讯云对象存储完成图片上传-详细-5.使用 JavaScript SDK
-
腾讯视频直播 02-推流-美颜滤镜 同样,腾讯云提供了 setBeautyFilter 方法来设置美颜风格、磨皮程度、美白程度和泛红程度 //style 磨皮风格:0:平滑 1:自然 2:朦胧 //美容级别:0-9。值为 0 时关闭美颜效果。默认值:0,关闭美颜效果。 //美白级别:取值 0-9。值为 0 时,将关闭美白效果。默认值:0,关闭美白效果。 //ruddyLevel:取值范围为 0-9。值为 0 时关闭美白效果。默认值:0,关闭美白效果。 public boolean setBeautyFilter(int style, int beautyLevel, int whiteningLevel, int ruddyLevel);; public boolean setBeautyFilter(int style, int beautyLevel, int whiteningLevel, int ruddyLevel) 滤镜 setFilter 方法可以设置滤镜效果,滤镜本身是一个直方图文件。setSpecialRatio 方法可以设置滤镜的程度,从 0 到 1,越大滤镜效果越明显,默认值为 0.5。 Bitmap bitmap = BitmapUtils.decodeResource(getResources, R.drawable.langman); if (mLivePusher) if (mLivePusher ! = null) { mLivePusher.setFilter(bmp); } 控制摄像头 腾讯云 sdk 默认为前置摄像头(可以通过修改 TXLivePushConfig 的配置函数 setFrontCamera 来修改默认值),调用一次 switchCamera 就切换一次,注意切换摄像头前要确保 TXLivePushConfig 和 TXLivePusher 对象已经初始化。 mLivePushConfig.setFrontCamera(true); // 默认前置摄像头。 mLivePusher.switchCamera; //切换摄像头。 ⑦ 设置徽标水印 腾讯视频云目前支持两种设置水印的方式:一种是在流媒体 SDK 中设置水印,原理是在 SDK 中对视频进行编码前在画面中设置水印。另一种方式是在云端设置水印,即由云端解析视频并添加水印标识。 建议使用 SDK 添加水印,因为在云端添加水印会有问题。下面是添加水印的 SDK 介绍: //设置视频水印 mLivePushConfig.setWatermark(BitmapFactory.decodeResource(getResources,R.drawable.watermark), 10, 10); // 最后两个参数是视频的水印。 //最后两个参数是水印位置的 X 轴和 Y 轴坐标。 mLivePusher.setConfig(mLivePushConfig); 如果需要对水印图像的位置进行模型适配,则需要调用水印规范化接口。 /设置视频水印 mLivePushConfig.setWatermark(mBitmap, 0.02f, 0.05f, 0.2f); //参数为水印图像。 //参数包括水印图像的位图、水印位置的 X 轴坐标、水印位置的 Y 轴坐标和水印宽度。后三个参数的范围是 [0,1]。 // 最后两个参数是水印位置的 X 轴坐标和 Y 轴坐标。 mLivePusher.setConfig(mLivePushConfig); TXLivePushConfig 中的 setHardwareAcceleration 方法可以启用或禁用硬件编码。 if (mHWVideoEncode){ if (mLivePushConfig ! = null) { if (Build.VERSION.SDK_INT < 18){ Toast.makeText(getApplicationContext, "Hardware acceleration failed, current phone API level is too low (min 18)"、 Toast.LENGTH_SHORT).show; mHWVideoEncode = false; } } } } mLivePushConfig.setHardwareAcceleration(mHWVideoEncode ? TXLiveConstants.ENCODE_VIDEO_HARDWARE : TXLiveConstants.ENCODE_VIDEO_SOFTWARE); mLivePusher.setConfig(mLivePushConfig); // 如果您不确定何时启用硬件加速,建议将其设置为 ENCODE_VIDEO_AUTO。 // 默认情况下启用软件编码,但如果手机的 CPU 使用率超过 80% 或帧速率为 10,SDK 将自动切换到硬件编码。 ⑨ 后台推流 在常规模式下,一旦应用程序进入后台,摄像头捕捉数据的能力就会被 Android 禁用,这意味着 SDK 无法继续捕捉和编码音频和视频数据。如果我们什么都不做,故事就会按照下面的脚本发展: 阶段 1(背景剪切后 10 秒 ->)- CDN 无法将视频流传输给观众,因为没有数据,观众看到的是主帧。 阶段 2(10 秒-> 70 秒)--观众一方的播放器因无法接收到直播流而退出,房间里空无一人。 第 3 阶段(70 秒后)--服务器直接断开了推送流媒体的 RTMP 链接,主播需要重新打开直播才能继续。 主播可能只是短暂地接了一个紧急电话,但各云提供商的安全措施会迫使主播的直播提前结束。 1) 设置 setPauseFlag 在开始推流之前,使用 TXLivePushConfig 的 setPauseImg 接口设置一个等待图像,其含义建议为 "主播将暂时离开,稍后再回来"。
-
如何将图片上传到腾讯云使用tdesign工具