读取 java 属性文件,值一直为空
Reading a java properties file, values keep going to null
我正在尝试从 java 属性文件中读取值,以便我可以在整个测试过程中使用它们。我有读取属性的 BaseTest。当我在读取 BaseTest 中的值后立即打印时,这些值会正确显示。但是当我尝试在扩展 BaseTest 的 test1 中访问它们时,这些值为空。我有一个 java class 像模板一样接受值,没有初始化。这些值由 TestValuesReader class 填充。我怎样才能让这些值在整个 code/not 期间一直保持为 null?
public class test1 extends BaseTest {
String var1 = TestValues.VAR1;
@Test
public void someFunction() throws InterruptedException {
System.out.println(var1);
}
}
public class BaseTest{
public void readValues() throws IOException {
Properties p = new Properties();
InputStream is = new FileInputStream("TestValues.properties");
p.load(is);
TestValuesReader valuesReader = new TestValuesReader();
valuesReader.readStrings(p);
}
@BeforeClass
public void setUp() throws IOException {
readValues();
System.out.println(TestValues.VAR1); //this will give back the correct value, but when called
//in someFunction() it is null
}
}
public class TestValues{
public static String VAR1; //not initialized, supposed to be ready from properties file
}
public class TestValuesReader {
public void readStrings(Properties p) {
TestValues.VAR1 = p.getProperty("VAR1");
}
}
FAILED: checkForCandidate
java.lang.IllegalArgumentException: Keys to send should be a not null CharSequence
at org.openqa.selenium.remote.RemoteWebElement.sendKeys(RemoteWebElement.java:97)
at com.iai.test.pages.SpatialQuery.setAimPointInput(SpatialQuery.java:50)
at com.iai.test.tests.cgm.CG_AX_001.checkForCandidate(CG_AX_001.java:21)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.testng.internal.MethodInvocationHelper.invokeMethod(MethodInvocationHelper.java:134)
at org.testng.internal.TestInvoker.invokeMethod(TestInvoker.java:597)
at org.testng.internal.TestInvoker.invokeTestMethod(TestInvoker.java:173)
at org.testng.internal.MethodRunner.runInSequence(MethodRunner.java:46)
at org.testng.internal.TestInvoker$MethodInvocationAgent.invoke(TestInvoker.java:816)
at org.testng.internal.TestInvoker.invokeTestMethods(TestInvoker.java:146)
at org.testng.internal.TestMethodWorker.invokeTestMethods(TestMethodWorker.java:146)
at org.testng.internal.TestMethodWorker.run(TestMethodWorker.java:128)
at java.util.ArrayList.forEach(ArrayList.java:1257)
at org.testng.TestRunner.privateRun(TestRunner.java:766)
at org.testng.TestRunner.run(TestRunner.java:587)
at org.testng.SuiteRunner.runTest(SuiteRunner.java:384)
at org.testng.SuiteRunner.runSequentially(SuiteRunner.java:378)
at org.testng.SuiteRunner.privateRun(SuiteRunner.java:337)
at org.testng.SuiteRunner.run(SuiteRunner.java:286)
at org.testng.SuiteRunnerWorker.runSuite(SuiteRunnerWorker.java:53)
at org.testng.SuiteRunnerWorker.run(SuiteRunnerWorker.java:96)
at org.testng.TestNG.runSuitesSequentially(TestNG.java:1187)
at org.testng.TestNG.runSuitesLocally(TestNG.java:1109)
at org.testng.TestNG.runSuites(TestNG.java:1039)
at org.testng.TestNG.run(TestNG.java:1007)
at org.testng.remote.AbstractRemoteTestNG.run(AbstractRemoteTestNG.java:115)
at org.testng.remote.RemoteTestNG.initAndRun(RemoteTestNG.java:251)
at org.testng.remote.RemoteTestNG.main(RemoteTestNG.java:77)
您需要 Static Initialization Block or a Constructor 才能调用 super.readValues()
。按如下方式操作:
使用静态初始化块:
public class Test1 extends BaseTest {
String var1;
{
try {
super.readValues();
} catch (IOException e) {
e.printStackTrace();
}
var1 = TestValues.VAR1;
}
@Test
public void someFunction() throws InterruptedException {
System.out.println(var1);
}
}
使用构造函数:
public class Test1 extends BaseTest {
String var1;
Test1() {
try {
super.readValues();
var1 = TestValues.VAR1;
} catch (IOException e) {
e.printStackTrace();
}
}
@Test
public void someFunction() throws InterruptedException {
System.out.println(var1);
}
}
我已经测试了这两种解决方案是否按预期工作。
补充说明:
- 确保您对密钥有一些价值,
VAR1
例如VAR1=x
在属性文件中。
- 我还建议您关注 Java naming conventions 例如根据命名约定,
class test1
应该是 class Test1
。
我正在尝试从 java 属性文件中读取值,以便我可以在整个测试过程中使用它们。我有读取属性的 BaseTest。当我在读取 BaseTest 中的值后立即打印时,这些值会正确显示。但是当我尝试在扩展 BaseTest 的 test1 中访问它们时,这些值为空。我有一个 java class 像模板一样接受值,没有初始化。这些值由 TestValuesReader class 填充。我怎样才能让这些值在整个 code/not 期间一直保持为 null?
public class test1 extends BaseTest {
String var1 = TestValues.VAR1;
@Test
public void someFunction() throws InterruptedException {
System.out.println(var1);
}
}
public class BaseTest{
public void readValues() throws IOException {
Properties p = new Properties();
InputStream is = new FileInputStream("TestValues.properties");
p.load(is);
TestValuesReader valuesReader = new TestValuesReader();
valuesReader.readStrings(p);
}
@BeforeClass
public void setUp() throws IOException {
readValues();
System.out.println(TestValues.VAR1); //this will give back the correct value, but when called
//in someFunction() it is null
}
}
public class TestValues{
public static String VAR1; //not initialized, supposed to be ready from properties file
}
public class TestValuesReader {
public void readStrings(Properties p) {
TestValues.VAR1 = p.getProperty("VAR1");
}
}
FAILED: checkForCandidate
java.lang.IllegalArgumentException: Keys to send should be a not null CharSequence
at org.openqa.selenium.remote.RemoteWebElement.sendKeys(RemoteWebElement.java:97)
at com.iai.test.pages.SpatialQuery.setAimPointInput(SpatialQuery.java:50)
at com.iai.test.tests.cgm.CG_AX_001.checkForCandidate(CG_AX_001.java:21)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.testng.internal.MethodInvocationHelper.invokeMethod(MethodInvocationHelper.java:134)
at org.testng.internal.TestInvoker.invokeMethod(TestInvoker.java:597)
at org.testng.internal.TestInvoker.invokeTestMethod(TestInvoker.java:173)
at org.testng.internal.MethodRunner.runInSequence(MethodRunner.java:46)
at org.testng.internal.TestInvoker$MethodInvocationAgent.invoke(TestInvoker.java:816)
at org.testng.internal.TestInvoker.invokeTestMethods(TestInvoker.java:146)
at org.testng.internal.TestMethodWorker.invokeTestMethods(TestMethodWorker.java:146)
at org.testng.internal.TestMethodWorker.run(TestMethodWorker.java:128)
at java.util.ArrayList.forEach(ArrayList.java:1257)
at org.testng.TestRunner.privateRun(TestRunner.java:766)
at org.testng.TestRunner.run(TestRunner.java:587)
at org.testng.SuiteRunner.runTest(SuiteRunner.java:384)
at org.testng.SuiteRunner.runSequentially(SuiteRunner.java:378)
at org.testng.SuiteRunner.privateRun(SuiteRunner.java:337)
at org.testng.SuiteRunner.run(SuiteRunner.java:286)
at org.testng.SuiteRunnerWorker.runSuite(SuiteRunnerWorker.java:53)
at org.testng.SuiteRunnerWorker.run(SuiteRunnerWorker.java:96)
at org.testng.TestNG.runSuitesSequentially(TestNG.java:1187)
at org.testng.TestNG.runSuitesLocally(TestNG.java:1109)
at org.testng.TestNG.runSuites(TestNG.java:1039)
at org.testng.TestNG.run(TestNG.java:1007)
at org.testng.remote.AbstractRemoteTestNG.run(AbstractRemoteTestNG.java:115)
at org.testng.remote.RemoteTestNG.initAndRun(RemoteTestNG.java:251)
at org.testng.remote.RemoteTestNG.main(RemoteTestNG.java:77)
您需要 Static Initialization Block or a Constructor 才能调用 super.readValues()
。按如下方式操作:
使用静态初始化块:
public class Test1 extends BaseTest {
String var1;
{
try {
super.readValues();
} catch (IOException e) {
e.printStackTrace();
}
var1 = TestValues.VAR1;
}
@Test
public void someFunction() throws InterruptedException {
System.out.println(var1);
}
}
使用构造函数:
public class Test1 extends BaseTest {
String var1;
Test1() {
try {
super.readValues();
var1 = TestValues.VAR1;
} catch (IOException e) {
e.printStackTrace();
}
}
@Test
public void someFunction() throws InterruptedException {
System.out.println(var1);
}
}
我已经测试了这两种解决方案是否按预期工作。
补充说明:
- 确保您对密钥有一些价值,
VAR1
例如VAR1=x
在属性文件中。 - 我还建议您关注 Java naming conventions 例如根据命名约定,
class test1
应该是class Test1
。