在我之前的文章 “Elasticsearch:Java 运用示例”,我讲述了如何在 Java 应用中创建一个索引,并写入数据。在今天的例子中,我来着重讲述如何有目的地创建按照我们需求的索引,并介绍几种常见的方法写入数据。
我们首先参考如下的文章来安装我们需要的 Elasticsearch 及 Kibana:
针对我们如下的练习,我们的 Elasticsearch 的访问地址为 http://localhost:9200。
此外,针对 Elastic Stack 7.15 及以后的版本,强烈建议对 Elasticsearch 进行安全配置。你可以参考文章 “Elasticsearch:设置 Elastic 账户安全”。我的 Elasticsearch 集群的超级用户 elastic 的密码为 password。
我们用自己喜欢的 IDE 创建一个 Java 应用。在本例中,我将创建一个 Maven 应用:
pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>org.liuxg</groupId>
<artifactId>Elasticsearch-Java</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
<dependency>
<groupId>org.elasticsearch.client</groupId>
<artifactId>elasticsearch-rest-high-level-client</artifactId>
<version>${elastic.version}</version>
</dependency>
<dependency>
<groupId>org.elasticsearch.client</groupId>
<artifactId>elasticsearch-rest-client</artifactId>
<version>${elastic.version}</version>
</dependency>
<dependency>
<groupId>org.elasticsearch</groupId>
<artifactId>elasticsearch</artifactId>
<version>${elastic.version}</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.11.1</version>
</dependency>
</dependencies>
<properties>
<maven.compiler.source>8</maven.compiler.source>
<maven.compiler.target>8</maven.compiler.target>
<elastic.version>7.15.0</elastic.version>
</properties>
</project>
在上面,我们创建一个叫做 ElasticsearchJava 的 class:
ElasticsearchJava.java
import org.apache.http.HttpHost;
import org.apache.http.auth.AuthScope;
import org.apache.http.auth.UsernamePasswordCredentials;
import org.apache.http.impl.client.BasicCredentialsProvider;
import org.apache.http.impl.nio.client.HttpAsyncClientBuilder;
import org.elasticsearch.client.RestClient;
import org.elasticsearch.client.RestClientBuilder;
import org.elasticsearch.client.RestHighLevelClient;
public class ElasticsearchJava {
private static RestHighLevelClient client = null;
private static synchronized RestHighLevelClient makeConnection() {
final BasicCredentialsProvider basicCredentialsProvider = new BasicCredentialsProvider();
basicCredentialsProvider
.setCredentials(AuthScope.ANY, new UsernamePasswordCredentials("elastic", "password"));
if (client == null) {
client = new RestHighLevelClient(
RestClient.builder(new HttpHost("localhost", 9200, "http"))
.setHttpClientConfigCallback(new RestClientBuilder.HttpClientConfigCallback() {
@Override
public HttpAsyncClientBuilder customizeHttpClient(HttpAsyncClientBuilder httpClientBuilder) {
httpClientBuilder.disableAuthCaching();
return httpClientBuilder.setDefaultCredentialsProvider(basicCredentialsProvider);
}
})
);
}
return client;
}
public static void main(String[] args) {
client = makeConnection();
}
}
在上面,我们创建了一个和 Elasticsearch 的连接。请注意我们使用了超级用户 elastic 极其密码。如果我们没有为我们的集群设置密码的话,我们其实可以非常简单地使用如下的代码来进行连接:
private static synchronized RestHighLevelClient makeConnection() {
if(client == null) {
restHighLevelClient = new RestHighLevelClient(
RestClient.builder( new HttpHost("localhost", "9200", "http")));
}
return client;
}
接下来,我们参照 Elastic 官方文档 “Create Index API”,我们使用如下的代码:
public static void main(String[] args) throws IOException {
client = makeConnection();
CreateIndexRequest request = new CreateIndexRequest("employees");
request.settings(Settings.builder()
.put("index.number_of_shards", 1)
.put("index.number_of_replicas", 0)
);
CreateIndexResponse createIndexResponse = client.indices().create(request, RequestOptions.DEFAULT);
System.out.println("response id: " + createIndexResponse.index());
}
}
在上面,我们建立一个连接,并创建一个叫做 employees 的索引。请注意在索引的名字中,我们不可以有大写字母,否则会导致错误。在上面,我们设置 number_of_shards 为1, number_of_replicas 为0。上面代码的输出为:
response id: employees
我们可以使用 Kibana 来检查我们的结果:

