`
gao_20022002
  • 浏览: 164822 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

奇怪而又无可厚非的类加载顺序

    博客分类:
  • Java
阅读更多

昨天看到这样一篇帖子,讨论类加载顺序的,也可以说是初始化顺序的,今天早上找不到了,本来想回复的,现在自己写博客回复了,呵呵

public class Singleton {
private static Singleton obj = new Singleton();
public static int counter1 ;
public static int counter2 = 0;
private Singleton()
{
counter1++;
counter2++;
}
public static Singleton getInstance()
{
return obj;
}
public static void main(String[] args) {
Singleton.getInstance();
System.out.println("obj.counter1=="+counter1);
System.out.println("obj.counter2=="+counter2);
}

}

我不了解为什么obj.counter1=1,obj.counter2=0.希望给我说说 

刚开始也很迷茫,我想不通的问题是什么时候声明的变量count1、count2?

以前的理解它的执行顺序应该是这样的:

1、加载类,当然首先执行的是

private static Singleton obj = new Singleton();

2、所以出现了对中对象的创建,于是执行构造方法:

private Singleton()
{
counter1++;
counter2++;
}

 3、按照顺序,应该执行下一条语句,即:

public static int counter1 ;
public static int counter2 = 0;

好了,类的加载过程完成了。

现在才开始执行main方法的第一条语句,随后的结论都成立了。

现在的问题出现了:什么时候声明的变量count1、count2?

 

查询了一些资料,终于解决了,我认为应该是这样的:

类加载的顺序其实以上的说法是有误的,也就是以前的理解是存在偏差的,除非出现想文章这样的程序才会暴露出来。好了,废话少说,看看真正的类加载顺序:

类加载分为三个过程:装载、链接、初始化。

 

装载的过程就是将class文件读入内存的过程,并且提取其中的类关键信息,比如:方法、变量等等。

 

而在链接中存在三个步骤:

a、进行字节码的检查,看是否符合class文件规范;

b、对类中的类变量进行分配空间,附初始值。此处专指基本类型。

c、对类中的引用变量进行分配空间。

 

随后才进行初始化,现在的初始化才是真正的,将按照语句一句一句执行了。

 

也就是说,在初始化执行以前所有的类变量以及引用变量都是分配了存储空间的,只是他们的数值是不可信任的,也就是系统默认的数据。

现在问题终于搞明白了。

 

所以有这样一个建议,或者说以前有人提醒的问题:

在进行构造方法中不允许进行业务逻辑的处理,只是进行简单的数据初始化,不然会出现意想不到的结果。

 

这个程序说明了这个问题。

如果将

private static Singleton obj = new Singleton();
public static int counter1 ;
public static int counter2 = 0;

 变换为

public static int counter1 ;
public static int counter2 = 0;
private static Singleton obj = new Singleton();

则执行结果则由

obj.counter1=1,obj.counter2=0

变成了

obj.counter1=1,obj.counter2=1.

 

这下明白了吧。

12
1
分享到:
评论
34 楼 okhaoba 2009-07-15  
主要原因是:
public static int counter2 = 0;
并不是一个原子操作,但给人的印象总是声明和初始化同时完成。
33 楼 daoyongyu 2008-11-21  
  不错不错,书中只说静态变量与实例变量的区别和作用,而很小提到两者初始化顺序的对比;

其根本区别在于实例变量是各实例自已拥有和维护的变量,所以在构造前必须对其进行分配内存空间和初始化;而静态变量是各实例共同拥有和维护的,它被预先分配内存,并初始化;

而在楼主的程序中,按照执行顺序,类实例的构造方法首先执行,由于counter1、counter2是静态变量,已经预先存在内存中,所以执行完构告方法后,counter1=1, counter2=1;在接着执行的public  static int counter2 = 0; 中又重新给counter2赋值为0,所以counter2的值为0

====================================
楼上的说的没错!!!!!!
32 楼 jxausea 2008-11-17  
public class Singleton {
//public static int counter1=1 ;
//public static int counter2 = 1;

private static Singleton obj = new Singleton();
public static int counter1 =2;
public static int counter2 = 1;
private Singleton()
{
System.out.println("前counter1="+counter1);
System.out.println("前counter2="+counter2);

counter1++;
counter2++;
System.out.println("后counter1="+counter1);
System.out.println("后counter2="+counter2);

}
public static Singleton getInstance()
{
return obj;
}
public static void main(String[] args) {
System.out.println("--->开始类实例化之前");
Singleton.getInstance();
System.out.println("--->类实例化以后");
System.out.println("obj.counter1=="+counter1);
System.out.println("obj.counter2=="+counter2);
}

}
上述程序运行结果是:
前counter1=0
前counter2=0
后counter1=1
后counter2=1
--->开始类实例化之前
--->类实例化以后
obj.counter1==2
obj.counter2==1
31 楼 jxausea 2008-11-17  
public class Singleton {
public static int counter1=1 ;
public static int counter2 = 1;

private static Singleton obj = new Singleton();
//public static int counter1 =2;
//public static int counter2 = 1;
private Singleton()
{
System.out.println("前counter1="+counter1);
System.out.println("前counter2="+counter2);

counter1++;
counter2++;
System.out.println("后counter1="+counter1);
System.out.println("后counter2="+counter2);

}
public static Singleton getInstance()
{
return obj;
}
public static void main(String[] args) {
System.out.println("--->开始类实例化之前");
Singleton.getInstance();
System.out.println("--->类实例化以后");
System.out.println("obj.counter1=="+counter1);
System.out.println("obj.counter2=="+counter2);
}

}
上述程序运行结果:
前counter2=1
后counter1=2
后counter2=2
--->开始类实例化之前
--->类实例化以后
obj.counter1==2
obj.counter2==2

30 楼 wangdi 2008-11-17  
呵呵,实际的执行顺序是
1、先执行ClassLoadDemo()
2、然后执行counter2 = 0的赋值操作。
也就是说在ClassLoadDemo()执行完成的时候counter2和counter1的值都是1,然后再被赋值为0,如果后来赋值为2,那么结果就变成了counter2 = 2了。
29 楼 luzl 2008-11-15  
gao_20022002 写道

引用说的很对,我觉得这能够很好的解释lz代码中出现的现象。1.静态变量赋值是放在static{}块中实现2.static{}块中的赋值是按照变量声明顺序另外再加二点:3.类只初始化一次静态变量。4.static{}块最先执行在类中。所以:在执行main{}中的Singleton.getInstance();之前obj已经在前面提到的static块中赋值了.也就是在它之前已经执行:A.产生变量counter1,counter2,obj.其中counter1,counter2因为是int型默认0B.赋值:第一步:执行构造函数,使得counter1,counter2都变成.第二步:赋值counter1因为没有指定值所以不执行保留值1. 第三步:counter2 赋值为 0这一步getInstance()再没有调用构造函数,应为静态变量只初始化一次.其实这些解释和以前回复中出现的类似的解释都是正确的。我想告诉大家的问题是着重了解一下虚拟机加载类的过程,在你懂得了这个以后,再出现类似的问题就会有思考的方向,而不是仅仅针对一个问题。现在再次重申一下,类得加载过程:引用类的加载过程分为三步:装载、链接、初始化。装载的过程就是将class文件读入内存的过程,并且提取其中的类关键信息,比如:方法、变量等等。而在链接中存在三个步骤:a、进行字节码的检查,看是否符合class文件规范;b、对类中的类变量进行分配空间,附初始值。此处专指基本类型。c、对类中的引用变量进行分配空间。随后才进行初始化,现在的初始化才是真正的,将按照语句一句一句执行了。也就是说,在初始化执行以前所有的类变量以及引用变量都是分配了存储空间的,只是他们的数值是不可信任的,也就是系统默认的数据。真正明白了这些才是关键。

Good
28 楼 johan 2008-10-24  
这只是类加载中static变量的顺序,就是因为静态变量比较特殊,大家可以看下
public class ClassLoadTest {
	private static ClassLoadTest obj = new ClassLoadTest();
	public  int counter1;
	public  int counter2 = 0;

	private ClassLoadTest() {
		counter1++;
		counter2++;
	}

	public static ClassLoadTest getInstance() {
		return obj;
	}

	public static void main(String[] args) {
		ClassLoadTest cllo = ClassLoadTest.getInstance();
		System.out.println("obj.counter1==" + cllo.counter1);
		System.out.println("obj.counter2==" + cllo.counter2);
	}

}

在跑下结果就看出区别了。
27 楼 gao_20022002 2008-10-24  
引用
说的很对,我觉得这能够很好的解释lz代码中出现的现象。
1.静态变量赋值是放在static{}块中实现
2.static{}块中的赋值是按照变量声明顺序

另外再加二点:
3.类只初始化一次静态变量。
4.static{}块最先执行在类中。

所以:在执行main{}中的Singleton.getInstance();之前obj已经在前面提到的static块中赋值了.也就是在它之前已经执行:
A.产生变量counter1,counter2,obj.其中counter1,counter2因为是int型默认0
B.赋值:第一步:执行构造函数,使得counter1,counter2都变成.第二步:赋值counter1因为没有指定值所以不执行保留值1. 第三步:counter2 赋值为 0

这一步getInstance()再没有调用构造函数,应为静态变量只初始化一次.


其实这些解释和以前回复中出现的类似的解释都是正确的。

我想告诉大家的问题是着重了解一下虚拟机加载类的过程,在你懂得了这个以后,再出现类似的问题就会有思考的方向,而不是仅仅针对一个问题。

现在再次重申一下,类得加载过程:

引用
类的加载过程分为三步:装载、链接、初始化。

装载的过程就是将class文件读入内存的过程,并且提取其中的类关键信息,比如:方法、变量等等。

而在链接中存在三个步骤:

a、进行字节码的检查,看是否符合class文件规范;

b、对类中的类变量进行分配空间,附初始值。此处专指基本类型。

c、对类中的引用变量进行分配空间。

随后才进行初始化,现在的初始化才是真正的,将按照语句一句一句执行了。

也就是说,在初始化执行以前所有的类变量以及引用变量都是分配了存储空间的,只是他们的数值是不可信任的,也就是系统默认的数据。


真正明白了这些才是关键。
26 楼 luzl 2008-10-24  
aspnetking 写道

个人认为静态声明并赋值,赋值会实际体现在static{}语名块中,并且是按声明顺序赋值的。而静态初始化块又会在静态构造函数之后执行,所以会出现这种情况。可以把"public static int counter2 = 0; "向前移两行,可以看出打印结果都是1

说的很对,我觉得这能够很好的解释lz代码中出现的现象。
1.静态变量赋值是放在static{}块中实现
2.static{}块中的赋值是按照变量声明顺序

另外再加二点:
3.类只初始化一次静态变量。
4.static{}块最先执行在类中。

所以:在执行main{}中的Singleton.getInstance();之前obj已经在前面提到的static块中赋值了.也就是在它之前已经执行:
A.产生变量counter1,counter2,obj.其中counter1,counter2因为是int型默认0
B.赋值:第一步:执行构造函数,使得counter1,counter2都变成.第二步:赋值counter1因为没有指定值所以不执行保留值1. 第三步:counter2 赋值为 0

这一步getInstance()再没有调用构造函数,应为静态变量只初始化一次.
25 楼 aspnetking 2008-10-23  
个人认为静态声明并赋值,赋值会实际体现在static{}语名块中,并且是按声明顺序赋值的。而静态初始化块又会在静态构造函数之后执行,所以会出现这种情况。可以把"public static int counter2 = 0; "向前移两行,可以看出打印结果都是1
24 楼 aspnetking 2008-10-23  
通过调试发现:
若将"public static int counter2 = 0"向上移两行,会发现打印值都为1
各人认为情况是这样:
static
23 楼 programmer 2008-10-23  
我认为整个顺序是这样的:
1.加载整个类,把变量都分配空间。这是
counter1=0 
counter2=0
因为这两个变量是基本类型变量,我们独知道基本类型变量加载的时候java会自动赋予一定的值。
2.初始化:
先执行 obj = new Singleton(); 这个语句 这时候会执行构造函数 所以
counter1=1
counter2=1
然后继续往下初始化 但要注意了 这句代码:
public static int counter1 ; 并不执行 因为没有任何的付值动作 ,而是执行:
public static int counter2=0 ;

所以这是:
counter1=1
counter2=0

完全是个人的理解,我觉得楼主这个的主题还是很好的,好多开发人员平时都不考虑这些东西,也让我学了不少东西,希望楼主继续推走这种有深度的主题。我将继续关注
22 楼 gao_20022002 2008-10-23  
引用


原因真的如你所说的吗?我认为原因是在public  static int counter2 = 0;这个地方又重新赋值了而已。

你说的是对的,我把这个类加了两个打印语句就能清楚地看到结果。


其实是这样的。

但是都忘了关注类是如何加载的,了解了类加载机制会更容易接受一点。慢慢体会,不要只看到表面。
一切从规范定义开始。
21 楼 programmer 2008-10-23  
dlovek 写道

原因真的如你所说的吗?我认为原因是在public  static int counter2 = 0;这个地方又重新赋值了而已。

你说的是对的,我把这个类加了两个打印语句就能清楚地看到结果。
public class Singleton { 
private static Singleton obj = new Singleton(); 
public static int counter1 ; 
public static int counter2 = 0; 
private Singleton(){ 
counter1++; 
counter2++;
//counter1=counter1+1;
//counter2=counter2+1;
System.out.println("counter1="+counter1);
System.out.println("counter2="+counter2);

public static Singleton getInstance(){ 
return obj; 

public static void main(String[] args) { 
Singleton.getInstance(); 
System.out.println("obj.counter1="+counter1); 
System.out.println("obj.counter2="+counter2); 
}

20 楼 programmer 2008-10-23  
我还是不明白
19 楼 icewubin 2008-10-23  
建议大家不要使用饿汉式初始化,如果要用的话,要确保初始化的代码在所有定义类变量的后面。

推荐类加载方式的延迟初始化的单例模式:
public class Singleton {
	private static class Holder {
		private static final Singleton instance = new Singleton();
	}
	public static Singleton getInstance() {
		return Holder.instance;
	}
}
18 楼 gao_20022002 2008-10-23  
引用
这个帖子以前有人发过,说是所有程序员都会犯的错误。


我也是新来javaeye的,对于这个就不知道了。

只要没看过的人有收获就好,看到过的权当复习了,顺便对有误解的地方提出指正,谢谢。
17 楼 harrison2010 2008-10-22  
这个帖子以前有人发过,说是所有程序员都会犯的错误。
16 楼 gao_20022002 2008-10-22  
引用


这个结果是一定发生的吗?
我感觉,不一定每次执行多少一样的结果。
因为jvm可以对非static得属性,可能进行优化,也就是说,初始化过程,不一定是按照顺序执行的,jvm会认为调换顺序对执行结果没有影响(虽然有时的确有影响,所以我们写代码的时候要注意)。

可能我们在平时测试不出来,但是如果拿到生产环境,这些问题可能就暴露了。


jvm对于非static属性的优化,这点还没有看到相关的说明或者文章,也不知道具体会是怎样。

现在在看jvm规范,还没有看到相关的问题说明,但是我相信类加载的策略是不会变化的,即使优化也不会改变执行顺序,只是实现方式或者执行效率问题。
就像文章中说的:类加载的三个过程装载、链接、初始化这个我想是不会变的,变的只是实现方式吧。

无论什么时候,像我们的单线程程序,我认为都是顺序执行的,当然初始化也是同样。

顺便说一下:也没有足够的测试,假如按照初始化策略实现方式不会变动的思路推下来的话,执行结果应该不会变化,在我的所有测试过程中还是很稳定的结果。

希望有了解这方面的人介绍介绍,我也没有充足的理由。
谢谢。
15 楼 zxming12345 2008-10-22  
dlovek 写道

gao_20022002 写道引用gao_20022002 写道引用原因真的如你所说的吗?我认为原因是在public  static int counter2 = 0;这个地方又重新赋值了而已。重新赋值?这就有点不可思议了,这个语句是重新赋值:public  static int counter2 = 0;这个主要的用途是声明一个变量吧,赋值只是为了说明与count1的区别。其实你没有看明白我的问题:在new时我怎么得到变量count1、 count2?不,因为counter1为默认值,而counter2却又被重新赋值过了。你可以将publi static int counter1;以及publi static int counter2 = 0;中的static去掉,将他们变成一般类变量,而不是静态的,再将刚才的那个Singleton obj = Singleton.getInstance()添加上,看看结果。这也是一个static与普通变量的区别。为什么会这样?去掉static后,couter1和counter2为实例属性,在调用构造之前会首先为实例属性分配内存空间,赋默认值,最后赋指定值,这些都完成之后才会执行构造内的语句。而static的counter1和counter2为类属性,它们的初始化是按照语句顺序执行的,所以调用构造之后,又会执行public static int counter2=0,这次会覆盖掉曾在构造方法内的值。不知道说明白没有。


这个结果是一定发生的吗?
我感觉,不一定每次执行多少一样的结果。
因为jvm可以对非static得属性,可能进行优化,也就是说,初始化过程,不一定是按照顺序执行的,jvm会认为调换顺序对执行结果没有影响(虽然有时的确有影响,所以我们写代码的时候要注意)。

可能我们在平时测试不出来,但是如果拿到生产环境,这些问题可能就暴露了。

相关推荐

    java全大撒大撒大苏打

    sdad

    (175820822)基于java的工资管理系统设计与实现

    本课程设计是某公司的工资管理系统。在这个计算机快速发展的世界里,计算机为信息处理提供了物美价廉的手段,对于推动我国管理信息处理现代化起到了重要作用。工资管理是一项琐碎、复杂而又十分细致的工作,工资计算、发放、核算的工作量很大,一般不允许出错,如果实行手工操作,每月发放工资须手工填制大量的表格,这就会耗费工作人员大量的时间和精力,计算机进行工资发放工作,不仅能够保证工资核算准确无误、快速输出,而且还可以利用计算机对有关工资的各种信息进行统计,既方便又快捷地完成员工工资的发放。 本课程设计过程中根据设计中的需求及对工资管理系统采用了模块化的设计思想,在机房我们在Windows XP 操作系统环境下,采用 myeclipse7作为开发工具,主要连接 Access 数据库来实现公司的工资管理系统的主要功能。在设计过程中,我们首先小组首先对整体的思路进行分析,然后进行分工。对数据库和类进行设计,实现了工资管理系统的功能。其功能主要包括公司用户管理、人员管理、部门管理、工资管理等功能.。内容来源于网络分享,如有侵权请联系我删除。另外如果没有积分的同学需要下载,请私信我。

    YOLO算法-水泥路面裂纹检测数据集-5005张图像带标签-裂纹.zip

    YOLO系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,包含数据集配置文件data.yaml,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中,文件名末尾是部分类别名称; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值; 【注】可以下拉页面,在资源详情处查看标签具体内容;

    基于鸟鸣声识别的鸟类分类系统项目源代码全套技术资料.zip

    基于鸟鸣声识别的鸟类分类系统项目源代码全套技术资料.zip

    zigbee CC2530无线自组网协议栈系统代码实现协议捕捉与数据分析.zip

    1、嵌入式物联网单片机项目开发例程,简单、方便、好用,节省开发时间。 2、代码使用IAR软件开发,当前在CC2530上运行,如果是其他型号芯片,请自行移植。 3、软件下载时,请注意接上硬件,并确认烧录器连接正常。 4、有偿指导v:wulianjishu666; 5、如果接入其他传感器,请查看账号发布的其他资料。 6、单片机与模块的接线,在代码当中均有定义,请自行对照。 7、若硬件有差异,请根据自身情况调整代码,程序仅供参考学习。 8、代码有注释说明,请耐心阅读。 9、例程具有一定专业性,非专业人士请谨慎操作。

    毕业设计前后端分离博客项目源代码.zip

    毕业设计前后端分离博客项目源代码.zip

    (170644008)Eclipse+MySql+JavaSwing选课成绩管理系统

    Eclipse+MySql+JavaSwing选课成绩管理系统,原文博客在https://blog.csdn.net/qq_50062694/article/details/124649345?spm=1001.2014.3001.5502。内容来源于网络分享,如有侵权请联系我删除。另外如果没有积分的同学需要下载,请私信我。

    IBM Process Mining流程挖掘

    什么是流程挖掘?为什么需要流程挖掘?流程挖掘面向的部门是哪些?流程挖掘之后做什么?如果想知道这些,请阅读此文。

    Android程序开发初级教程WORD文档doc格式最新版本

    ### Android程序开发初级教程(一):初识Android **平台概述** Google推出的Android操作系统平台已经正式亮相,这是一个基于Linux内核的开源操作系统。对于开发者而言,了解其架构和支持的开发语言至关重要。以下是Android平台的架构概览: **平台架构及功能** 1. **应用框架(Application Framework)**:包含可重用和可替换的组件,确保所有软件在该层面上的平等性。 2. **Dalvik虚拟机(Dalvik Virtual Machine)**:一个基于Linux的虚拟机,为Android应用提供运行环境。 3. **集成浏览器(Integrated Browser)**:基于开源WebKit引擎的浏览器,位于应用层。 4. **优化图形(Optimized Graphics)**:包括自定义的2D图形库和遵循OpenGL ES 1.0标准的3D实现。 5. **SQLite数据库**:用于数据存储。 6. **多媒体支持(Media Support)**:支持通用音频、视频以及多种图片格式(如MPEG4, H.264

    java毕设项目之ssm小型企业办公自动化系统的设计和开发+vue(完整前后端+说明文档+mysql+lw).zip

    项目包含完整前后端源码和数据库文件 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7

    212) Outgrid - 多用途 Elementor WordPress 主题 v2.0.0.zip

    212) Outgrid - 多用途 Elementor WordPress 主题 v2.0.0.zip

    weixin138社区互助养老+ssm(论文+源码)-kaic.zip

    weixin138社区互助养老+ssm(论文+源码)_kaic.zip

    深圳建筑安装公司“高处作业安全技术操作规程”.docx

    深圳建筑安装公司“高处作业安全技术操作规程”

    计算机视觉项目:Swin-Transformer 【tiny、small、base】模型实现的图像识别项目:番茄病害图像分类

    【项目简介】 代码主干网络采用Swin-Transformer 家族系列,包括【tiny、small、base】三种模型。pretrained和freeze_layers参数为是否采用官方预训练模型和是否仅训练分类头。为了做对比消融试验,优化器采用了Adam和SGD、AdamW三种。损失函数采用多类别的交叉熵、学习率优化策略采用cos余弦退火算法 【评估网络】 评估的指标采用loss和准确率(accuracy),分别会在训练集和验证集上进行评估、输出、绘制曲线图像。同时会在训练集、验证集进行一系列评估,包含混淆矩阵、recall、precision、F1 score等等曲线图像,以及recall、precision、F1 score、特异度的输出信息等等。 【具体各类别的指标在json文件中查看】 【如果想要更换数据集训练,参考readme文件】 【本项目为8种番茄病害图片(约4k张数据),包含数据集和标签,可以一键运行】

    城市公交查询-java-基于springBoot的城市公交查询系统设计与实现(毕业论文)

    城市公交查询功能描述 城市公交查询系统的主要目的是为市民提供便捷的公交信息查询服务,帮助用户快速获取公交线路、站点、时刻表等信息,从而提高出行效率。以下是该系统可能具备的功能描述: 1. 公交线路查询 线路搜索:用户可以通过输入公交线路编号或线路名称,快速查询到该线路的详细信息。 线路详情:展示所选线路的起点、终点、途经站点、首末班车时间、发车间隔等信息。 线路图展示:提供线路的可视化地图,显示线路走向及各个站点位置。 2. 站点查询 站点搜索:用户可以通过输入站点名称或编号,查询该站点的相关信息。 站点详情:展示所选站点的上下车线路、周边设施、换乘信息等。 实时到站信息:提供该站点即将到达的公交车信息,包括预计到达时间和车牌号。 3. 实时公交信息 实时位置追踪:用户可以查看公交车的实时位置,了解公交车的行驶状态。 到站预测:根据实时数据,预测公交车到达各个站点的时间,帮助用户合理安排出行。 4. 换乘查询 换乘方案推荐:用户输入起点和终点后,系统提供最佳的换乘方案,包括所需的公交线路、换乘站点及步行距离。 换乘时间估算:计算并展示换乘所需的总时间,包括等车时间和步行时间。 5.

    交通旅游订票-JAVA-基于spring boot的交通旅游订票系统设计与实现(毕业论文)

    交通旅游订票功能描述 交通旅游订票系统是为了简化旅游出行过程,提升用户的预定体验。该系统通常集成了机票、火车票、汽车票、船票、景区门票等多种交通和旅游产品的预订、支付及管理功能。以下是该系统可能具备的功能描述: 1. 用户管理 用户注册与登录:提供游客注册与登录功能,支持邮箱、手机号等多种方式注册,保证用户信息安全。 个人信息管理:用户可以查看和编辑个人信息,如身份证号、联系方式、常用地址等。 乘客信息保存:可保存常用乘客信息,如身份证、护照、儿童票信息,方便快速预定。 2. 交通票务管理 票务查询:提供交通工具的实时查询功能,支持机票、火车票、汽车票、船票等的查询,包含出发时间、到达时间、票价、座位情况等信息。 多种票务类型支持:支持单程票、往返票、联程票、团体票等多种票种,满足不同用户需求。 票价比较:根据日期、交通工具等条件,自动比较票价,帮助用户选择最合适的票务。 票务预订与支付:提供便捷的在线预订和支付功能,支持多种支付方式(如银行卡、支付宝、微信等)。 票务改签与退票:用户可以在线申请改签和退票,并查看相关费用及政策。 3. 旅游产品预订 景点门票预订:用户可以在线选择

    企业数据管理系统项目源代码.zip

    企业数据管理系统项目源代码.zip

    java毕设项目之ssm高校专业信息管理系统设计与实现+jsp(完整前后端+说明文档+mysql+lw).zip

    项目包含完整前后端源码和数据库文件 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7

    基于plc的污水处理,组态王动画仿真,带PLC源代码,组态王源代码,图纸,IO地址分配

    基于plc的污水处理,组态王动画仿真,带PLC源代码,组态王源代码,图纸,IO地址分配

    SINAMICS S120驱动第三方直线永磁同步电机系列视频-配置和优化.mp4

    SINAMICS S120驱动第三方直线永磁同步电机系列视频_配置和优化.mp4

Global site tag (gtag.js) - Google Analytics