概述
Azure云存储是一种由微软提供的云端存储服务
,允许用户通过互联网存储、管理和访问数据。Azure云存储提供了多种存储类型,满足不同的应用场景需求,从简单的文件存储到大规模的非结构化数据处理。它的高可用性、全球覆盖、弹性扩展、数据备份与恢复功能,使其成为企业和开发人员常用的云存储解决方案
主要功能:
- Blob存储:用于存储非结构化数据(如图片、视频、日志文件)。支持三种存储层级(热、冷、归档),根据访问频率优化存储成本。
- 文件存储:通过SMB协议提供共享文件存储服务,支持多用户和跨平台访问。
- 表存储:提供非关系型的键值对存储,适合大规模的数据处理和查询操作。
- 队列存储:提供消息队列服务,支持大规模消息传递,用于异步处理和应用间通信。
优势:
- 全球数据中心:Azure在全球拥有多个数据中心,提供全球范围内的数据访问和冗余。
- 弹性和扩展性:可根据应用需求动态扩展存储容量和性能。
- 安全性和合规性:Azure提供全面的数据加密、访问控制和合规性认证,确保数据安全。
- 成本优化:通过多层存储和自动化的成本管理工具,有效降低存储成本。
集成
配置
1 2 3
| azure: device-position-contain-name: order-test-data
|
配置类
yaml配置,下面的写法非常不安全,建议大家到抽离到配置文件中
1 2 3 4 5 6 7 8 9 10
| @Configuration public class AzureBlobConfig {
@Bean public BlobServiceClient blobServiceClient() { return new BlobServiceClientBuilder() .connectionString("DefaultEndpointsProtocol=https;AccountName=your-accountName;AccountKey=your-accounntKey;EndpointSuffix=core.windows.net") .buildClient(); } }
|
服务类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
|
public interface AzureBlobService {
String getSSAUrlOnlyRead(String containName, String blobName);
String upload(String containName, MultipartFile file);
List<String> upload(String containName, MultipartFile[] file);
List<String> update(String containName, MultipartFile[] file,List<String> fileNames);
void delete(String containName, String fileName); }
|
服务实现类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
| @Slf4j @Service public class AzureBlobServiceImpl implements AzureBlobService {
@Autowired private BlobServiceClient blobServiceClient;
@Override public String getSSAUrlOnlyRead(String containName, String blobName) { OffsetDateTime expiryTime = OffsetDateTime.now().plusYears(99); BlobClient blobClient = blobServiceClient.getBlobContainerClient(containName).getBlobClient(blobName); BlobSasPermission blobSasPermission = new BlobSasPermission().setReadPermission(true); BlobServiceSasSignatureValues serviceSasValues = new BlobServiceSasSignatureValues(expiryTime, blobSasPermission); return blobClient.getBlobUrl() + "?" + blobClient.generateSas(serviceSasValues); }
@Override public List<String> upload(String containName, MultipartFile[] files) { List<String> result = new ArrayList<>(); for (int i = 0; i < files.length; i++) { String blobName = System.currentTimeMillis() + "_" + files[i].getOriginalFilename(); result.add(blobName); BlockBlobClient blockBlobClient = blobServiceClient.getBlobContainerClient(containName).getBlobClient(blobName).getBlockBlobClient(); try (BlobOutputStream blobOS = blockBlobClient.getBlobOutputStream()) { blobOS.write(files[i].getBytes()); } catch (IOException e) { throw new RuntimeException("上传到azure失败"); } } return result; }
@Override public List<String> update(String containName, MultipartFile[] files, List<String> fileNames) { List<String> result = new ArrayList<>(); for (int i = 0; i < files.length; i++) { BlockBlobClient blockBlobClient = blobServiceClient.getBlobContainerClient(containName).getBlobClient(files[i].getOriginalFilename()).getBlockBlobClient(); BlockBlobClient uploadBlockBlobClient; try (ByteArrayInputStream dataStream = new ByteArrayInputStream(files[i].getBytes())) { if (!blockBlobClient.exists()) { String blobName = System.currentTimeMillis() + "_" + files[i].getOriginalFilename(); uploadBlockBlobClient = blobServiceClient.getBlobContainerClient(containName).getBlobClient(blobName).getBlockBlobClient(); result.add(blobName); uploadBlockBlobClient.upload(dataStream, files[i].getSize(), false); } else { if (!fileNames.contains(files[i].getOriginalFilename())) { String blobName = System.currentTimeMillis() + "_" + files[i].getOriginalFilename(); uploadBlockBlobClient = blobServiceClient.getBlobContainerClient(containName).getBlobClient(blobName).getBlockBlobClient(); result.add(blobName); uploadBlockBlobClient.upload(dataStream, files[i].getSize(), false); } else { uploadBlockBlobClient = blockBlobClient; uploadBlockBlobClient.upload(dataStream, files[i].getSize(), true); } } } catch (IOException e) { throw new RuntimeException(files[i].getOriginalFilename() + "更新到azure失败"); } } return result; }
@Override public void delete(String containName, String fileName) { BlockBlobClient blockBlobClient = blobServiceClient.getBlobContainerClient(containName).getBlobClient(fileName).getBlockBlobClient(); if (blockBlobClient.exists()) { blockBlobClient.delete(); } }
public void deleteAllBlobs(String containerName) { BlobContainerClient containerClient = blobServiceClient.getBlobContainerClient(containerName); for (BlobItem blobItem : containerClient.listBlobs()) { BlobClient blobClient = containerClient.getBlobClient(blobItem.getName()); blobClient.delete(); System.out.println("Deleted blob: " + blobItem.getName()); } }
@Override public String upload(String containName, MultipartFile file) { checkContain(containName); String blobName = file.getOriginalFilename(); BlockBlobClient blockBlobClient = blobServiceClient .getBlobContainerClient(containName) .getBlobClient(blobName) .getBlockBlobClient(); try (BlobOutputStream blobOS = blockBlobClient.getBlobOutputStream()) { blobOS.write(file.getBytes()); } catch (IOException e) { throw new RuntimeException("上传到azure失败,msg:" + e.getMessage()); } return blobName; }
private void checkContain(String containName) { BlobContainerClient containerClient = blobServiceClient.getBlobContainerClient(containName); if (!containerClient.exists()) { containerClient.create(); } } }
|
MultipartFile的实现类
其中一个实现类是只有test中能用,那么需要我们自定义一个
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
| public class CustomMultipartFile implements MultipartFile { private final byte[] fileContent; private final String originalFilename; private final String contentType;
public CustomMultipartFile(File file) throws IOException { this.fileContent = java.nio.file.Files.readAllBytes(file.toPath()); this.originalFilename = file.getName(); this.contentType = getMimeType(file.getName()); }
public CustomMultipartFile(String originalFilename, String contentType, byte[] fileContent) { this.fileContent = fileContent; this.originalFilename = originalFilename; this.contentType = contentType != null ? contentType : getMimeType(originalFilename); }
@Override public String getName() { return originalFilename; }
@Override public String getOriginalFilename() { return originalFilename; }
@Override public String getContentType() { return contentType; }
@Override public boolean isEmpty() { return fileContent.length == 0; }
@Override public long getSize() { return fileContent.length; }
@Override public byte[] getBytes() { return fileContent; }
@Override public InputStream getInputStream() { return new ByteArrayInputStream(fileContent); }
@Override public void transferTo(File dest) throws IOException { try (FileOutputStream fos = new FileOutputStream(dest)) { fos.write(fileContent); } }
private String getMimeType(String filename) { if (filename.endsWith(".xlsx")) { return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"; } else if (filename.endsWith(".csv")) { return "text/csv"; } else if (filename.endsWith(".txt")) { return "text/plain"; } else { return "application/octet-stream"; } } }
|
测试方法
在一个控制器调用下面的方法,将exce上传到azure的存储库
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
| private String final BACK_ORDER_LIST = "BACK_ORDER_LIST:%d:%d";
@Value("${azure.device-position-contain-name}") private String devicePositionContainName;
private void testUpload(Long id) { Long userId = getUserId(); Long shopId = getShopId(); try { List<Order> orderList = orderService.getList(); if (orderList.isEmpty()) { return; } ByteArrayOutputStream out = new ByteArrayOutputStream(); String fileName = "a/" + new Date().getTime() + ".xlsx"; log.info("\n== file name : {}==\n", fileName); EasyExcel.write(out, PositionExcelEntity.class).sheet("order").doWrite(orderList); MultipartFile multipartFile = new CustomMultipartFile( fileName, "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", out.toByteArray() ); azureBlobService.upload(devicePositionContainName, multipartFile); String key = String.format(BACK_ORDER, shopId, userId); String val = azureBlobService.getSSAUrlOnlyRead(devicePositionContainName, fileName); redisTemplate.opsForValue().set(key, val); log.info("\n == testUpload file success, file url : {} ==\n", val); } catch (Exception e) { log.error("\n == testUpload file fail, error msg : {} ==\n", e.getMessage()); } finally { log.info("\n = testUpload file end =\n"); } }
|
可视化
安装 Microsoft Azure Storage Explorer
应用查看容器下的存储Bolb等数据