从上面的结果中,我们可以看出来 employees 索引已经被成功地创建。
也许你对创建一个索引的 mapping 也感兴趣,那么你可以使用如下的代码来实现:
public static void main(String[] args) throws IOException {
client = makeConnection();
String mappings = "{\n" +
" \"properties\": {\n" +
" \"id\": {\n" +
" \"type\": \"keyword\"\n" +
" },\n" +
" \"name\": {\n" +
" \"type\": \"text\"\n" +
" }\n" + " }\n" +
"}";
System.out.println("mapping is as follows: ");
System.out.println(mappings);
try {
CreateIndexRequest request = new CreateIndexRequest("employees");
request.settings(Settings.builder()
.put("index.number_of_shards", 1)
.put("index.number_of_replicas", 0)
);
request.mapping(mappings, XContentType.JSON);
CreateIndexResponse createIndexResponse = client.indices().create(request, RequestOptions.DEFAULT);
System.out.println("response id: " + createIndexResponse.index());
} catch (Exception e) {
// e.printStackTrace();
}
}
上面代码运行的结果为:
mapping is as follows:
{
"properties": {
"id": {
"type": "keyword"
},
"name": {
"type": "text"
}
}
}
response id: employees
我们可以在 Kibana 中进行查看:
GET employees/_mapping
上面命令的结果为:
{
"employees" : {
"mappings" : {
"properties" : {
"id" : {
"type" : "keyword"
},
"name" : {
"type" : "text"
}
}
}
}
}
上面的代码类似于 Kibana 中如下的命令:
PUT employees
{
"settings": {
"number_of_shards": 1,
"number_of_replicas": 0
},
"mappings": {
"properties": {
"id": {
"type": "keyword"
},
"name": {
"type": "text"
}
}
}
}
上面的命令创建一个叫做 employees 的索引,并对它进行设置和定义 mappings。
接下来,我们参照另外一个文档 “Index API” 来对已经创建的索引进行写入操作:
// Write documents into employees index
IndexRequest request = new IndexRequest("employees");
request.id("1");
String jsonString = "{" +
"\"id\":\"1\"," +
"\"name\":\"liuxg\"" +
"}";
request.source(jsonString, XContentType.JSON);
IndexResponse indexResponse = client.index(request, RequestOptions.DEFAULT);
System.out.println("response id: "+indexResponse.getId());
System.out.println("response name: "+indexResponse.getResult().name());
上面的代码想已经创建的 employees 索引写入文档。其中的文档时以 JSON 形式写入的。当然如果我们之前没有创建 employees 这个索引,上面的 API 也将会自动帮我们生成 employees 这个索引,并把相应的文档写入。当然这个索引的 settings 及 mappings 也许不是我们想要的,而是系统按照默认的方式给出的。
上面的命令类似于在 Kibana 中的如下的命令:
PUT employees/_doc/1
{
"id": "1",
"name": "liuxg"
}
我们重新编译并运行我们的代码:
mapping is as follows:
{
"properties": {
"id": {
"type": "keyword"
},
"name": {
"type": "text"
}
}
}
response id: 1
response name: CREATED
运行完后,我们可以在 Kibana 中通过如下的方式来进行查看:
GET employees/_search
上面的命令显示的结果为:
{
"took" : 7,
"timed_out" : false,
"_shards" : {
"total" : 1,
"successful" : 1,
"skipped" : 0,
"failed" : 0
},
"hits" : {
"total" : {
"value" : 1,
"relation" : "eq"
},
"max_score" : 1.0,
"hits" : [
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "1",
"_score" : 1.0,
"_source" : {
"id" : "1",
"name" : "liuxg"
}
}
]
}
}
我们可以看到文档已经被正确地写入了。
接下来,我们介绍另外一种写入的方法:
// Method 2: Write documents into employees index
Map<String, Object> jsonMap = new HashMap<>();
jsonMap.put("id", "2");
jsonMap.put("name", "Nancy");
IndexRequest indexRequest = new IndexRequest("employees")
.id("2").source(jsonMap);
IndexResponse indexResponse2 = client.index(indexRequest, RequestOptions.DEFAULT);
System.out.println("response id: "+indexResponse2.getId());
System.out.println("response name: "+indexResponse2.getResult().name());
运行代码:
mapping is as follows:
{
"properties": {
"id": {
"type": "keyword"
},
"name": {
"type": "text"
}
}
}
response id: 1
response name: UPDATED
response id: 2
response name: CREATED
我们重新在 Kibana 中进行查看:
GET employees/_search
上面的命令显示的结果为:
{
"took" : 165,
"timed_out" : false,
"_shards" : {
"total" : 1,
"successful" : 1,
"skipped" : 0,
"failed" : 0
},
"hits" : {
"total" : {
"value" : 2,
"relation" : "eq"
},
"max_score" : 1.0,
"hits" : [
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "1",
"_score" : 1.0,
"_source" : {
"id" : "1",
"name" : "liuxg"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "2",
"_score" : 1.0,
"_source" : {
"name" : "Nancy",
"id" : "2"
}
}
]
}
}
我们看到文档 2 已经被正确地写入。
按照官方的文档,我们可以有另外一种写入的方法:
// Method 3: Write documents into employees index
XContentBuilder builder = XContentFactory.jsonBuilder();
builder.startObject();
{
builder.field("id", "3");
builder.field("name", "Jason");
}
builder.endObject();
IndexRequest indexRequest3 = new IndexRequest("employees")
.id("3").source(builder);
IndexResponse indexResponse3 = client.index(indexRequest3, RequestOptions.DEFAULT);
System.out.println("response id: "+indexResponse3.getId());
System.out.println("response name: "+indexResponse3.getResult().name());
还有:
// Method 4: Write documents into employees index
IndexRequest indexRequest4 = new IndexRequest("employees")
.id("4")
.source("id", "4",
"name", "Mark");
IndexResponse indexResponse4 = client.index(indexRequest4, RequestOptions.DEFAULT);
System.out.println("response id: "+indexResponse4.getId());
System.out.println("response name: "+indexResponse4.getResult().name());
运行上面的代码,我们可以在 Kibana 中进行查看:
"hits" : [
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "1",
"_score" : 1.0,
"_source" : {
"id" : "1",
"name" : "liuxg"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "2",
"_score" : 1.0,
"_source" : {
"name" : "Nancy",
"id" : "2"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "3",
"_score" : 1.0,
"_source" : {
"id" : "3",
"name" : "Jason"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "4",
"_score" : 1.0,
"_source" : {
"id" : "4",
"name" : "Mark"
}
}
]
最后,我们创建一个叫做 Employee 的 Java class:
Employee.java
public class Employee {
private String id;
private String name;
public Employee(String id, String name) {
this.id = id;
this.name = name;
}
public String getId() {
return id;
}
public String getName() {
return name;
}
public void setId(String id) {
this.id = id;
}
public void setName(String name) {
this.name = name;
}
}
在上面一定要注意的是要实现 setters 及 getters。
我们接下来使用如下的代码来写入:
// Method 5: Write documents into employees index
Employee employee = new Employee("5", "Martin");
IndexRequest indexRequest5 = new IndexRequest("employees");
indexRequest.id("5");
indexRequest.source(new ObjectMapper().writeValueAsString(employee), XContentType.JSON);
IndexResponse indexResponse5 = client.index(indexRequest, RequestOptions.DEFAULT);
System.out.println("response id: "+indexResponse5.getId());
System.out.println("response name: "+indexResponse5.getResult().name());
重新运行代码,并在 Kibana 中进行查看:
GET employees/_search
我们可以看到如下的结果:
"hits" : [
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "1",
"_score" : 1.0,
"_source" : {
"id" : "1",
"name" : "liuxg"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "2",
"_score" : 1.0,
"_source" : {
"name" : "Nancy",
"id" : "2"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "3",
"_score" : 1.0,
"_source" : {
"id" : "3",
"name" : "Jason"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "4",
"_score" : 1.0,
"_source" : {
"id" : "4",
"name" : "Mark"
}
},
{
"_index" : "employees",
"_type" : "_doc",
"_id" : "5",
"_score" : 1.0,
"_source" : {
"id" : "5",
"name" : "Martin"
}
}
]
为了方便大家学习,我把源码放于 github:https://github.com/liu-xiao-guo/Elasticsearch-java
出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在
如何使用RSpec::Core::RakeTask初始化RSpecRake任务?require'rspec/core/rake_task'RSpec::Core::RakeTask.newdo|t|#whatdoIputinhere?endInitialize函数记录在http://rubydoc.info/github/rspec/rspec-core/RSpec/Core/RakeTask#initialize-instance_method没有很好的记录;它只是说:-(RakeTask)initialize(*args,&task_block)AnewinstanceofRake
好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信
关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
我正在阅读SandiMetz的POODR,并且遇到了一个我不太了解的编码原则。这是代码:classBicycleattr_reader:size,:chain,:tire_sizedefinitialize(args={})@size=args[:size]||1@chain=args[:chain]||2@tire_size=args[:tire_size]||3post_initialize(args)endendclassMountainBike此代码将为其各自的属性输出1,2,3,4,5。我不明白的是查找方法。当一辆山地自行车被实例化时,因为它没有自己的initialize方法